2021-09-18 20:25:59 +08:00
|
|
|
package asynqmon
|
2020-11-24 22:54:00 +08:00
|
|
|
|
|
|
|
import (
|
|
|
|
"time"
|
2021-06-02 11:37:18 +08:00
|
|
|
"unicode"
|
|
|
|
"unicode/utf8"
|
2021-09-18 22:23:42 +08:00
|
|
|
|
|
|
|
"github.com/hibiken/asynq"
|
2020-11-24 22:54:00 +08:00
|
|
|
)
|
|
|
|
|
2020-12-02 23:19:06 +08:00
|
|
|
// ****************************************************************************
|
|
|
|
// This file defines:
|
|
|
|
// - internal types with JSON struct tags
|
|
|
|
// - conversion function from an external type to an internal type
|
|
|
|
// ****************************************************************************
|
|
|
|
|
2021-10-01 02:09:41 +08:00
|
|
|
// BytesStringer can be used to convert payload bytes to string to show in web ui.
|
|
|
|
type BytesStringer interface {
|
2021-09-18 20:25:59 +08:00
|
|
|
String([]byte) string
|
|
|
|
}
|
|
|
|
|
2021-10-01 02:09:41 +08:00
|
|
|
type BytesStringerFunc func([]byte) string
|
2021-09-18 20:25:59 +08:00
|
|
|
|
2021-10-01 02:09:41 +08:00
|
|
|
func (f BytesStringerFunc) String(b []byte) string {
|
2021-09-18 20:25:59 +08:00
|
|
|
return f(b)
|
|
|
|
}
|
|
|
|
|
2021-10-01 02:09:41 +08:00
|
|
|
var defaultBytesStringer = BytesStringerFunc(func(payload []byte) string {
|
2021-09-18 20:25:59 +08:00
|
|
|
if !isPrintable(payload) {
|
|
|
|
return "non-printable bytes"
|
|
|
|
}
|
|
|
|
return string(payload)
|
|
|
|
})
|
|
|
|
|
2021-09-18 20:26:00 +08:00
|
|
|
// isPrintable reports whether the given data is comprised of all printable runes.
|
|
|
|
func isPrintable(data []byte) bool {
|
|
|
|
if !utf8.Valid(data) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
isAllSpace := true
|
|
|
|
for _, r := range string(data) {
|
|
|
|
if !unicode.IsPrint(r) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
if !unicode.IsSpace(r) {
|
|
|
|
isAllSpace = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return !isAllSpace
|
|
|
|
}
|
|
|
|
|
2020-11-24 22:54:00 +08:00
|
|
|
type QueueStateSnapshot struct {
|
|
|
|
// Name of the queue.
|
|
|
|
Queue string `json:"queue"`
|
2021-01-27 07:56:45 +08:00
|
|
|
// Total number of bytes the queue and its tasks require to be stored in redis.
|
|
|
|
MemoryUsage int64 `json:"memory_usage_bytes"`
|
2020-11-24 22:54:00 +08:00
|
|
|
// Total number of tasks in the queue.
|
|
|
|
Size int `json:"size"`
|
|
|
|
// Number of tasks in each state.
|
|
|
|
Active int `json:"active"`
|
|
|
|
Pending int `json:"pending"`
|
|
|
|
Scheduled int `json:"scheduled"`
|
|
|
|
Retry int `json:"retry"`
|
2021-01-13 03:59:44 +08:00
|
|
|
Archived int `json:"archived"`
|
2020-11-24 22:54:00 +08:00
|
|
|
|
|
|
|
// Total number of tasks processed during the given date.
|
|
|
|
// The number includes both succeeded and failed tasks.
|
|
|
|
Processed int `json:"processed"`
|
|
|
|
// Breakdown of processed tasks.
|
|
|
|
Succeeded int `json:"succeeded"`
|
|
|
|
Failed int `json:"failed"`
|
|
|
|
// Paused indicates whether the queue is paused.
|
|
|
|
// If true, tasks in the queue will not be processed.
|
|
|
|
Paused bool `json:"paused"`
|
|
|
|
// Time when this snapshot was taken.
|
|
|
|
Timestamp time.Time `json:"timestamp"`
|
|
|
|
}
|
|
|
|
|
2021-10-01 02:49:41 +08:00
|
|
|
func toQueueStateSnapshot(s *asynq.QueueInfo) *QueueStateSnapshot {
|
2020-11-24 22:54:00 +08:00
|
|
|
return &QueueStateSnapshot{
|
2021-01-27 07:56:45 +08:00
|
|
|
Queue: s.Queue,
|
|
|
|
MemoryUsage: s.MemoryUsage,
|
|
|
|
Size: s.Size,
|
|
|
|
Active: s.Active,
|
|
|
|
Pending: s.Pending,
|
|
|
|
Scheduled: s.Scheduled,
|
|
|
|
Retry: s.Retry,
|
|
|
|
Archived: s.Archived,
|
|
|
|
Processed: s.Processed,
|
|
|
|
Succeeded: s.Processed - s.Failed,
|
|
|
|
Failed: s.Failed,
|
|
|
|
Paused: s.Paused,
|
|
|
|
Timestamp: s.Timestamp,
|
2020-11-24 22:54:00 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type DailyStats struct {
|
2020-12-28 08:45:28 +08:00
|
|
|
Queue string `json:"queue"`
|
|
|
|
Processed int `json:"processed"`
|
|
|
|
Succeeded int `json:"succeeded"`
|
|
|
|
Failed int `json:"failed"`
|
|
|
|
Date string `json:"date"`
|
2020-11-24 22:54:00 +08:00
|
|
|
}
|
|
|
|
|
2021-10-01 02:49:41 +08:00
|
|
|
func toDailyStats(s *asynq.DailyStats) *DailyStats {
|
2020-11-24 22:54:00 +08:00
|
|
|
return &DailyStats{
|
|
|
|
Queue: s.Queue,
|
|
|
|
Processed: s.Processed,
|
|
|
|
Succeeded: s.Processed - s.Failed,
|
|
|
|
Failed: s.Failed,
|
2020-12-28 08:45:28 +08:00
|
|
|
Date: s.Date.Format("2006-01-02"),
|
2020-11-24 22:54:00 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-01 02:49:41 +08:00
|
|
|
func toDailyStatsList(in []*asynq.DailyStats) []*DailyStats {
|
2020-12-28 07:45:54 +08:00
|
|
|
out := make([]*DailyStats, len(in))
|
|
|
|
for i, s := range in {
|
2021-10-01 02:49:41 +08:00
|
|
|
out[i] = toDailyStats(s)
|
2020-12-28 07:45:54 +08:00
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
2021-07-30 20:53:14 +08:00
|
|
|
type TaskInfo struct {
|
|
|
|
// ID is the identifier of the task.
|
|
|
|
ID string `json:"id"`
|
|
|
|
// Queue is the name of the queue in which the task belongs.
|
|
|
|
Queue string `json:"queue"`
|
|
|
|
// Type is the type name of the task.
|
|
|
|
Type string `json:"type"`
|
|
|
|
// Payload is the payload data of the task.
|
|
|
|
Payload string `json:"payload"`
|
|
|
|
// State indicates the task state.
|
|
|
|
State string `json:"state"`
|
|
|
|
// MaxRetry is the maximum number of times the task can be retried.
|
|
|
|
MaxRetry int `json:"max_retry"`
|
|
|
|
// Retried is the number of times the task has retried so far.
|
|
|
|
Retried int `json:"retried"`
|
|
|
|
// LastErr is the error message from the last failure.
|
|
|
|
LastErr string `json:"error_message"`
|
|
|
|
// LastFailedAt is the time time of the last failure in RFC3339 format.
|
|
|
|
// If the task has no failures, empty string.
|
|
|
|
LastFailedAt string `json:"last_failed_at"`
|
|
|
|
// Timeout is the number of seconds the task can be processed by Handler before being retried.
|
|
|
|
Timeout int `json:"timeout_seconds"`
|
|
|
|
// Deadline is the deadline for the task in RFC3339 format. If not set, empty string.
|
|
|
|
Deadline string `json:"deadline"`
|
|
|
|
// NextProcessAt is the time the task is scheduled to be processed in RFC3339 format.
|
|
|
|
// If not applicable, empty string.
|
|
|
|
NextProcessAt string `json:"next_process_at"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// formatTimeInRFC3339 formats t in RFC3339 if the value is non-zero.
|
|
|
|
// If t is zero time (i.e. time.Time{}), returns empty string
|
|
|
|
func formatTimeInRFC3339(t time.Time) string {
|
|
|
|
if t.IsZero() {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
return t.Format(time.RFC3339)
|
|
|
|
}
|
|
|
|
|
2021-10-01 02:49:41 +08:00
|
|
|
func toTaskInfo(info *asynq.TaskInfo, bs BytesStringer) *TaskInfo {
|
2021-07-30 20:53:14 +08:00
|
|
|
return &TaskInfo{
|
|
|
|
ID: info.ID,
|
|
|
|
Queue: info.Queue,
|
|
|
|
Type: info.Type,
|
2021-10-01 02:49:41 +08:00
|
|
|
Payload: bs.String(info.Payload),
|
2021-07-30 20:53:14 +08:00
|
|
|
State: info.State.String(),
|
|
|
|
MaxRetry: info.MaxRetry,
|
|
|
|
Retried: info.Retried,
|
|
|
|
LastErr: info.LastErr,
|
|
|
|
LastFailedAt: formatTimeInRFC3339(info.LastFailedAt),
|
|
|
|
Timeout: int(info.Timeout.Seconds()),
|
|
|
|
Deadline: formatTimeInRFC3339(info.Deadline),
|
|
|
|
NextProcessAt: formatTimeInRFC3339(info.NextProcessAt),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-24 22:54:00 +08:00
|
|
|
type BaseTask struct {
|
2021-04-05 06:15:54 +08:00
|
|
|
ID string `json:"id"`
|
|
|
|
Type string `json:"type"`
|
2021-06-02 11:37:18 +08:00
|
|
|
Payload string `json:"payload"`
|
2021-04-05 06:15:54 +08:00
|
|
|
Queue string `json:"queue"`
|
|
|
|
MaxRetry int `json:"max_retry"`
|
|
|
|
Retried int `json:"retried"`
|
|
|
|
LastError string `json:"error_message"`
|
2020-11-24 22:54:00 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
type ActiveTask struct {
|
|
|
|
*BaseTask
|
2021-01-24 04:06:50 +08:00
|
|
|
|
|
|
|
// Started time indicates when a worker started working on ths task.
|
|
|
|
//
|
|
|
|
// Value is either time formatted in RFC3339 format, or "-" which indicates
|
|
|
|
// a worker started working on the task only a few moments ago, and started time
|
|
|
|
// data is not available.
|
|
|
|
Started string `json:"start_time"`
|
2021-01-28 08:16:38 +08:00
|
|
|
|
|
|
|
// Deadline indicates the time by which the worker needs to finish its task.
|
|
|
|
//
|
|
|
|
// Value is either time formatted in RFC3339 format, or "-" which indicates that
|
|
|
|
// the data is not available yet.
|
|
|
|
Deadline string `json:"deadline"`
|
2020-11-24 22:54:00 +08:00
|
|
|
}
|
|
|
|
|
2021-10-01 02:49:41 +08:00
|
|
|
func toActiveTask(ti *asynq.TaskInfo, bs BytesStringer) *ActiveTask {
|
2020-11-24 22:54:00 +08:00
|
|
|
base := &BaseTask{
|
2021-09-18 20:26:00 +08:00
|
|
|
ID: ti.ID,
|
|
|
|
Type: ti.Type,
|
2021-10-01 02:49:41 +08:00
|
|
|
Payload: bs.String(ti.Payload),
|
2021-09-18 20:26:00 +08:00
|
|
|
Queue: ti.Queue,
|
|
|
|
MaxRetry: ti.MaxRetry,
|
|
|
|
Retried: ti.Retried,
|
|
|
|
LastError: ti.LastErr,
|
2020-11-24 22:54:00 +08:00
|
|
|
}
|
2021-01-24 04:06:50 +08:00
|
|
|
return &ActiveTask{BaseTask: base}
|
2020-11-24 22:54:00 +08:00
|
|
|
}
|
|
|
|
|
2021-10-01 02:49:41 +08:00
|
|
|
func toActiveTasks(in []*asynq.TaskInfo, bs BytesStringer) []*ActiveTask {
|
2020-11-24 22:54:00 +08:00
|
|
|
out := make([]*ActiveTask, len(in))
|
2021-09-18 20:26:00 +08:00
|
|
|
for i, ti := range in {
|
2021-10-01 02:49:41 +08:00
|
|
|
out[i] = toActiveTask(ti, bs)
|
2020-11-24 22:54:00 +08:00
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
2021-05-29 05:40:09 +08:00
|
|
|
// TODO: Maybe we don't need state specific type, just use TaskInfo
|
2020-11-24 22:54:00 +08:00
|
|
|
type PendingTask struct {
|
|
|
|
*BaseTask
|
|
|
|
}
|
|
|
|
|
2021-10-01 02:49:41 +08:00
|
|
|
func toPendingTask(ti *asynq.TaskInfo, bs BytesStringer) *PendingTask {
|
2020-11-24 22:54:00 +08:00
|
|
|
base := &BaseTask{
|
2021-09-18 20:26:00 +08:00
|
|
|
ID: ti.ID,
|
|
|
|
Type: ti.Type,
|
2021-10-01 02:49:41 +08:00
|
|
|
Payload: bs.String(ti.Payload),
|
2021-09-18 20:26:00 +08:00
|
|
|
Queue: ti.Queue,
|
|
|
|
MaxRetry: ti.MaxRetry,
|
|
|
|
Retried: ti.Retried,
|
|
|
|
LastError: ti.LastErr,
|
2020-11-24 22:54:00 +08:00
|
|
|
}
|
2021-01-21 22:47:56 +08:00
|
|
|
return &PendingTask{
|
|
|
|
BaseTask: base,
|
|
|
|
}
|
2020-11-24 22:54:00 +08:00
|
|
|
}
|
|
|
|
|
2021-10-01 02:49:41 +08:00
|
|
|
func toPendingTasks(in []*asynq.TaskInfo, bs BytesStringer) []*PendingTask {
|
2020-11-24 22:54:00 +08:00
|
|
|
out := make([]*PendingTask, len(in))
|
2021-09-18 20:26:00 +08:00
|
|
|
for i, ti := range in {
|
2021-10-01 02:49:41 +08:00
|
|
|
out[i] = toPendingTask(ti, bs)
|
2020-11-24 22:54:00 +08:00
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
|
|
|
type ScheduledTask struct {
|
|
|
|
*BaseTask
|
|
|
|
NextProcessAt time.Time `json:"next_process_at"`
|
|
|
|
}
|
|
|
|
|
2021-10-01 02:49:41 +08:00
|
|
|
func toScheduledTask(ti *asynq.TaskInfo, bs BytesStringer) *ScheduledTask {
|
2020-11-24 22:54:00 +08:00
|
|
|
base := &BaseTask{
|
2021-09-18 20:26:00 +08:00
|
|
|
ID: ti.ID,
|
|
|
|
Type: ti.Type,
|
2021-10-01 02:49:41 +08:00
|
|
|
Payload: bs.String(ti.Payload),
|
2021-09-18 20:26:00 +08:00
|
|
|
Queue: ti.Queue,
|
|
|
|
MaxRetry: ti.MaxRetry,
|
|
|
|
Retried: ti.Retried,
|
|
|
|
LastError: ti.LastErr,
|
2020-11-24 22:54:00 +08:00
|
|
|
}
|
|
|
|
return &ScheduledTask{
|
|
|
|
BaseTask: base,
|
2021-09-18 20:26:00 +08:00
|
|
|
NextProcessAt: ti.NextProcessAt,
|
2020-11-24 22:54:00 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-01 02:49:41 +08:00
|
|
|
func toScheduledTasks(in []*asynq.TaskInfo, bs BytesStringer) []*ScheduledTask {
|
2020-11-24 22:54:00 +08:00
|
|
|
out := make([]*ScheduledTask, len(in))
|
2021-09-18 20:26:00 +08:00
|
|
|
for i, ti := range in {
|
2021-10-01 02:49:41 +08:00
|
|
|
out[i] = toScheduledTask(ti, bs)
|
2020-11-24 22:54:00 +08:00
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
|
|
|
type RetryTask struct {
|
|
|
|
*BaseTask
|
|
|
|
NextProcessAt time.Time `json:"next_process_at"`
|
|
|
|
}
|
|
|
|
|
2021-10-01 02:49:41 +08:00
|
|
|
func toRetryTask(ti *asynq.TaskInfo, bs BytesStringer) *RetryTask {
|
2020-11-24 22:54:00 +08:00
|
|
|
base := &BaseTask{
|
2021-09-18 20:26:00 +08:00
|
|
|
ID: ti.ID,
|
|
|
|
Type: ti.Type,
|
2021-10-01 02:49:41 +08:00
|
|
|
Payload: bs.String(ti.Payload),
|
2021-09-18 20:26:00 +08:00
|
|
|
Queue: ti.Queue,
|
|
|
|
MaxRetry: ti.MaxRetry,
|
|
|
|
Retried: ti.Retried,
|
|
|
|
LastError: ti.LastErr,
|
2020-11-24 22:54:00 +08:00
|
|
|
}
|
|
|
|
return &RetryTask{
|
|
|
|
BaseTask: base,
|
2021-09-18 20:26:00 +08:00
|
|
|
NextProcessAt: ti.NextProcessAt,
|
2020-11-24 22:54:00 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-01 02:49:41 +08:00
|
|
|
func toRetryTasks(in []*asynq.TaskInfo, bs BytesStringer) []*RetryTask {
|
2020-11-24 22:54:00 +08:00
|
|
|
out := make([]*RetryTask, len(in))
|
2021-09-18 20:26:00 +08:00
|
|
|
for i, ti := range in {
|
2021-10-01 02:49:41 +08:00
|
|
|
out[i] = toRetryTask(ti, bs)
|
2020-11-24 22:54:00 +08:00
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
2021-01-13 03:59:44 +08:00
|
|
|
type ArchivedTask struct {
|
2020-11-24 22:54:00 +08:00
|
|
|
*BaseTask
|
|
|
|
LastFailedAt time.Time `json:"last_failed_at"`
|
|
|
|
}
|
|
|
|
|
2021-10-01 02:49:41 +08:00
|
|
|
func toArchivedTask(ti *asynq.TaskInfo, bs BytesStringer) *ArchivedTask {
|
2020-11-24 22:54:00 +08:00
|
|
|
base := &BaseTask{
|
2021-09-18 20:26:00 +08:00
|
|
|
ID: ti.ID,
|
|
|
|
Type: ti.Type,
|
2021-10-01 02:49:41 +08:00
|
|
|
Payload: bs.String(ti.Payload),
|
2021-09-18 20:26:00 +08:00
|
|
|
Queue: ti.Queue,
|
|
|
|
MaxRetry: ti.MaxRetry,
|
|
|
|
Retried: ti.Retried,
|
|
|
|
LastError: ti.LastErr,
|
2020-11-24 22:54:00 +08:00
|
|
|
}
|
2021-01-13 03:59:44 +08:00
|
|
|
return &ArchivedTask{
|
2020-11-24 22:54:00 +08:00
|
|
|
BaseTask: base,
|
2021-09-18 20:26:00 +08:00
|
|
|
LastFailedAt: ti.LastFailedAt,
|
2020-11-24 22:54:00 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-01 02:49:41 +08:00
|
|
|
func toArchivedTasks(in []*asynq.TaskInfo, bs BytesStringer) []*ArchivedTask {
|
2021-01-13 03:59:44 +08:00
|
|
|
out := make([]*ArchivedTask, len(in))
|
2021-09-18 20:26:00 +08:00
|
|
|
for i, ti := range in {
|
2021-10-01 02:49:41 +08:00
|
|
|
out[i] = toArchivedTask(ti, bs)
|
2020-11-24 22:54:00 +08:00
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|
2020-12-02 23:19:06 +08:00
|
|
|
|
|
|
|
type SchedulerEntry struct {
|
2021-04-05 06:15:54 +08:00
|
|
|
ID string `json:"id"`
|
|
|
|
Spec string `json:"spec"`
|
|
|
|
TaskType string `json:"task_type"`
|
2021-06-02 11:37:18 +08:00
|
|
|
TaskPayload string `json:"task_payload"`
|
2021-04-05 06:15:54 +08:00
|
|
|
Opts []string `json:"options"`
|
|
|
|
NextEnqueueAt string `json:"next_enqueue_at"`
|
2020-12-04 22:48:58 +08:00
|
|
|
// This field is omitted if there were no previous enqueue events.
|
|
|
|
PrevEnqueueAt string `json:"prev_enqueue_at,omitempty"`
|
2020-12-02 23:19:06 +08:00
|
|
|
}
|
|
|
|
|
2021-10-01 02:49:41 +08:00
|
|
|
func toSchedulerEntry(e *asynq.SchedulerEntry, bs BytesStringer) *SchedulerEntry {
|
2020-12-02 23:19:06 +08:00
|
|
|
opts := make([]string, 0) // create a non-nil, empty slice to avoid null in json output
|
|
|
|
for _, o := range e.Opts {
|
|
|
|
opts = append(opts, o.String())
|
|
|
|
}
|
2020-12-04 22:48:58 +08:00
|
|
|
prev := ""
|
|
|
|
if !e.Prev.IsZero() {
|
|
|
|
prev = e.Prev.Format(time.RFC3339)
|
|
|
|
}
|
2020-12-02 23:19:06 +08:00
|
|
|
return &SchedulerEntry{
|
|
|
|
ID: e.ID,
|
|
|
|
Spec: e.Spec,
|
2021-04-05 06:15:54 +08:00
|
|
|
TaskType: e.Task.Type(),
|
2021-10-01 02:49:41 +08:00
|
|
|
TaskPayload: bs.String(e.Task.Payload()),
|
2020-12-02 23:19:06 +08:00
|
|
|
Opts: opts,
|
2020-12-04 22:48:58 +08:00
|
|
|
NextEnqueueAt: e.Next.Format(time.RFC3339),
|
|
|
|
PrevEnqueueAt: prev,
|
2020-12-02 23:19:06 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-01 02:49:41 +08:00
|
|
|
func toSchedulerEntries(in []*asynq.SchedulerEntry, bs BytesStringer) []*SchedulerEntry {
|
2020-12-02 23:19:06 +08:00
|
|
|
out := make([]*SchedulerEntry, len(in))
|
|
|
|
for i, e := range in {
|
2021-10-01 02:49:41 +08:00
|
|
|
out[i] = toSchedulerEntry(e, bs)
|
2020-12-02 23:19:06 +08:00
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|
2020-12-27 02:05:19 +08:00
|
|
|
|
|
|
|
type SchedulerEnqueueEvent struct {
|
|
|
|
TaskID string `json:"task_id"`
|
|
|
|
EnqueuedAt string `json:"enqueued_at"`
|
|
|
|
}
|
|
|
|
|
2021-10-01 02:49:41 +08:00
|
|
|
func toSchedulerEnqueueEvent(e *asynq.SchedulerEnqueueEvent) *SchedulerEnqueueEvent {
|
2020-12-27 02:05:19 +08:00
|
|
|
return &SchedulerEnqueueEvent{
|
|
|
|
TaskID: e.TaskID,
|
|
|
|
EnqueuedAt: e.EnqueuedAt.Format(time.RFC3339),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-01 02:49:41 +08:00
|
|
|
func toSchedulerEnqueueEvents(in []*asynq.SchedulerEnqueueEvent) []*SchedulerEnqueueEvent {
|
2020-12-27 02:05:19 +08:00
|
|
|
out := make([]*SchedulerEnqueueEvent, len(in))
|
|
|
|
for i, e := range in {
|
2021-10-01 02:49:41 +08:00
|
|
|
out[i] = toSchedulerEnqueueEvent(e)
|
2020-12-27 02:05:19 +08:00
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|
2020-12-31 00:58:50 +08:00
|
|
|
|
|
|
|
type ServerInfo struct {
|
|
|
|
ID string `json:"id"`
|
|
|
|
Host string `json:"host"`
|
|
|
|
PID int `json:"pid"`
|
|
|
|
Concurrency int `json:"concurrency"`
|
|
|
|
Queues map[string]int `json:"queue_priorities"`
|
|
|
|
StrictPriority bool `json:"strict_priority_enabled"`
|
|
|
|
Started string `json:"start_time"`
|
|
|
|
Status string `json:"status"`
|
|
|
|
ActiveWorkers []*WorkerInfo `json:"active_workers"`
|
|
|
|
}
|
|
|
|
|
2021-10-01 02:49:41 +08:00
|
|
|
func toServerInfo(info *asynq.ServerInfo, bs BytesStringer) *ServerInfo {
|
2020-12-31 00:58:50 +08:00
|
|
|
return &ServerInfo{
|
|
|
|
ID: info.ID,
|
|
|
|
Host: info.Host,
|
|
|
|
PID: info.PID,
|
|
|
|
Concurrency: info.Concurrency,
|
|
|
|
Queues: info.Queues,
|
|
|
|
StrictPriority: info.StrictPriority,
|
|
|
|
Started: info.Started.Format(time.RFC3339),
|
|
|
|
Status: info.Status,
|
2021-10-01 02:49:41 +08:00
|
|
|
ActiveWorkers: toWorkerInfoList(info.ActiveWorkers, bs),
|
2020-12-31 00:58:50 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-01 02:49:41 +08:00
|
|
|
func toServerInfoList(in []*asynq.ServerInfo, bs BytesStringer) []*ServerInfo {
|
2020-12-31 00:58:50 +08:00
|
|
|
out := make([]*ServerInfo, len(in))
|
|
|
|
for i, s := range in {
|
2021-10-01 02:49:41 +08:00
|
|
|
out[i] = toServerInfo(s, bs)
|
2020-12-31 00:58:50 +08:00
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
|
|
|
type WorkerInfo struct {
|
2021-05-29 05:40:09 +08:00
|
|
|
TaskID string `json:"task_id"`
|
|
|
|
Queue string `json:"queue"`
|
|
|
|
TaskType string `json:"task_type"`
|
2021-06-02 11:37:18 +08:00
|
|
|
TakPayload string `json:"task_payload"`
|
2021-05-29 05:40:09 +08:00
|
|
|
Started string `json:"start_time"`
|
2020-12-31 00:58:50 +08:00
|
|
|
}
|
|
|
|
|
2021-10-01 02:49:41 +08:00
|
|
|
func toWorkerInfo(info *asynq.WorkerInfo, bs BytesStringer) *WorkerInfo {
|
2020-12-31 00:58:50 +08:00
|
|
|
return &WorkerInfo{
|
2021-05-29 05:40:09 +08:00
|
|
|
TaskID: info.TaskID,
|
|
|
|
Queue: info.Queue,
|
|
|
|
TaskType: info.TaskType,
|
2021-10-01 02:49:41 +08:00
|
|
|
TakPayload: bs.String(info.TaskPayload),
|
2021-05-29 05:40:09 +08:00
|
|
|
Started: info.Started.Format(time.RFC3339),
|
2020-12-31 00:58:50 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-01 02:49:41 +08:00
|
|
|
func toWorkerInfoList(in []*asynq.WorkerInfo, bs BytesStringer) []*WorkerInfo {
|
2020-12-31 00:58:50 +08:00
|
|
|
out := make([]*WorkerInfo, len(in))
|
|
|
|
for i, w := range in {
|
2021-10-01 02:49:41 +08:00
|
|
|
out[i] = toWorkerInfo(w, bs)
|
2020-12-31 00:58:50 +08:00
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|