htmgo/examples/chat/ws/manager.go

208 lines
4.3 KiB
Go
Raw Normal View History

package ws
import (
2024-10-02 03:26:03 +00:00
"fmt"
"github.com/puzpuzpuz/xsync/v3"
2024-10-02 03:26:03 +00:00
"net/http"
)
2024-09-30 22:31:09 +00:00
type EventType string
const (
ConnectedEvent EventType = "connected"
DisconnectedEvent EventType = "disconnected"
MessageEvent EventType = "message"
)
type SocketEvent struct {
Id string
2024-10-01 03:08:52 +00:00
RoomId string
2024-09-30 22:31:09 +00:00
Type EventType
Payload map[string]any
}
2024-10-02 03:26:03 +00:00
type CloseEvent struct {
Code int
Reason string
}
2024-10-01 03:08:52 +00:00
type SocketConnection struct {
Id string
2024-10-02 03:26:03 +00:00
Writer http.ResponseWriter
2024-10-01 03:08:52 +00:00
RoomId string
2024-10-02 03:26:03 +00:00
Done chan CloseEvent
Flush chan bool
2024-10-01 03:08:52 +00:00
}
type SocketManager struct {
2024-10-01 18:42:14 +00:00
sockets *xsync.MapOf[string, *xsync.MapOf[string, SocketConnection]]
idToRoom *xsync.MapOf[string, string]
2024-09-30 22:31:09 +00:00
listeners []chan SocketEvent
}
func NewSocketManager() *SocketManager {
return &SocketManager{
2024-10-01 18:42:14 +00:00
sockets: xsync.NewMapOf[string, *xsync.MapOf[string, SocketConnection]](),
idToRoom: xsync.NewMapOf[string, string](),
}
}
2024-10-01 03:49:03 +00:00
func (manager *SocketManager) ForEachSocket(roomId string, cb func(conn SocketConnection)) {
2024-10-01 18:42:14 +00:00
sockets, ok := manager.sockets.Load(roomId)
if !ok {
return
}
sockets.Range(func(id string, conn SocketConnection) bool {
cb(conn)
2024-10-01 03:49:03 +00:00
return true
})
}
2024-09-30 22:31:09 +00:00
func (manager *SocketManager) Listen(listener chan SocketEvent) {
if manager.listeners == nil {
2024-09-30 22:31:09 +00:00
manager.listeners = make([]chan SocketEvent, 0)
}
manager.listeners = append(manager.listeners, listener)
}
2024-09-30 22:31:09 +00:00
func (manager *SocketManager) dispatch(event SocketEvent) {
for _, listener := range manager.listeners {
2024-09-30 22:31:09 +00:00
listener <- event
}
}
2024-09-30 22:31:09 +00:00
func (manager *SocketManager) OnMessage(id string, message map[string]any) {
2024-10-01 03:08:52 +00:00
socket := manager.Get(id)
if socket == nil {
return
}
2024-09-30 22:31:09 +00:00
manager.dispatch(SocketEvent{
Id: id,
Type: MessageEvent,
Payload: message,
2024-10-01 03:08:52 +00:00
RoomId: socket.RoomId,
2024-09-30 22:31:09 +00:00
})
}
2024-10-02 03:26:03 +00:00
func (manager *SocketManager) Add(roomId string, id string, writer http.ResponseWriter, done chan CloseEvent, flush chan bool) {
2024-10-01 18:42:14 +00:00
manager.idToRoom.Store(id, roomId)
sockets, ok := manager.sockets.LoadOrCompute(roomId, func() *xsync.MapOf[string, SocketConnection] {
return xsync.NewMapOf[string, SocketConnection]()
})
sockets.Store(id, SocketConnection{
2024-10-01 03:08:52 +00:00
Id: id,
2024-10-02 03:26:03 +00:00
Writer: writer,
2024-10-01 03:08:52 +00:00
RoomId: roomId,
2024-10-02 03:26:03 +00:00
Done: done,
Flush: flush,
2024-10-01 03:08:52 +00:00
})
2024-10-01 18:42:14 +00:00
s, ok := sockets.Load(id)
2024-10-01 03:49:03 +00:00
if !ok {
return
}
2024-10-01 18:42:14 +00:00
2024-09-30 22:31:09 +00:00
manager.dispatch(SocketEvent{
2024-10-01 03:49:03 +00:00
Id: s.Id,
2024-09-30 22:31:09 +00:00
Type: ConnectedEvent,
2024-10-01 03:49:03 +00:00
RoomId: s.RoomId,
2024-09-30 22:31:09 +00:00
Payload: map[string]any{},
})
}
func (manager *SocketManager) OnClose(id string) {
2024-10-01 03:08:52 +00:00
socket := manager.Get(id)
if socket == nil {
return
}
2024-09-30 22:31:09 +00:00
manager.dispatch(SocketEvent{
Id: id,
Type: DisconnectedEvent,
2024-10-01 03:08:52 +00:00
RoomId: socket.RoomId,
2024-09-30 22:31:09 +00:00
Payload: map[string]any{},
})
manager.sockets.Delete(id)
}
2024-10-02 03:26:03 +00:00
func (manager *SocketManager) CloseWithError(id string, code int, message string) {
conn := manager.Get(id)
if conn != nil {
2024-10-01 17:09:22 +00:00
go manager.OnClose(id)
2024-10-02 03:26:03 +00:00
conn.Done <- CloseEvent{
Code: code,
Reason: message,
}
}
}
func (manager *SocketManager) Disconnect(id string) {
conn := manager.Get(id)
if conn != nil {
2024-10-01 17:09:22 +00:00
go manager.OnClose(id)
2024-10-02 03:26:03 +00:00
conn.Done <- CloseEvent{
Code: -1,
Reason: "",
}
}
}
2024-10-01 03:08:52 +00:00
func (manager *SocketManager) Get(id string) *SocketConnection {
2024-10-01 18:42:14 +00:00
roomId, ok := manager.idToRoom.Load(id)
2024-10-01 03:08:52 +00:00
if !ok {
return nil
}
2024-10-01 18:42:14 +00:00
sockets, ok := manager.sockets.Load(roomId)
if !ok {
return nil
}
conn, ok := sockets.Load(id)
2024-10-01 03:08:52 +00:00
return &conn
}
2024-10-02 03:26:03 +00:00
func (manager *SocketManager) Ping(id string) {
conn := manager.Get(id)
if conn != nil {
manager.writeText(*conn, "ping", "")
}
}
func (manager *SocketManager) writeText(socket SocketConnection, event string, message string) {
if socket.Writer == nil {
return
}
var err error
if event != "" {
_, err = fmt.Fprintf(socket.Writer, "event: %s\ndata: %s\n\n", event, message)
} else {
_, err = fmt.Fprintf(socket.Writer, "data: %s\n\n", message)
}
if err != nil {
manager.CloseWithError(socket.Id, 1008, "failed to write message")
}
socket.Flush <- true
}
func (manager *SocketManager) BroadcastText(roomId string, message string, predicate func(conn SocketConnection) bool) {
2024-10-01 18:42:14 +00:00
sockets, ok := manager.sockets.Load(roomId)
if !ok {
return
}
sockets.Range(func(id string, conn SocketConnection) bool {
if predicate(conn) {
2024-10-02 03:26:03 +00:00
manager.writeText(conn, "", message)
}
return true
})
}
2024-09-30 22:31:09 +00:00
func (manager *SocketManager) SendText(id string, message string) {
conn := manager.Get(id)
if conn != nil {
2024-10-02 03:26:03 +00:00
manager.writeText(*conn, "", message)
2024-09-30 22:31:09 +00:00
}
}