init: publisher

This commit is contained in:
sujit
2024-09-30 23:00:40 +05:45
parent afd43302ef
commit a2859aa4be
6 changed files with 235 additions and 170 deletions

106
broker.go
View File

@@ -71,7 +71,6 @@ type Task struct {
CreatedAt time.Time `json:"created_at"` CreatedAt time.Time `json:"created_at"`
ProcessedAt time.Time `json:"processed_at"` ProcessedAt time.Time `json:"processed_at"`
CurrentQueue string `json:"current_queue"` CurrentQueue string `json:"current_queue"`
Result json.RawMessage `json:"result"`
Status string `json:"status"` Status string `json:"status"`
Error error `json:"error"` Error error `json:"error"`
} }
@@ -141,7 +140,7 @@ func (b *Broker) sendToPublisher(ctx context.Context, publisherID string, result
return pub.send(ctx, result) return pub.send(ctx, result)
} }
func (b *Broker) onClose(ctx context.Context, conn net.Conn) error { func (b *Broker) onClose(ctx context.Context, _ net.Conn) error {
consumerID, ok := GetConsumerID(ctx) consumerID, ok := GetConsumerID(ctx)
if ok && consumerID != "" { if ok && consumerID != "" {
if con, exists := b.consumers.Get(consumerID); exists { if con, exists := b.consumers.Get(consumerID); exists {
@@ -163,7 +162,7 @@ func (b *Broker) onClose(ctx context.Context, conn net.Conn) error {
return nil return nil
} }
func (b *Broker) onError(ctx context.Context, conn net.Conn, err error) { func (b *Broker) onError(_ context.Context, conn net.Conn, err error) {
fmt.Println("Error reading from connection:", err, conn.RemoteAddr()) fmt.Println("Error reading from connection:", err, conn.RemoteAddr())
} }
@@ -186,18 +185,24 @@ func (b *Broker) Start(ctx context.Context) error {
} }
} }
func (b *Broker) Publish(ctx context.Context, message Task, queueName string) (*Task, error) { func (b *Broker) Publish(ctx context.Context, message Task, queueName string) Result {
queue, task, err := b.AddMessageToQueue(&message, queueName) queue, task, err := b.AddMessageToQueue(&message, queueName)
if err != nil { if err != nil {
return nil, err return Result{Error: err}
}
result := Result{
Command: "PUBLISH",
Payload: message.Payload,
Queue: queueName,
MessageID: task.ID,
} }
if queue.consumers.Size() == 0 { if queue.consumers.Size() == 0 {
queue.deferred.Set(NewID(), &message) queue.deferred.Set(NewID(), &message)
fmt.Println("task deferred as no consumers are connected", queueName) fmt.Println("task deferred as no consumers are connected", queueName)
return task, nil return result
} }
queue.send(ctx, message) queue.send(ctx, message)
return task, nil return result
} }
func (b *Broker) NewQueue(qName string) *Queue { func (b *Broker) NewQueue(qName string) *Queue {
@@ -210,44 +215,32 @@ func (b *Broker) NewQueue(qName string) *Queue {
return q return q
} }
func (b *Broker) AddMessageToQueue(message *Task, queueName string) (*Queue, *Task, error) { func (b *Broker) AddMessageToQueue(task *Task, queueName string) (*Queue, *Task, error) {
queue := b.NewQueue(queueName) queue := b.NewQueue(queueName)
if message.ID == "" { if task.ID == "" {
message.ID = NewID() task.ID = NewID()
} }
if queueName != "" { if queueName != "" {
message.CurrentQueue = queueName task.CurrentQueue = queueName
} }
message.CreatedAt = time.Now() task.CreatedAt = time.Now()
queue.messages.Set(message.ID, message) queue.messages.Set(task.ID, task)
return queue, message, nil return queue, task, nil
} }
func (b *Broker) HandleProcessedMessage(ctx context.Context, clientMsg Result) error { func (b *Broker) HandleProcessedMessage(ctx context.Context, result Result) error {
publisherID, ok := GetPublisherID(ctx) publisherID, ok := GetPublisherID(ctx)
if ok && publisherID != "" { if ok && publisherID != "" {
err := b.sendToPublisher(ctx, publisherID, clientMsg) err := b.sendToPublisher(ctx, publisherID, result)
if err != nil { if err != nil {
return err return err
} }
} }
if queue, ok := b.queues.Get(clientMsg.Queue); ok { for _, callback := range b.opts.callback {
if msg, ok := queue.messages.Get(clientMsg.MessageID); ok { if callback != nil {
msg.ProcessedAt = time.Now() rs := callback(ctx, result)
msg.Status = clientMsg.Status if rs.Error != nil {
msg.Result = clientMsg.Payload return rs.Error
msg.Error = clientMsg.Error
msg.CurrentQueue = clientMsg.Queue
if clientMsg.Error != nil {
msg.Status = "error"
}
for _, callback := range b.opts.callback {
if callback != nil {
result := callback(ctx, msg)
if result.Error != nil {
return result.Error
}
}
} }
} }
} }
@@ -327,6 +320,10 @@ func (b *Broker) handleTaskMessage(ctx context.Context, _ net.Conn, msg Result)
} }
func (b *Broker) publish(ctx context.Context, conn net.Conn, msg Command) error { func (b *Broker) publish(ctx context.Context, conn net.Conn, msg Command) error {
status := "PUBLISH"
if msg.Command == REQUEST {
status = "REQUEST"
}
b.addPublisher(ctx, msg.Queue, conn) b.addPublisher(ctx, msg.Queue, conn)
task := Task{ task := Task{
ID: msg.MessageID, ID: msg.MessageID,
@@ -334,41 +331,14 @@ func (b *Broker) publish(ctx context.Context, conn net.Conn, msg Command) error
CreatedAt: time.Now(), CreatedAt: time.Now(),
CurrentQueue: msg.Queue, CurrentQueue: msg.Queue,
} }
_, err := b.Publish(ctx, task, msg.Queue) result := b.Publish(ctx, task, msg.Queue)
if err != nil { if result.Error != nil {
return err return result.Error
} }
if task.ID != "" { if task.ID != "" {
result := Result{ result.Status = status
Command: "PUBLISH", result.MessageID = task.ID
MessageID: task.ID, result.Queue = msg.Queue
Status: "success",
Queue: msg.Queue,
}
return Write(ctx, conn, result)
}
return nil
}
func (b *Broker) request(ctx context.Context, conn net.Conn, msg Command) error {
b.addPublisher(ctx, msg.Queue, conn)
task := Task{
ID: msg.MessageID,
Payload: msg.Payload,
CreatedAt: time.Now(),
CurrentQueue: msg.Queue,
}
_, err := b.Publish(ctx, task, msg.Queue)
if err != nil {
return err
}
if task.ID != "" {
result := Result{
Command: "REQUEST",
MessageID: task.ID,
Status: "success",
Queue: msg.Queue,
}
return Write(ctx, conn, result) return Write(ctx, conn, result)
} }
return nil return nil
@@ -379,10 +349,8 @@ func (b *Broker) handleCommandMessage(ctx context.Context, conn net.Conn, msg Co
case SUBSCRIBE: case SUBSCRIBE:
b.subscribe(ctx, msg.Queue, conn) b.subscribe(ctx, msg.Queue, conn)
return nil return nil
case PUBLISH: case PUBLISH, REQUEST:
return b.publish(ctx, conn, msg) return b.publish(ctx, conn, msg)
case REQUEST:
return b.request(ctx, conn, msg)
default: default:
return fmt.Errorf("unknown command: %d", msg.Command) return fmt.Errorf("unknown command: %d", msg.Command)
} }

View File

@@ -21,7 +21,7 @@ type DAG struct {
FirstNode string FirstNode string
server *mq.Broker server *mq.Broker
nodes map[string]*mq.Consumer nodes map[string]*mq.Consumer
edges map[string][]string edges map[string]string
loopEdges map[string][]string loopEdges map[string][]string
taskChMap map[string]chan mq.Result taskChMap map[string]chan mq.Result
taskResults map[string]map[string]*taskContext taskResults map[string]map[string]*taskContext
@@ -31,7 +31,7 @@ type DAG struct {
func New(opts ...mq.Option) *DAG { func New(opts ...mq.Option) *DAG {
d := &DAG{ d := &DAG{
nodes: make(map[string]*mq.Consumer), nodes: make(map[string]*mq.Consumer),
edges: make(map[string][]string), edges: make(map[string]string),
loopEdges: make(map[string][]string), loopEdges: make(map[string][]string),
taskChMap: make(map[string]chan mq.Result), taskChMap: make(map[string]chan mq.Result),
taskResults: make(map[string]map[string]*taskContext), taskResults: make(map[string]map[string]*taskContext),
@@ -50,7 +50,7 @@ func (d *DAG) AddNode(name string, handler mq.Handler, firstNode ...bool) {
d.nodes[name] = con d.nodes[name] = con
} }
func (d *DAG) AddEdge(fromNode string, toNodes ...string) { func (d *DAG) AddEdge(fromNode string, toNodes string) {
d.edges[fromNode] = toNodes d.edges[fromNode] = toNodes
} }
@@ -58,13 +58,17 @@ func (d *DAG) AddLoop(fromNode string, toNode ...string) {
d.loopEdges[fromNode] = toNode d.loopEdges[fromNode] = toNode
} }
func (d *DAG) Start(ctx context.Context) error { func (d *DAG) Prepare() {
if d.FirstNode == "" { if d.FirstNode == "" {
firstNode, ok := d.FindFirstNode() firstNode, ok := d.FindFirstNode()
if ok && firstNode != "" { if ok && firstNode != "" {
d.FirstNode = firstNode d.FirstNode = firstNode
} }
} }
}
func (d *DAG) Start(ctx context.Context) error {
d.Prepare()
if d.server.SyncMode() { if d.server.SyncMode() {
return nil return nil
} }
@@ -74,7 +78,7 @@ func (d *DAG) Start(ctx context.Context) error {
return d.server.Start(ctx) return d.server.Start(ctx)
} }
func (d *DAG) PublishTask(ctx context.Context, payload []byte, queueName string, taskID ...string) (*mq.Task, error) { func (d *DAG) PublishTask(ctx context.Context, payload []byte, queueName string, taskID ...string) mq.Result {
task := mq.Task{ task := mq.Task{
Payload: payload, Payload: payload,
} }
@@ -89,10 +93,8 @@ func (d *DAG) FindFirstNode() (string, bool) {
for n, _ := range d.nodes { for n, _ := range d.nodes {
inDegree[n] = 0 inDegree[n] = 0
} }
for _, targets := range d.edges { for _, outNode := range d.edges {
for _, outNode := range targets { inDegree[outNode]++
inDegree[outNode]++
}
} }
for _, targets := range d.loopEdges { for _, targets := range d.loopEdges {
for _, outNode := range targets { for _, outNode := range targets {
@@ -107,23 +109,89 @@ func (d *DAG) FindFirstNode() (string, bool) {
return "", false return "", false
} }
func (d *DAG) Send(payload []byte) mq.Result { func (d *DAG) Request(ctx context.Context, payload []byte) mq.Result {
return d.sendSync(ctx, mq.Result{Payload: payload})
}
func (d *DAG) Send(ctx context.Context, payload []byte) mq.Result {
if d.FirstNode == "" { if d.FirstNode == "" {
return mq.Result{Error: fmt.Errorf("initial node not defined")} return mq.Result{Error: fmt.Errorf("initial node not defined")}
} }
if d.server.SyncMode() {
return d.sendSync(ctx, mq.Result{Payload: payload})
}
resultCh := make(chan mq.Result) resultCh := make(chan mq.Result)
task, err := d.PublishTask(context.TODO(), payload, d.FirstNode) result := d.PublishTask(ctx, payload, d.FirstNode)
if err != nil { if result.Error != nil {
return mq.Result{Error: err} return result
} }
d.mu.Lock() d.mu.Lock()
d.taskChMap[task.ID] = resultCh d.taskChMap[result.MessageID] = resultCh
d.mu.Unlock() d.mu.Unlock()
finalResult := <-resultCh finalResult := <-resultCh
return finalResult return finalResult
} }
func (d *DAG) TaskCallback(ctx context.Context, task *mq.Task) mq.Result { func (d *DAG) processNode(ctx context.Context, task mq.Result) mq.Result {
if con, ok := d.nodes[task.Queue]; ok {
return con.ProcessTask(ctx, mq.Task{
ID: task.MessageID,
Payload: task.Payload,
CurrentQueue: task.Queue,
})
}
return mq.Result{Error: fmt.Errorf("no consumer to process %s", task.Queue)}
}
func (d *DAG) sendSync(ctx context.Context, task mq.Result) mq.Result {
if task.MessageID == "" {
task.MessageID = mq.NewID()
}
if task.Queue == "" {
task.Queue = d.FirstNode
}
result := d.processNode(ctx, task)
if result.Error != nil {
return result
}
for _, target := range d.loopEdges[task.Queue] {
var items, results []json.RawMessage
if err := json.Unmarshal(result.Payload, &items); err != nil {
return mq.Result{Error: err}
}
for _, item := range items {
result = d.sendSync(ctx, mq.Result{
Command: result.Command,
Payload: item,
Queue: target,
MessageID: result.MessageID,
})
if result.Error != nil {
return result
}
results = append(results, result.Payload)
}
bt, err := json.Marshal(results)
if err != nil {
return mq.Result{Error: err}
}
result.Payload = bt
}
if target, ok := d.edges[task.Queue]; ok {
result = d.sendSync(ctx, mq.Result{
Command: result.Command,
Payload: result.Payload,
Queue: target,
MessageID: result.MessageID,
})
if result.Error != nil {
return result
}
}
return result
}
func (d *DAG) TaskCallback(ctx context.Context, task mq.Result) mq.Result {
if task.Error != nil { if task.Error != nil {
return mq.Result{Error: task.Error} return mq.Result{Error: task.Error}
} }
@@ -133,7 +201,7 @@ func (d *DAG) TaskCallback(ctx context.Context, task *mq.Task) mq.Result {
completed := false completed := false
var nodeType string var nodeType string
if ok && triggeredNode != "" { if ok && triggeredNode != "" {
taskResults, ok := d.taskResults[task.ID] taskResults, ok := d.taskResults[task.MessageID]
if ok { if ok {
nodeResult, exists := taskResults[triggeredNode] nodeResult, exists := taskResults[triggeredNode]
if exists { if exists {
@@ -143,12 +211,16 @@ func (d *DAG) TaskCallback(ctx context.Context, task *mq.Task) mq.Result {
} }
switch nodeResult.nodeType { switch nodeResult.nodeType {
case "loop": case "loop":
nodeResult.results = append(nodeResult.results, task.Result) nodeResult.results = append(nodeResult.results, task.Payload)
result = nodeResult.results if completed {
result = nodeResult.results
}
nodeType = "loop" nodeType = "loop"
case "edge": case "edge":
nodeResult.result = task.Result nodeResult.result = task.Payload
result = nodeResult.result if completed {
result = nodeResult.result
}
nodeType = "edge" nodeType = "edge"
} }
} }
@@ -160,26 +232,26 @@ func (d *DAG) TaskCallback(ctx context.Context, task *mq.Task) mq.Result {
if completed { if completed {
payload, _ = json.Marshal(result) payload, _ = json.Marshal(result)
} else { } else {
payload = task.Result payload = task.Payload
} }
if loopNodes, exists := d.loopEdges[task.CurrentQueue]; exists { if loopNodes, exists := d.loopEdges[task.Queue]; exists {
var items []json.RawMessage var items []json.RawMessage
if err := json.Unmarshal(payload, &items); err != nil { if err := json.Unmarshal(payload, &items); err != nil {
return mq.Result{Error: task.Error} return mq.Result{Error: task.Error}
} }
d.taskResults[task.ID] = map[string]*taskContext{ d.taskResults[task.MessageID] = map[string]*taskContext{
task.CurrentQueue: { task.Queue: {
totalItems: len(items), totalItems: len(items),
nodeType: "loop", nodeType: "loop",
}, },
} }
ctx = mq.SetHeaders(ctx, map[string]string{mq.TriggerNode: task.CurrentQueue}) ctx = mq.SetHeaders(ctx, map[string]string{mq.TriggerNode: task.Queue})
for _, loopNode := range loopNodes { for _, loopNode := range loopNodes {
for _, item := range items { for _, item := range items {
_, err := d.PublishTask(ctx, item, loopNode, task.ID) rs := d.PublishTask(ctx, item, loopNode, task.MessageID)
if err != nil { if rs.Error != nil {
return mq.Result{Error: task.Error} return rs
} }
} }
} }
@@ -187,37 +259,35 @@ func (d *DAG) TaskCallback(ctx context.Context, task *mq.Task) mq.Result {
return mq.Result{} return mq.Result{}
} }
if nodeType == "loop" && completed { if nodeType == "loop" && completed {
task.CurrentQueue = triggeredNode task.Queue = triggeredNode
} }
ctx = mq.SetHeaders(ctx, map[string]string{mq.TriggerNode: task.CurrentQueue}) ctx = mq.SetHeaders(ctx, map[string]string{mq.TriggerNode: task.Queue})
edges, exists := d.edges[task.CurrentQueue] edge, exists := d.edges[task.Queue]
if exists { if exists {
d.taskResults[task.ID] = map[string]*taskContext{ d.taskResults[task.MessageID] = map[string]*taskContext{
task.CurrentQueue: { task.Queue: {
totalItems: 1, totalItems: 1,
nodeType: "edge", nodeType: "edge",
}, },
} }
for _, edge := range edges { rs := d.PublishTask(ctx, payload, edge, task.MessageID)
_, err := d.PublishTask(ctx, payload, edge, task.ID) if rs.Error != nil {
if err != nil { return rs
return mq.Result{Error: task.Error}
}
} }
} else if completed { } else if completed {
d.mu.Lock() d.mu.Lock()
if resultCh, ok := d.taskChMap[task.ID]; ok { if resultCh, ok := d.taskChMap[task.MessageID]; ok {
resultCh <- mq.Result{ resultCh <- mq.Result{
Command: "complete", Command: "complete",
Payload: payload, Payload: payload,
Queue: task.CurrentQueue, Queue: task.Queue,
MessageID: task.ID, MessageID: task.MessageID,
Status: "done", Status: "done",
} }
delete(d.taskChMap, task.ID) delete(d.taskChMap, task.MessageID)
delete(d.taskResults, task.ID) delete(d.taskResults, task.MessageID)
} }
d.mu.Unlock() d.mu.Unlock()
} }
return mq.Result{} return task
} }

View File

@@ -3,84 +3,71 @@ package main
import ( import (
"context" "context"
"encoding/json" "encoding/json"
"fmt"
"io" "io"
"log" "log"
"net/http" "net/http"
"github.com/oarkflow/mq" "github.com/oarkflow/mq"
"github.com/oarkflow/mq/dag" "github.com/oarkflow/mq/dag"
"github.com/oarkflow/mq/examples/tasks"
) )
var d *dag.DAG var d *dag.DAG
func main() { func main() {
d = dag.New() d = dag.New()
d.AddNode("queue1", func(ctx context.Context, task mq.Task) mq.Result { d.AddNode("queue1", tasks.Node1)
return mq.Result{Payload: task.Payload, MessageID: task.ID} d.AddNode("queue2", tasks.Node2)
}) d.AddNode("queue3", tasks.Node3)
d.AddNode("queue2", func(ctx context.Context, task mq.Task) mq.Result { d.AddNode("queue4", tasks.Node4)
return mq.Result{Payload: task.Payload, MessageID: task.ID}
})
d.AddNode("queue3", func(ctx context.Context, task mq.Task) mq.Result {
var data map[string]any
err := json.Unmarshal(task.Payload, &data)
if err != nil {
return mq.Result{Error: err}
}
data["salary"] = fmt.Sprintf("12000%v", data["user_id"])
bt, _ := json.Marshal(data)
return mq.Result{Payload: bt, MessageID: task.ID}
})
d.AddNode("queue4", func(ctx context.Context, task mq.Task) mq.Result {
var data []map[string]any
err := json.Unmarshal(task.Payload, &data)
if err != nil {
return mq.Result{Error: err}
}
payload := map[string]any{"storage": data}
bt, _ := json.Marshal(payload)
return mq.Result{Payload: bt, MessageID: task.ID}
})
d.AddEdge("queue1", "queue2") d.AddEdge("queue1", "queue2")
d.AddLoop("queue2", "queue3") d.AddLoop("queue2", "queue3")
d.AddEdge("queue2", "queue4") d.AddEdge("queue2", "queue4")
d.Prepare()
go func() { go func() {
err := d.Start(context.TODO()) err := d.Start(context.TODO())
if err != nil { if err != nil {
panic(err) panic(err)
} }
}() }()
http.HandleFunc("/send-task", sendTaskHandler) http.HandleFunc("/publish", requestHandler("publish"))
http.HandleFunc("/request", requestHandler("request"))
log.Println("HTTP server started on http://localhost:8083") log.Println("HTTP server started on http://localhost:8083")
log.Fatal(http.ListenAndServe(":8083", nil)) log.Fatal(http.ListenAndServe(":8083", nil))
} }
func sendTaskHandler(w http.ResponseWriter, r *http.Request) {
if r.Method != http.MethodPost { func requestHandler(requestType string) func(w http.ResponseWriter, r *http.Request) {
http.Error(w, "Invalid request method", http.StatusMethodNotAllowed) return func(w http.ResponseWriter, r *http.Request) {
return if r.Method != http.MethodPost {
} http.Error(w, "Invalid request method", http.StatusMethodNotAllowed)
var payload []byte
if r.Body != nil {
defer r.Body.Close()
var err error
payload, err = io.ReadAll(r.Body)
if err != nil {
http.Error(w, "Failed to read request body", http.StatusBadRequest)
return return
} }
} else { var payload []byte
http.Error(w, "Empty request body", http.StatusBadRequest) if r.Body != nil {
return defer r.Body.Close()
var err error
payload, err = io.ReadAll(r.Body)
if err != nil {
http.Error(w, "Failed to read request body", http.StatusBadRequest)
return
}
} else {
http.Error(w, "Empty request body", http.StatusBadRequest)
return
}
var rs mq.Result
if requestType == "request" {
rs = d.Request(context.Background(), payload)
} else {
rs = d.Send(context.Background(), payload)
}
w.Header().Set("Content-Type", "application/json")
result := map[string]any{
"message_id": rs.MessageID,
"payload": string(rs.Payload),
"error": rs.Error,
}
json.NewEncoder(w).Encode(result)
} }
fmt.Println(string(payload))
finalResult := d.Send(payload)
w.Header().Set("Content-Type", "application/json")
result := map[string]any{
"message_id": finalResult.MessageID,
"payload": string(finalResult.Payload),
"error": finalResult.Error,
}
json.NewEncoder(w).Encode(result)
} }

View File

@@ -8,8 +8,8 @@ import (
) )
func main() { func main() {
b := mq.NewBroker(mq.WithCallback(func(ctx context.Context, task *mq.Task) mq.Result { b := mq.NewBroker(mq.WithCallback(func(ctx context.Context, task mq.Result) mq.Result {
fmt.Println("Received task", task.ID, "Payload", string(task.Payload), "Result", string(task.Result), task.Error, task.CurrentQueue) fmt.Println("Received task", task.MessageID, "Payload", string(task.Payload), task.Error, task.Queue)
return mq.Result{} return mq.Result{}
})) }))
b.NewQueue("queue1") b.NewQueue("queue1")

40
examples/tasks/tasks.go Normal file
View File

@@ -0,0 +1,40 @@
package tasks
import (
"context"
"encoding/json"
"fmt"
"github.com/oarkflow/mq"
)
func Node1(ctx context.Context, task mq.Task) mq.Result {
fmt.Println("Processing queue1")
return mq.Result{Payload: task.Payload, MessageID: task.ID}
}
func Node2(ctx context.Context, task mq.Task) mq.Result {
return mq.Result{Payload: task.Payload, MessageID: task.ID}
}
func Node3(ctx context.Context, task mq.Task) mq.Result {
var data map[string]any
err := json.Unmarshal(task.Payload, &data)
if err != nil {
return mq.Result{Error: err}
}
data["salary"] = fmt.Sprintf("12000%v", data["user_id"])
bt, _ := json.Marshal(data)
return mq.Result{Payload: bt, MessageID: task.ID}
}
func Node4(ctx context.Context, task mq.Task) mq.Result {
var data []map[string]any
err := json.Unmarshal(task.Payload, &data)
if err != nil {
return mq.Result{Error: err}
}
payload := map[string]any{"storage": data}
bt, _ := json.Marshal(payload)
return mq.Result{Payload: bt, MessageID: task.ID}
}

View File

@@ -11,7 +11,7 @@ type Options struct {
messageHandler MessageHandler messageHandler MessageHandler
closeHandler CloseHandler closeHandler CloseHandler
errorHandler ErrorHandler errorHandler ErrorHandler
callback []func(context.Context, *Task) Result callback []func(context.Context, Result) Result
maxRetries int maxRetries int
initialDelay time.Duration initialDelay time.Duration
maxBackoff time.Duration maxBackoff time.Duration
@@ -68,7 +68,7 @@ func WithMaxBackoff(val time.Duration) Option {
} }
// WithCallback - // WithCallback -
func WithCallback(val ...func(context.Context, *Task) Result) Option { func WithCallback(val ...func(context.Context, Result) Result) Option {
return func(opts *Options) { return func(opts *Options) {
opts.callback = val opts.callback = val
} }