-
Notifications
You must be signed in to change notification settings - Fork 20
/
Copy pathbroker.go
73 lines (65 loc) · 1.97 KB
/
broker.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
package dashing
// An eventCache stores the latest event for each key, so that new clients can
// catch up.
type eventCache map[string]*Event
// A Broker broadcasts events to multiple clients.
type Broker struct {
// Create a map of clients, the keys of the map are the channels
// over which we can push messages to attached clients. (The values
// are just booleans and are meaningless)
clients map[chan *Event]bool
// Channel into which new clients can be pushed
newClients chan chan *Event
// Channel into which disconnected clients should be pushed
defunctClients chan chan *Event
// Channel into which events are pushed to be broadcast out
// to attached clients
events chan *Event
// Cache for most recent events with a certain ID
cache eventCache
}
// Start managing client connections and event broadcasts.
func (b *Broker) Start() {
go func() {
for {
// Block until we receive from one of the
// three following channels.
select {
case s := <-b.newClients:
// There is a new client attached and we
// want to start sending them events.
b.clients[s] = true
// Send all the cached events so that when a new client connects, it
// doesn't miss previous events
for _, e := range b.cache {
s <- e
}
// log.Println("Added new client")
case s := <-b.defunctClients:
// A client has detached and we want to
// stop sending them events.
delete(b.clients, s)
// log.Println("Removed client")
case event := <-b.events:
b.cache[event.ID] = event
// There is a new event to send. For each
// attached client, push the new event
// into the client's channel.
for s := range b.clients {
s <- event
}
// log.Printf("Broadcast event to %d clients", len(b.clients))
}
}
}()
}
// NewBroker creates a Broker instance.
func NewBroker() *Broker {
return &Broker{
make(map[chan *Event]bool),
make(chan (chan *Event)),
make(chan (chan *Event)),
make(chan *Event),
map[string]*Event{},
}
}