-
Notifications
You must be signed in to change notification settings - Fork 10
/
Copy pathcelery.go
102 lines (88 loc) · 2.42 KB
/
celery.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
// Package celery provides a Golang client library for
// calling Celery tasks using an AMQP backend,
// it depends on channels created by http://github.com/streadway/amqp,
//
// for additional info see http://www.celeryproject.org
package celery
import (
"encoding/json"
"github.com/nu7hatch/gouuid"
"github.com/streadway/amqp"
"time"
)
// Celery task representation,
// Task - task name,
// Id - task UUID,
// Args - optional task args,
// KWArgs - optional task kwargs,
// Retries - optional number of retries,
// ETA - optional time for a scheduled task,
// Expires - optional time for task expiration
type Task struct {
Task string
Id string
Args []string
KWArgs map[string]interface{}
Retries int
ETA time.Time
Expires time.Time
}
const timeFormat = "2006-01-02T15:04:05.999999"
// Returns a pointer to a new task object
func NewTask(task string, args []string, kwargs map[string]interface{}) (*Task, error) {
id, err := uuid.NewV4()
if err != nil {
return nil, err
}
t := Task{
Task: task,
Id: id.String(),
Args: args,
KWArgs: kwargs,
}
return &t, nil
}
// Marshals a Task object into JSON bytes array,
// time objects are converted to UTC and formatted in ISO8601
func (t *Task) MarshalJSON() ([]byte, error) {
type FormattedTask struct {
Task string `json:"task"`
Id string `json:"id"`
Args []string `json:"args,omitempty"`
KWArgs map[string]interface{} `json:"kwargs,omitempty"`
Retries int `json:"retries,omitempty"`
ETA string `json:"eta,omitempty"`
Expires string `json:"expires,omitempty"`
}
out := FormattedTask{
Task: t.Task,
Id: t.Id,
Args: t.Args,
KWArgs: t.KWArgs,
Retries: t.Retries,
}
if !t.ETA.IsZero() {
out.ETA = t.ETA.UTC().Format(timeFormat)
}
if !t.Expires.IsZero() {
out.Expires = t.Expires.UTC().Format(timeFormat)
}
return json.Marshal(out)
}
// Publish a task to an AMQP channel,
// default exchange is "",
// default routing key is "celery"
func (t *Task) Publish(ch *amqp.Channel, exchange, key string) error {
body, err := json.Marshal(t)
if err != nil {
return err
}
msg := amqp.Publishing{
DeliveryMode: amqp.Persistent,
Timestamp: time.Now(),
ContentType: "application/json",
ContentEncoding: "utf-8",
Body: body,
}
return ch.Publish(exchange, key, false, false, msg)
}