/
feed.go
84 lines (68 loc) · 1.52 KB
/
feed.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
package events
import (
"sync"
"github.com/ethereum/go-ethereum/event"
)
type Callback[T any] func(data T)
type Subscription[T any] struct {
s event.Subscription
c chan T
w *sync.WaitGroup
}
// Event feed with data of type T.
//
// Wrapper of go-ethereum/event.FeedOf that provides easier Subscribe and
// Unsubscribe calls
type FeedOf[T any] struct {
once sync.Once
feed event.FeedOf[T]
subscriptions map[string]*Subscription[T]
}
func (e *FeedOf[T]) Send(data T) (sent int) {
return e.feed.Send(data)
}
func (e *FeedOf[T]) init() {
e.subscriptions = make(map[string]*Subscription[T])
}
func (e *FeedOf[T]) Subscribe(id string, callback Callback[T]) {
e.once.Do(e.init)
e.Unsubscribe(id)
sub := &Subscription[T]{c: make(chan T), w: &sync.WaitGroup{}}
sub.s = e.feed.Subscribe(sub.c)
sub.w.Add(1)
go func() {
defer sub.w.Done()
for {
select {
case t := <-sub.c:
callback(t)
case <-sub.s.Err():
return
}
}
}()
e.subscriptions[id] = sub
}
func (e *FeedOf[T]) Unsubscribe(id string) *sync.WaitGroup {
e.once.Do(e.init)
sub, ok := e.subscriptions[id]
if ok {
delete(e.subscriptions, id)
sub.s.Unsubscribe()
return sub.w
}
return &sync.WaitGroup{}
}
// Event feed without data.
type Feed struct {
FeedOf[struct{}]
}
func (f *Feed) Send() (sent int) {
return f.feed.Send(struct{}{})
}
func (f *Feed) Subscribe(id string, callback func()) {
f.FeedOf.Subscribe(id, func(struct{}) { callback() })
}
func (f *Feed) Unsubscribe(id string) *sync.WaitGroup {
return f.FeedOf.Unsubscribe(id)
}