Refactor hub

This commit is contained in:
Ola
2024-03-06 17:10:39 +01:00
parent a6cfa93561
commit 4c79256c40
+78 -44
View File
@@ -2,90 +2,124 @@ package melody
import ( import (
"sync" "sync"
"sync/atomic"
) )
type sessionSet struct {
mu sync.RWMutex
members map[*Session]struct{}
}
func (ss *sessionSet) add(s *Session) {
ss.mu.Lock()
defer ss.mu.Unlock()
ss.members[s] = struct{}{}
}
func (ss *sessionSet) del(s *Session) {
ss.mu.Lock()
defer ss.mu.Unlock()
delete(ss.members, s)
}
func (ss *sessionSet) clear() {
ss.mu.Lock()
defer ss.mu.Unlock()
ss.members = make(map[*Session]struct{})
}
func (ss *sessionSet) each(cb func(*Session)) {
ss.mu.RLock()
defer ss.mu.RUnlock()
for s := range ss.members {
cb(s)
}
}
func (ss *sessionSet) len() int {
ss.mu.RLock()
defer ss.mu.RUnlock()
return len(ss.members)
}
func (ss *sessionSet) all() []*Session {
ss.mu.RLock()
defer ss.mu.RUnlock()
s := make([]*Session, 0, len(ss.members))
for k := range ss.members {
s = append(s, k)
}
return s
}
type hub struct { type hub struct {
sessions map[*Session]bool sessions sessionSet
broadcast chan *envelope broadcast chan *envelope
register chan *Session register chan *Session
unregister chan *Session unregister chan *Session
exit chan *envelope exit chan *envelope
open bool open atomic.Bool
rwmutex *sync.RWMutex
} }
func newHub() *hub { func newHub() *hub {
return &hub{ return &hub{
sessions: make(map[*Session]bool), sessions: sessionSet{
members: make(map[*Session]struct{}),
},
broadcast: make(chan *envelope), broadcast: make(chan *envelope),
register: make(chan *Session), register: make(chan *Session),
unregister: make(chan *Session), unregister: make(chan *Session),
exit: make(chan *envelope), exit: make(chan *envelope),
open: true,
rwmutex: &sync.RWMutex{},
} }
} }
func (h *hub) run() { func (h *hub) run() {
h.open.Store(true)
loop: loop:
for { for {
select { select {
case s := <-h.register: case s := <-h.register:
h.rwmutex.Lock() h.sessions.add(s)
h.sessions[s] = true
h.rwmutex.Unlock()
case s := <-h.unregister: case s := <-h.unregister:
if _, ok := h.sessions[s]; ok { h.sessions.del(s)
h.rwmutex.Lock()
delete(h.sessions, s)
h.rwmutex.Unlock()
}
case m := <-h.broadcast: case m := <-h.broadcast:
h.rwmutex.RLock() h.sessions.each(func(s *Session) {
for s := range h.sessions { if m.filter == nil {
if m.filter != nil { s.writeMessage(m)
if m.filter(s) { } else if m.filter(s) {
s.writeMessage(m)
}
} else {
s.writeMessage(m) s.writeMessage(m)
} }
} })
h.rwmutex.RUnlock()
case m := <-h.exit: case m := <-h.exit:
h.rwmutex.Lock() h.sessions.each(func(s *Session) {
for s := range h.sessions {
s.writeMessage(m) s.writeMessage(m)
delete(h.sessions, s)
s.Close() s.Close()
} })
h.open = false
h.rwmutex.Unlock() h.sessions.clear()
h.open.Store(false)
break loop break loop
} }
} }
} }
func (h *hub) closed() bool { func (h *hub) closed() bool {
h.rwmutex.RLock() return !h.open.Load()
defer h.rwmutex.RUnlock()
return !h.open
} }
func (h *hub) len() int { func (h *hub) len() int {
h.rwmutex.RLock() return h.sessions.len()
defer h.rwmutex.RUnlock()
return len(h.sessions)
} }
func (h *hub) all() []*Session { func (h *hub) all() []*Session {
h.rwmutex.RLock() return h.sessions.all()
defer h.rwmutex.RUnlock()
s := make([]*Session, 0, len(h.sessions))
for k := range h.sessions {
s = append(s, k)
}
return s
} }