2023-01-11 06:38:48 +00:00
|
|
|
package barertc
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
2023-01-27 06:54:02 +00:00
|
|
|
"errors"
|
2023-01-11 06:38:48 +00:00
|
|
|
"fmt"
|
2023-11-11 22:59:49 +00:00
|
|
|
"io"
|
2023-01-11 06:38:48 +00:00
|
|
|
"net/http"
|
2023-04-01 02:46:42 +00:00
|
|
|
"sort"
|
2023-02-05 08:53:50 +00:00
|
|
|
"strings"
|
2023-03-23 03:21:04 +00:00
|
|
|
"sync"
|
2023-01-11 06:38:48 +00:00
|
|
|
"time"
|
|
|
|
|
2023-03-22 04:29:24 +00:00
|
|
|
"git.kirsle.net/apps/barertc/pkg/config"
|
2023-02-06 01:42:09 +00:00
|
|
|
"git.kirsle.net/apps/barertc/pkg/jwt"
|
2023-01-11 06:38:48 +00:00
|
|
|
"git.kirsle.net/apps/barertc/pkg/log"
|
2023-08-14 02:21:27 +00:00
|
|
|
"git.kirsle.net/apps/barertc/pkg/messages"
|
2023-02-09 04:01:06 +00:00
|
|
|
"git.kirsle.net/apps/barertc/pkg/util"
|
2023-01-11 06:38:48 +00:00
|
|
|
"nhooyr.io/websocket"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Subscriber represents a connected WebSocket session.
|
|
|
|
type Subscriber struct {
|
2023-01-27 04:34:58 +00:00
|
|
|
// User properties
|
2023-07-01 01:41:06 +00:00
|
|
|
ID int // ID assigned by server
|
|
|
|
Username string
|
|
|
|
ChatStatus string
|
|
|
|
VideoStatus int
|
2023-08-29 00:49:50 +00:00
|
|
|
DND bool // Do Not Disturb status (DMs are closed)
|
2023-07-01 01:41:06 +00:00
|
|
|
JWTClaims *jwt.Claims
|
|
|
|
authenticated bool // has passed the login step
|
2023-08-07 04:06:27 +00:00
|
|
|
loginAt time.Time
|
2023-12-11 02:43:18 +00:00
|
|
|
|
|
|
|
// Connection details (WebSocket).
|
|
|
|
conn *websocket.Conn // WebSocket user
|
|
|
|
ctx context.Context
|
|
|
|
cancel context.CancelFunc
|
|
|
|
messages chan []byte
|
|
|
|
closeSlow func()
|
|
|
|
|
|
|
|
// Polling API users.
|
|
|
|
usePolling bool
|
|
|
|
sessionID string
|
|
|
|
lastPollAt time.Time
|
|
|
|
lastPollJWT time.Time // give a new JWT once in a while
|
2023-03-23 03:21:04 +00:00
|
|
|
|
2023-09-04 20:36:12 +00:00
|
|
|
muteMu sync.RWMutex
|
|
|
|
booted map[string]struct{} // usernames booted off your camera
|
|
|
|
blocked map[string]struct{} // usernames you have blocked
|
|
|
|
muted map[string]struct{} // usernames you muted
|
2023-06-24 20:08:15 +00:00
|
|
|
|
|
|
|
// Record which message IDs belong to this user.
|
|
|
|
midMu sync.Mutex
|
2023-09-30 02:10:34 +00:00
|
|
|
messageIDs map[int64]struct{}
|
2023-11-11 22:59:49 +00:00
|
|
|
|
|
|
|
// Logging.
|
|
|
|
log bool
|
2023-11-18 23:38:02 +00:00
|
|
|
logfh map[string]io.WriteCloser
|
2023-01-11 06:38:48 +00:00
|
|
|
}
|
|
|
|
|
2023-12-11 02:43:18 +00:00
|
|
|
// NewSubscriber initializes a connected chat user.
|
|
|
|
func (s *Server) NewSubscriber(ctx context.Context, cancelFunc func()) *Subscriber {
|
|
|
|
return &Subscriber{
|
|
|
|
ctx: ctx,
|
|
|
|
cancel: cancelFunc,
|
|
|
|
messages: make(chan []byte, s.subscriberMessageBuffer),
|
|
|
|
booted: make(map[string]struct{}),
|
|
|
|
muted: make(map[string]struct{}),
|
|
|
|
blocked: make(map[string]struct{}),
|
|
|
|
messageIDs: make(map[int64]struct{}),
|
|
|
|
ChatStatus: "online",
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewWebSocketSubscriber returns a new subscriber with a WebSocket connection.
|
|
|
|
func (s *Server) NewWebSocketSubscriber(ctx context.Context, conn *websocket.Conn, cancelFunc func()) *Subscriber {
|
|
|
|
sub := s.NewSubscriber(ctx, cancelFunc)
|
|
|
|
sub.conn = conn
|
|
|
|
sub.closeSlow = func() {
|
|
|
|
conn.Close(websocket.StatusPolicyViolation, "connection too slow to keep up with messages")
|
|
|
|
}
|
|
|
|
return sub
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewPollingSubscriber returns a new subscriber using the polling API.
|
|
|
|
func (s *Server) NewPollingSubscriber(ctx context.Context, cancelFunc func()) *Subscriber {
|
|
|
|
sub := s.NewSubscriber(ctx, cancelFunc)
|
|
|
|
sub.usePolling = true
|
|
|
|
sub.lastPollAt = time.Now()
|
|
|
|
sub.lastPollJWT = time.Now()
|
|
|
|
sub.closeSlow = func() {
|
|
|
|
// Their outbox is filled up, disconnect them.
|
|
|
|
log.Error("Polling subscriber %s#%d: inbox is filled up!", sub.Username, sub.ID)
|
|
|
|
|
|
|
|
// Send an exit message.
|
|
|
|
if sub.authenticated && sub.ChatStatus != "hidden" {
|
|
|
|
sub.authenticated = false
|
|
|
|
s.Broadcast(messages.Message{
|
|
|
|
Action: messages.ActionPresence,
|
|
|
|
Username: sub.Username,
|
2024-03-15 06:04:24 +00:00
|
|
|
Message: messages.PresenceExited,
|
2023-12-11 02:43:18 +00:00
|
|
|
})
|
|
|
|
s.SendWhoList()
|
|
|
|
}
|
|
|
|
|
|
|
|
s.DeleteSubscriber(sub)
|
|
|
|
}
|
|
|
|
return sub
|
|
|
|
}
|
|
|
|
|
|
|
|
// OnClientMessage handles a chat protocol message from the user's WebSocket or polling API.
|
|
|
|
func (s *Server) OnClientMessage(sub *Subscriber, msg messages.Message) {
|
|
|
|
// What action are they performing?
|
|
|
|
switch msg.Action {
|
|
|
|
case messages.ActionLogin:
|
|
|
|
s.OnLogin(sub, msg)
|
|
|
|
case messages.ActionMessage:
|
|
|
|
s.OnMessage(sub, msg)
|
|
|
|
case messages.ActionFile:
|
|
|
|
s.OnFile(sub, msg)
|
|
|
|
case messages.ActionMe:
|
|
|
|
s.OnMe(sub, msg)
|
|
|
|
case messages.ActionOpen:
|
|
|
|
s.OnOpen(sub, msg)
|
|
|
|
case messages.ActionBoot:
|
|
|
|
s.OnBoot(sub, msg, true)
|
|
|
|
case messages.ActionUnboot:
|
|
|
|
s.OnBoot(sub, msg, false)
|
|
|
|
case messages.ActionMute, messages.ActionUnmute:
|
|
|
|
s.OnMute(sub, msg, msg.Action == messages.ActionMute)
|
|
|
|
case messages.ActionBlock:
|
|
|
|
s.OnBlock(sub, msg)
|
|
|
|
case messages.ActionBlocklist:
|
|
|
|
s.OnBlocklist(sub, msg)
|
|
|
|
case messages.ActionCandidate:
|
|
|
|
s.OnCandidate(sub, msg)
|
|
|
|
case messages.ActionSDP:
|
|
|
|
s.OnSDP(sub, msg)
|
|
|
|
case messages.ActionWatch:
|
|
|
|
s.OnWatch(sub, msg)
|
|
|
|
case messages.ActionUnwatch:
|
|
|
|
s.OnUnwatch(sub, msg)
|
|
|
|
case messages.ActionTakeback:
|
|
|
|
s.OnTakeback(sub, msg)
|
|
|
|
case messages.ActionReact:
|
|
|
|
s.OnReact(sub, msg)
|
|
|
|
case messages.ActionReport:
|
|
|
|
s.OnReport(sub, msg)
|
|
|
|
case messages.ActionPing:
|
|
|
|
default:
|
2024-05-17 06:33:19 +00:00
|
|
|
sub.ChatServer("Unsupported message type: %s", msg.Action)
|
2023-12-11 02:43:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-11 06:38:48 +00:00
|
|
|
// ReadLoop spawns a goroutine that reads from the websocket connection.
|
|
|
|
func (sub *Subscriber) ReadLoop(s *Server) {
|
|
|
|
go func() {
|
|
|
|
for {
|
|
|
|
msgType, data, err := sub.conn.Read(sub.ctx)
|
|
|
|
if err != nil {
|
2023-02-09 04:01:06 +00:00
|
|
|
log.Error("ReadLoop error(%d=%s): %+v", sub.ID, sub.Username, err)
|
2023-01-27 04:34:58 +00:00
|
|
|
s.DeleteSubscriber(sub)
|
2023-02-09 04:01:06 +00:00
|
|
|
|
2023-03-29 01:09:13 +00:00
|
|
|
// Notify if this user was auth'd and not hidden
|
|
|
|
if sub.authenticated && sub.ChatStatus != "hidden" {
|
2023-08-14 02:21:27 +00:00
|
|
|
s.Broadcast(messages.Message{
|
|
|
|
Action: messages.ActionPresence,
|
2023-02-09 04:01:06 +00:00
|
|
|
Username: sub.Username,
|
2024-03-15 06:04:24 +00:00
|
|
|
Message: messages.PresenceExited,
|
2023-02-09 04:01:06 +00:00
|
|
|
})
|
|
|
|
s.SendWhoList()
|
|
|
|
}
|
2023-01-11 06:38:48 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if msgType != websocket.MessageText {
|
|
|
|
log.Error("Unexpected MessageType")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read the user's posted message.
|
2023-08-14 02:21:27 +00:00
|
|
|
var msg messages.Message
|
2023-01-11 06:38:48 +00:00
|
|
|
if err := json.Unmarshal(data, &msg); err != nil {
|
2023-02-09 04:01:06 +00:00
|
|
|
log.Error("Read(%d=%s) Message error: %s", sub.ID, sub.Username, err)
|
2023-01-11 06:38:48 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2023-08-14 02:21:27 +00:00
|
|
|
if msg.Action != messages.ActionFile {
|
2023-03-22 04:29:24 +00:00
|
|
|
log.Debug("Read(%d=%s): %s", sub.ID, sub.Username, data)
|
|
|
|
}
|
|
|
|
|
2023-12-11 02:43:18 +00:00
|
|
|
// Handle their message.
|
|
|
|
s.OnClientMessage(sub, msg)
|
2023-01-11 06:38:48 +00:00
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
2023-08-05 02:24:42 +00:00
|
|
|
// IsAdmin safely checks if the subscriber is an admin.
|
|
|
|
func (sub *Subscriber) IsAdmin() bool {
|
|
|
|
return sub.JWTClaims != nil && sub.JWTClaims.IsAdmin
|
|
|
|
}
|
|
|
|
|
2023-09-03 19:08:23 +00:00
|
|
|
// IsVIP safely checks if the subscriber has VIP status.
|
|
|
|
func (sub *Subscriber) IsVIP() bool {
|
|
|
|
return sub.JWTClaims != nil && sub.JWTClaims.VIP
|
|
|
|
}
|
|
|
|
|
2023-01-11 06:38:48 +00:00
|
|
|
// SendJSON sends a JSON message to the websocket client.
|
|
|
|
func (sub *Subscriber) SendJSON(v interface{}) error {
|
|
|
|
data, err := json.Marshal(v)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-03-23 03:21:04 +00:00
|
|
|
log.Debug("SendJSON(%d=%s): %s", sub.ID, sub.Username, data)
|
2023-09-30 02:10:34 +00:00
|
|
|
|
|
|
|
// Add the message to the recipient's queue. If the queue is too full,
|
|
|
|
// disconnect the client as they can't keep up.
|
|
|
|
select {
|
|
|
|
case sub.messages <- data:
|
|
|
|
default:
|
|
|
|
go sub.closeSlow()
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
2023-01-11 06:38:48 +00:00
|
|
|
}
|
|
|
|
|
2023-01-27 04:34:58 +00:00
|
|
|
// SendMe sends the current user state to the client.
|
|
|
|
func (sub *Subscriber) SendMe() {
|
2023-08-14 02:21:27 +00:00
|
|
|
sub.SendJSON(messages.Message{
|
|
|
|
Action: messages.ActionMe,
|
2023-01-27 04:34:58 +00:00
|
|
|
Username: sub.Username,
|
2023-07-01 01:41:06 +00:00
|
|
|
VideoStatus: sub.VideoStatus,
|
2023-01-27 04:34:58 +00:00
|
|
|
})
|
2024-05-17 06:33:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// SendCut sends the client a 'cut' message to deactivate their camera.
|
|
|
|
func (sub *Subscriber) SendCut() {
|
|
|
|
sub.SendJSON(messages.Message{
|
|
|
|
Action: messages.ActionCut,
|
|
|
|
})
|
2023-01-27 04:34:58 +00:00
|
|
|
}
|
|
|
|
|
2023-01-27 06:54:02 +00:00
|
|
|
// ChatServer is a convenience function to deliver a ChatServer error to the client.
|
|
|
|
func (sub *Subscriber) ChatServer(message string, v ...interface{}) {
|
2023-08-14 02:21:27 +00:00
|
|
|
sub.SendJSON(messages.Message{
|
|
|
|
Action: messages.ActionError,
|
2023-01-27 06:54:02 +00:00
|
|
|
Username: "ChatServer",
|
|
|
|
Message: fmt.Sprintf(message, v...),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-03-22 04:29:24 +00:00
|
|
|
// WebSocket handles the /ws websocket connection endpoint.
|
2023-01-11 06:38:48 +00:00
|
|
|
func (s *Server) WebSocket() http.HandlerFunc {
|
|
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
2023-02-09 04:01:06 +00:00
|
|
|
ip := util.IPAddress(r)
|
|
|
|
log.Info("WebSocket connection from %s - %s", ip, r.Header.Get("User-Agent"))
|
|
|
|
log.Debug("Headers: %+v", r.Header)
|
2023-03-31 19:40:55 +00:00
|
|
|
c, err := websocket.Accept(w, r, &websocket.AcceptOptions{
|
|
|
|
CompressionMode: websocket.CompressionDisabled,
|
|
|
|
})
|
2023-01-11 06:38:48 +00:00
|
|
|
if err != nil {
|
|
|
|
w.WriteHeader(http.StatusInternalServerError)
|
2023-02-09 04:01:06 +00:00
|
|
|
fmt.Fprintf(w, "Could not accept websocket connection: %s", err)
|
2023-01-11 06:38:48 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
defer c.Close(websocket.StatusInternalError, "the sky is falling")
|
|
|
|
|
2023-02-09 04:01:06 +00:00
|
|
|
log.Debug("WebSocket: %s has connected", ip)
|
2023-03-22 04:29:24 +00:00
|
|
|
c.SetReadLimit(config.Current.WebSocketReadLimit)
|
2023-01-11 06:38:48 +00:00
|
|
|
|
|
|
|
// CloseRead starts a goroutine that will read from the connection
|
|
|
|
// until it is closed.
|
|
|
|
// ctx := c.CloseRead(r.Context())
|
2023-02-05 05:00:01 +00:00
|
|
|
ctx, cancel := context.WithCancel(r.Context())
|
2023-01-11 06:38:48 +00:00
|
|
|
|
2023-12-11 02:43:18 +00:00
|
|
|
sub := s.NewWebSocketSubscriber(ctx, c, cancel)
|
2023-01-11 06:38:48 +00:00
|
|
|
|
|
|
|
s.AddSubscriber(sub)
|
2023-02-09 04:01:06 +00:00
|
|
|
defer s.DeleteSubscriber(sub)
|
2023-01-11 06:38:48 +00:00
|
|
|
|
|
|
|
go sub.ReadLoop(s)
|
2023-02-05 05:00:01 +00:00
|
|
|
pinger := time.NewTicker(PingInterval)
|
2023-01-11 06:38:48 +00:00
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case msg := <-sub.messages:
|
2023-09-30 19:46:45 +00:00
|
|
|
err = writeTimeout(ctx, time.Second*time.Duration(config.Current.WebSocketSendTimeout), c, msg)
|
2023-01-11 06:38:48 +00:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
2023-02-11 06:46:39 +00:00
|
|
|
case <-pinger.C:
|
2023-04-20 02:55:39 +00:00
|
|
|
// Send a ping, and a refreshed JWT token if the user sent one.
|
|
|
|
var token string
|
|
|
|
if sub.JWTClaims != nil {
|
|
|
|
if jwt, err := sub.JWTClaims.ReSign(); err != nil {
|
2023-10-24 02:05:02 +00:00
|
|
|
log.Error("ReSign JWT token for %s#%d: %s", sub.Username, sub.ID, err)
|
2023-04-20 02:55:39 +00:00
|
|
|
} else {
|
|
|
|
token = jwt
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-14 02:21:27 +00:00
|
|
|
sub.SendJSON(messages.Message{
|
|
|
|
Action: messages.ActionPing,
|
2023-04-20 02:55:39 +00:00
|
|
|
JWTToken: token,
|
2023-02-05 05:00:01 +00:00
|
|
|
})
|
2023-01-11 06:38:48 +00:00
|
|
|
case <-ctx.Done():
|
2023-02-05 05:00:01 +00:00
|
|
|
pinger.Stop()
|
2023-01-11 06:38:48 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-01-27 04:34:58 +00:00
|
|
|
// Auto incrementing Subscriber ID, assigned in AddSubscriber.
|
|
|
|
var SubscriberID int
|
|
|
|
|
2023-01-11 06:38:48 +00:00
|
|
|
// AddSubscriber adds a WebSocket subscriber to the server.
|
|
|
|
func (s *Server) AddSubscriber(sub *Subscriber) {
|
2023-01-27 04:34:58 +00:00
|
|
|
// Assign a unique ID.
|
|
|
|
SubscriberID++
|
|
|
|
sub.ID = SubscriberID
|
2023-02-09 04:01:06 +00:00
|
|
|
log.Debug("AddSubscriber: ID #%d", sub.ID)
|
2023-01-27 04:34:58 +00:00
|
|
|
|
2023-01-11 06:38:48 +00:00
|
|
|
s.subscribersMu.Lock()
|
|
|
|
s.subscribers[sub] = struct{}{}
|
|
|
|
s.subscribersMu.Unlock()
|
|
|
|
}
|
|
|
|
|
2023-01-27 06:54:02 +00:00
|
|
|
// GetSubscriber by username.
|
|
|
|
func (s *Server) GetSubscriber(username string) (*Subscriber, error) {
|
2023-07-30 00:54:49 +00:00
|
|
|
for _, sub := range s.IterSubscribers() {
|
2023-01-27 06:54:02 +00:00
|
|
|
if sub.Username == username {
|
|
|
|
return sub, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil, errors.New("not found")
|
|
|
|
}
|
|
|
|
|
2023-01-11 06:38:48 +00:00
|
|
|
// DeleteSubscriber removes a subscriber from the server.
|
|
|
|
func (s *Server) DeleteSubscriber(sub *Subscriber) {
|
2023-12-11 02:43:18 +00:00
|
|
|
if sub == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-01-27 04:34:58 +00:00
|
|
|
log.Error("DeleteSubscriber: %s", sub.Username)
|
2023-02-05 05:00:01 +00:00
|
|
|
|
|
|
|
// Cancel its context to clean up the for-loop goroutine.
|
|
|
|
if sub.cancel != nil {
|
2023-10-24 02:05:02 +00:00
|
|
|
log.Info("Calling sub.cancel() on subscriber: %s#%d", sub.Username, sub.ID)
|
2023-02-05 05:00:01 +00:00
|
|
|
sub.cancel()
|
|
|
|
}
|
|
|
|
|
2023-11-18 23:38:02 +00:00
|
|
|
// Clean up any log files.
|
|
|
|
sub.teardownLogs()
|
|
|
|
|
2023-01-11 06:38:48 +00:00
|
|
|
s.subscribersMu.Lock()
|
|
|
|
delete(s.subscribers, sub)
|
|
|
|
s.subscribersMu.Unlock()
|
|
|
|
}
|
|
|
|
|
2023-08-29 00:49:50 +00:00
|
|
|
// IterSubscribers loops over the subscriber list with a read lock.
|
|
|
|
func (s *Server) IterSubscribers() []*Subscriber {
|
2023-01-27 06:54:02 +00:00
|
|
|
var result = []*Subscriber{}
|
2023-01-27 04:34:58 +00:00
|
|
|
|
2023-08-29 00:49:50 +00:00
|
|
|
// Lock for reads.
|
|
|
|
s.subscribersMu.RLock()
|
2023-01-27 06:54:02 +00:00
|
|
|
for sub := range s.subscribers {
|
|
|
|
result = append(result, sub)
|
|
|
|
}
|
2023-08-29 00:49:50 +00:00
|
|
|
s.subscribersMu.RUnlock()
|
2023-01-27 04:34:58 +00:00
|
|
|
|
2023-01-27 06:54:02 +00:00
|
|
|
return result
|
2023-01-27 04:34:58 +00:00
|
|
|
}
|
|
|
|
|
2023-07-18 03:38:07 +00:00
|
|
|
// UniqueUsername ensures a username will be unique or renames it. If the name is already unique, the error result is nil.
|
|
|
|
func (s *Server) UniqueUsername(username string) (string, error) {
|
2023-04-19 05:18:12 +00:00
|
|
|
var (
|
|
|
|
subs = s.IterSubscribers()
|
|
|
|
usernames = map[string]interface{}{}
|
|
|
|
origUsername = username
|
|
|
|
counter = 2
|
|
|
|
)
|
|
|
|
for _, sub := range subs {
|
|
|
|
usernames[sub.Username] = nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check until unique.
|
|
|
|
for {
|
|
|
|
if _, ok := usernames[username]; ok {
|
|
|
|
username = fmt.Sprintf("%s %d", origUsername, counter)
|
|
|
|
counter++
|
|
|
|
} else {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-18 03:38:07 +00:00
|
|
|
if username != origUsername {
|
|
|
|
return username, errors.New("username was not unique and a unique name has been returned")
|
|
|
|
}
|
|
|
|
|
|
|
|
return username, nil
|
2023-04-19 05:18:12 +00:00
|
|
|
}
|
|
|
|
|
2023-01-11 06:38:48 +00:00
|
|
|
// Broadcast a message to the chat room.
|
2023-08-14 02:21:27 +00:00
|
|
|
func (s *Server) Broadcast(msg messages.Message) {
|
2023-03-22 04:29:24 +00:00
|
|
|
if len(msg.Message) < 1024 {
|
|
|
|
log.Debug("Broadcast: %+v", msg)
|
|
|
|
}
|
|
|
|
|
2023-09-04 20:36:12 +00:00
|
|
|
// Get the sender of this message.
|
|
|
|
sender, err := s.GetSubscriber(msg.Username)
|
|
|
|
if err != nil {
|
|
|
|
log.Error("Broadcast: sender name %s not found as a current subscriber!", msg.Username)
|
|
|
|
sender = nil
|
|
|
|
}
|
|
|
|
|
2023-07-30 00:54:49 +00:00
|
|
|
// Get the list of users who are online NOW, so we don't hold the mutex lock too long.
|
|
|
|
// Example: sending a fat GIF to a large audience could hang up the server for a long
|
|
|
|
// time until every copy of the GIF has been sent.
|
|
|
|
var subs = s.IterSubscribers()
|
|
|
|
for _, sub := range subs {
|
2023-02-06 01:42:09 +00:00
|
|
|
if !sub.authenticated {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2023-03-23 03:21:04 +00:00
|
|
|
// Don't deliver it if the receiver has muted us.
|
|
|
|
if sub.Mutes(msg.Username) {
|
|
|
|
log.Debug("Do not broadcast message to %s: they have muted or booted %s", sub.Username, msg.Username)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2023-09-04 20:36:12 +00:00
|
|
|
// Don't deliver it if there is any blocking between sender and receiver.
|
|
|
|
if sender != nil && sender.Blocks(sub) {
|
|
|
|
log.Debug("Do not broadcast message to %s: blocking between them and %s", msg.Username, sub.Username)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2023-09-03 19:48:21 +00:00
|
|
|
// VIP channels: only deliver to subscribed VIP users.
|
2023-09-08 02:43:03 +00:00
|
|
|
if ch, ok := config.Current.GetChannel(msg.Channel); ok && ch.VIP && !sub.IsVIP() && !sub.IsAdmin() {
|
2023-09-03 19:48:21 +00:00
|
|
|
log.Debug("Do not broadcast message to %s: VIP channel and they are not VIP", sub.Username)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2023-02-06 21:27:29 +00:00
|
|
|
sub.SendJSON(msg)
|
2023-01-11 06:38:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-05 08:53:50 +00:00
|
|
|
// SendTo sends a message to a given username.
|
2023-08-14 02:21:27 +00:00
|
|
|
func (s *Server) SendTo(username string, msg messages.Message) error {
|
2023-02-05 08:53:50 +00:00
|
|
|
log.Debug("SendTo(%s): %+v", username, msg)
|
|
|
|
username = strings.TrimPrefix(username, "@")
|
2023-02-06 21:27:29 +00:00
|
|
|
|
|
|
|
var found bool
|
2023-07-30 00:54:49 +00:00
|
|
|
var subs = s.IterSubscribers()
|
|
|
|
for _, sub := range subs {
|
2023-02-05 08:53:50 +00:00
|
|
|
if sub.Username == username {
|
2023-02-06 21:27:29 +00:00
|
|
|
found = true
|
2023-08-14 02:21:27 +00:00
|
|
|
sub.SendJSON(messages.Message{
|
2023-06-24 20:47:20 +00:00
|
|
|
Action: msg.Action,
|
|
|
|
Channel: msg.Channel,
|
|
|
|
Username: msg.Username,
|
|
|
|
Message: msg.Message,
|
|
|
|
MessageID: msg.MessageID,
|
2023-02-05 08:53:50 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2023-02-06 21:27:29 +00:00
|
|
|
|
|
|
|
if !found {
|
|
|
|
return fmt.Errorf("%s is not online", username)
|
|
|
|
}
|
|
|
|
return nil
|
2023-02-05 08:53:50 +00:00
|
|
|
}
|
|
|
|
|
2023-01-27 04:34:58 +00:00
|
|
|
// SendWhoList broadcasts the connected members to everybody in the room.
|
|
|
|
func (s *Server) SendWhoList() {
|
2023-01-27 06:54:02 +00:00
|
|
|
var (
|
|
|
|
subscribers = s.IterSubscribers()
|
2023-04-01 02:46:42 +00:00
|
|
|
usernames = []string{} // distinct and sorted usernames
|
|
|
|
userSub = map[string]*Subscriber{}
|
2023-01-27 06:54:02 +00:00
|
|
|
)
|
|
|
|
|
2023-04-01 02:46:42 +00:00
|
|
|
for _, sub := range subscribers {
|
2023-04-02 06:44:15 +00:00
|
|
|
if !sub.authenticated {
|
|
|
|
continue
|
|
|
|
}
|
2023-04-01 02:46:42 +00:00
|
|
|
usernames = append(usernames, sub.Username)
|
|
|
|
userSub[sub.Username] = sub
|
|
|
|
}
|
|
|
|
sort.Strings(usernames)
|
|
|
|
|
2023-03-23 03:21:04 +00:00
|
|
|
// Build the WhoList for each subscriber.
|
|
|
|
// TODO: it's the only way to fake videoActive for booted user views.
|
2023-01-27 06:54:02 +00:00
|
|
|
for _, sub := range subscribers {
|
2023-02-06 01:42:09 +00:00
|
|
|
if !sub.authenticated {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2023-08-14 02:21:27 +00:00
|
|
|
var users = []messages.WhoList{}
|
2023-04-01 02:46:42 +00:00
|
|
|
for _, un := range usernames {
|
|
|
|
user := userSub[un]
|
2023-03-29 01:09:13 +00:00
|
|
|
if user.ChatStatus == "hidden" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2023-09-04 20:36:12 +00:00
|
|
|
// Blocking: hide the presence of both people from the Who List.
|
|
|
|
if user.Blocks(sub) {
|
|
|
|
log.Debug("WhoList: hide %s from %s (blocking)", user.Username, sub.Username)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2023-08-14 02:21:27 +00:00
|
|
|
who := messages.WhoList{
|
2023-07-01 01:41:06 +00:00
|
|
|
Username: user.Username,
|
|
|
|
Status: user.ChatStatus,
|
|
|
|
Video: user.VideoStatus,
|
2023-08-29 00:49:50 +00:00
|
|
|
DND: user.DND,
|
2023-08-07 04:06:27 +00:00
|
|
|
LoginAt: user.loginAt.Unix(),
|
2023-03-23 03:21:04 +00:00
|
|
|
}
|
|
|
|
|
2023-09-03 19:08:23 +00:00
|
|
|
// Hide video flags of other users (never for the current user).
|
|
|
|
if user.Username != sub.Username {
|
|
|
|
|
|
|
|
// If this person had booted us, force their camera to "off"
|
2023-09-13 03:03:10 +00:00
|
|
|
if user.Boots(sub.Username) || user.Mutes(sub.Username) {
|
|
|
|
if sub.IsAdmin() {
|
|
|
|
// They kicked the admin off, but admin can reopen the cam if they want.
|
|
|
|
// But, unset the user's "auto-open your camera" flag, so if the admin
|
|
|
|
// reopens it, the admin's cam won't open on the recipient's screen.
|
|
|
|
who.Video ^= messages.VideoFlagMutualOpen
|
|
|
|
} else {
|
|
|
|
// Force their video to "off"
|
|
|
|
who.Video = 0
|
|
|
|
}
|
2023-09-03 19:08:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// If this person's VideoFlag is set to VIP Only, force their camera to "off"
|
|
|
|
// except when the person looking has the VIP status.
|
|
|
|
if (user.VideoStatus&messages.VideoFlagOnlyVIP == messages.VideoFlagOnlyVIP) && (!sub.IsVIP() && !sub.IsAdmin()) {
|
|
|
|
who.Video = 0
|
|
|
|
}
|
2023-03-23 03:21:04 +00:00
|
|
|
}
|
|
|
|
|
2023-03-29 01:09:13 +00:00
|
|
|
if user.JWTClaims != nil {
|
2023-03-23 03:21:04 +00:00
|
|
|
who.Operator = user.JWTClaims.IsAdmin
|
|
|
|
who.Avatar = user.JWTClaims.Avatar
|
|
|
|
who.ProfileURL = user.JWTClaims.ProfileURL
|
2023-04-19 05:18:12 +00:00
|
|
|
who.Nickname = user.JWTClaims.Nick
|
2023-08-06 02:38:04 +00:00
|
|
|
who.Emoji = user.JWTClaims.Emoji
|
|
|
|
who.Gender = user.JWTClaims.Gender
|
2023-09-03 19:08:23 +00:00
|
|
|
|
|
|
|
// VIP flags: if we are in MutuallySecret mode, only VIPs can see
|
|
|
|
// other VIP flags on the Who List.
|
|
|
|
if config.Current.VIP.MutuallySecret {
|
2024-05-14 01:51:54 +00:00
|
|
|
if sub.IsVIP() {
|
2023-09-03 19:08:23 +00:00
|
|
|
who.VIP = user.JWTClaims.VIP
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
who.VIP = user.JWTClaims.VIP
|
|
|
|
}
|
2023-03-23 03:21:04 +00:00
|
|
|
}
|
|
|
|
users = append(users, who)
|
2023-02-06 01:42:09 +00:00
|
|
|
}
|
2023-01-27 04:34:58 +00:00
|
|
|
|
2023-08-14 02:21:27 +00:00
|
|
|
sub.SendJSON(messages.Message{
|
|
|
|
Action: messages.ActionWhoList,
|
2023-02-11 06:46:39 +00:00
|
|
|
WhoList: users,
|
2023-01-27 04:34:58 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-23 03:21:04 +00:00
|
|
|
// Boots checks whether the subscriber has blocked username from their camera.
|
|
|
|
func (s *Subscriber) Boots(username string) bool {
|
|
|
|
s.muteMu.RLock()
|
|
|
|
defer s.muteMu.RUnlock()
|
|
|
|
_, ok := s.booted[username]
|
|
|
|
return ok
|
|
|
|
}
|
|
|
|
|
|
|
|
// Mutes checks whether the subscriber has muted username.
|
|
|
|
func (s *Subscriber) Mutes(username string) bool {
|
|
|
|
s.muteMu.RLock()
|
|
|
|
defer s.muteMu.RUnlock()
|
|
|
|
_, ok := s.muted[username]
|
|
|
|
return ok
|
|
|
|
}
|
|
|
|
|
2023-09-04 20:36:12 +00:00
|
|
|
// Blocks checks whether the subscriber blocks the username, or vice versa (blocking goes both directions).
|
|
|
|
func (s *Subscriber) Blocks(other *Subscriber) bool {
|
2023-09-04 20:37:21 +00:00
|
|
|
if s == nil || other == nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2023-09-08 02:43:03 +00:00
|
|
|
// If either side is an admin, blocking is not allowed.
|
|
|
|
if s.IsAdmin() || other.IsAdmin() {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2023-09-04 20:36:12 +00:00
|
|
|
s.muteMu.RLock()
|
|
|
|
defer s.muteMu.RUnlock()
|
|
|
|
|
|
|
|
// Forward block?
|
|
|
|
if _, ok := s.blocked[other.Username]; ok {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reverse block?
|
|
|
|
other.muteMu.RLock()
|
|
|
|
defer other.muteMu.RUnlock()
|
|
|
|
_, ok := other.blocked[s.Username]
|
|
|
|
return ok
|
|
|
|
}
|
|
|
|
|
2023-01-11 06:38:48 +00:00
|
|
|
func writeTimeout(ctx context.Context, timeout time.Duration, c *websocket.Conn, msg []byte) error {
|
|
|
|
ctx, cancel := context.WithTimeout(ctx, timeout)
|
|
|
|
defer cancel()
|
|
|
|
return c.Write(ctx, websocket.MessageText, msg)
|
|
|
|
}
|