2020-06-22 20:11:56 -05:00
package chat
import (
2021-07-19 19:22:29 -07:00
"encoding/json"
2021-11-02 19:27:41 -07:00
"fmt"
2020-06-22 20:11:56 -05:00
"net/http"
2020-12-21 19:42:47 -08:00
"sync"
2020-06-22 20:11:56 -05:00
"time"
log "github.com/sirupsen/logrus"
2020-06-23 01:52:50 -05:00
2021-07-19 19:22:29 -07:00
"github.com/gorilla/websocket"
2022-03-06 23:26:24 -08:00
"github.com/owncast/owncast/config"
2021-07-19 19:22:29 -07:00
"github.com/owncast/owncast/core/chat/events"
2021-02-18 23:05:52 -08:00
"github.com/owncast/owncast/core/data"
2021-07-19 19:22:29 -07:00
"github.com/owncast/owncast/core/user"
2021-02-18 23:05:52 -08:00
"github.com/owncast/owncast/core/webhooks"
2021-08-12 14:55:21 -07:00
"github.com/owncast/owncast/geoip"
2021-07-19 19:22:29 -07:00
"github.com/owncast/owncast/utils"
2020-06-22 20:11:56 -05:00
)
2021-09-12 00:18:15 -07:00
var _server * Server
2020-06-23 15:11:01 -05:00
2021-09-12 00:18:15 -07:00
// Server represents an instance of the chat server.
type Server struct {
2023-05-30 10:31:43 -07:00
clients map [ uint ] * Client
2020-12-21 19:42:47 -08:00
2021-07-19 19:22:29 -07:00
// send outbound message payload to all clients
outbound chan [ ] byte
2020-06-22 20:11:56 -05:00
2021-07-19 19:22:29 -07:00
// receive inbound message payload from all clients
inbound chan chatClientEvent
2020-06-23 15:11:01 -05:00
2021-07-19 19:22:29 -07:00
// unregister requests from clients.
2021-07-26 17:30:28 -07:00
unregister chan uint // the ChatClient id
2021-10-12 15:21:37 -05:00
2023-10-08 14:22:28 -07:00
geoipClient * geoip . Client
// a map of user IDs and timers that fire for chat part messages.
userPartedTimers map [ string ] * time . Ticker
2023-05-30 10:31:43 -07:00
seq uint
maxSocketConnectionLimit int64
mu sync . RWMutex
2020-06-22 20:11:56 -05:00
}
2021-09-12 00:18:15 -07:00
// NewChat will return a new instance of the chat server.
func NewChat ( ) * Server {
2021-08-13 15:26:44 -07:00
maximumConcurrentConnectionLimit := getMaximumConcurrentConnectionLimit ( )
setSystemConcurrentConnectionLimit ( maximumConcurrentConnectionLimit )
2021-09-12 00:18:15 -07:00
server := & Server {
clients : map [ uint ] * Client { } ,
2021-08-13 15:26:44 -07:00
outbound : make ( chan [ ] byte ) ,
inbound : make ( chan chatClientEvent ) ,
unregister : make ( chan uint ) ,
maxSocketConnectionLimit : maximumConcurrentConnectionLimit ,
2021-10-12 15:21:37 -05:00
geoipClient : geoip . NewClient ( ) ,
2023-09-10 10:58:11 -07:00
userPartedTimers : map [ string ] * time . Ticker { } ,
2021-07-19 19:22:29 -07:00
}
2020-06-22 20:11:56 -05:00
2021-07-19 19:22:29 -07:00
return server
2020-06-22 20:11:56 -05:00
}
2021-09-12 00:18:15 -07:00
// Run will start the chat server.
func ( s * Server ) Run ( ) {
2021-07-19 19:22:29 -07:00
for {
select {
2021-09-12 00:18:15 -07:00
case clientID := <- s . unregister :
2023-09-10 10:58:11 -07:00
if client , ok := s . clients [ clientID ] ; ok {
s . handleClientDisconnected ( client )
2021-07-19 19:22:29 -07:00
s . mu . Lock ( )
2021-09-12 00:18:15 -07:00
delete ( s . clients , clientID )
2021-07-19 19:22:29 -07:00
s . mu . Unlock ( )
}
case message := <- s . inbound :
s . eventReceived ( message )
}
}
2020-06-22 20:11:56 -05:00
}
2021-07-19 19:22:29 -07:00
// Addclient registers new connection as a User.
2021-09-12 00:18:15 -07:00
func ( s * Server ) Addclient ( conn * websocket . Conn , user * user . User , accessToken string , userAgent string , ipAddress string ) * Client {
client := & Client {
2021-07-19 19:22:29 -07:00
server : s ,
conn : conn ,
User : user ,
2022-03-06 20:34:49 -08:00
IPAddress : ipAddress ,
2021-07-19 19:22:29 -07:00
accessToken : accessToken ,
2021-07-31 12:48:42 -07:00
send : make ( chan [ ] byte , 256 ) ,
2021-07-19 19:22:29 -07:00
UserAgent : userAgent ,
ConnectedAt : time . Now ( ) ,
}
2023-09-10 10:58:11 -07:00
shouldSendJoinedMessages := data . GetChatJoinPartMessagesEnabled ( )
2021-09-21 14:06:23 -07:00
2024-02-18 15:49:50 -08:00
// If there are existing clients connected for this user do not send
// a user joined message. Do not put this under a mutex, as
// GetClientsForUser already has a lock.
if existingConnectedClients , _ := GetClientsForUser ( user . ID ) ; len ( existingConnectedClients ) > 0 {
shouldSendJoinedMessages = false
}
2021-07-19 19:22:29 -07:00
s . mu . Lock ( )
{
2023-09-10 10:58:11 -07:00
// If there is a pending disconnect timer then clear it.
// Do not send user joined message if enough time hasn't passed where the
// user chat part message hasn't been sent yet.
if ticker , ok := s . userPartedTimers [ user . ID ] ; ok {
ticker . Stop ( )
delete ( s . userPartedTimers , user . ID )
shouldSendJoinedMessages = false
}
2022-12-13 19:17:32 -08:00
client . Id = s . seq
s . clients [ client . Id ] = client
2021-07-19 19:22:29 -07:00
s . seq ++
}
s . mu . Unlock ( )
2022-12-13 19:17:32 -08:00
log . Traceln ( "Adding client" , client . Id , "total count:" , len ( s . clients ) )
2021-07-19 19:22:29 -07:00
go client . writePump ( )
go client . readPump ( )
client . sendConnectedClientInfo ( )
if getStatus ( ) . Online {
2021-09-21 14:06:23 -07:00
if shouldSendJoinedMessages {
s . sendUserJoinedMessage ( client )
}
2021-07-19 19:22:29 -07:00
s . sendWelcomeMessageToClient ( client )
}
2021-08-12 14:55:21 -07:00
// Asynchronously, optionally, fetch GeoIP data.
2021-09-12 00:18:15 -07:00
go func ( client * Client ) {
2021-10-12 15:21:37 -05:00
client . Geo = s . geoipClient . GetGeoFromIP ( ipAddress )
2021-08-12 14:55:21 -07:00
} ( client )
2021-07-19 19:22:29 -07:00
return client
2020-06-22 20:11:56 -05:00
}
2021-09-12 00:18:15 -07:00
func ( s * Server ) sendUserJoinedMessage ( c * Client ) {
2021-07-19 19:22:29 -07:00
userJoinedEvent := events . UserJoinedEvent { }
userJoinedEvent . SetDefaults ( )
userJoinedEvent . User = c . User
2022-12-13 19:17:32 -08:00
userJoinedEvent . ClientID = c . Id
2021-07-19 19:22:29 -07:00
if err := s . Broadcast ( userJoinedEvent . GetBroadcastPayload ( ) ) ; err != nil {
log . Errorln ( "error adding client to chat server" , err )
2020-06-22 20:11:56 -05:00
}
2021-07-19 19:22:29 -07:00
// Send chat user joined webhook
webhooks . SendChatEventUserJoined ( userJoinedEvent )
2020-06-22 20:11:56 -05:00
}
2023-09-10 10:58:11 -07:00
func ( s * Server ) handleClientDisconnected ( c * Client ) {
2022-12-13 19:17:32 -08:00
if _ , ok := s . clients [ c . Id ] ; ok {
log . Debugln ( "Deleting" , c . Id )
delete ( s . clients , c . Id )
2020-06-22 20:11:56 -05:00
}
2023-09-10 10:58:11 -07:00
additionalClientCheck , _ := GetClientsForUser ( c . User . ID )
if len ( additionalClientCheck ) > 0 {
// This user is still connected to chat with another client.
return
}
s . userPartedTimers [ c . User . ID ] = time . NewTicker ( 10 * time . Second )
go func ( ) {
<- s . userPartedTimers [ c . User . ID ] . C
s . sendUserPartedMessage ( c )
} ( )
}
func ( s * Server ) sendUserPartedMessage ( c * Client ) {
s . userPartedTimers [ c . User . ID ] . Stop ( )
delete ( s . userPartedTimers , c . User . ID )
userPartEvent := events . UserPartEvent { }
userPartEvent . SetDefaults ( )
userPartEvent . User = c . User
userPartEvent . ClientID = c . Id
// If part messages are disabled.
if data . GetChatJoinPartMessagesEnabled ( ) {
if err := s . Broadcast ( userPartEvent . GetBroadcastPayload ( ) ) ; err != nil {
log . Errorln ( "error sending chat part message" , err )
}
}
// Send chat user joined webhook
webhooks . SendChatEventUserParted ( userPartEvent )
2020-06-22 20:11:56 -05:00
}
2021-09-12 00:18:15 -07:00
// HandleClientConnection is fired when a single client connects to the websocket.
func ( s * Server ) HandleClientConnection ( w http . ResponseWriter , r * http . Request ) {
2021-07-19 19:22:29 -07:00
if data . GetChatDisabled ( ) {
_ , _ = w . Write ( [ ] byte ( events . ChatDisabled ) )
return
2020-07-28 21:30:03 -07:00
}
2021-02-18 23:05:52 -08:00
2022-03-06 20:34:49 -08:00
ipAddress := utils . GetIPAddressFromRequest ( r )
// Check if this client's IP address is banned. If so send a rejection.
if blocked , err := data . IsIPAddressBanned ( ipAddress ) ; blocked {
log . Debugln ( "Client ip address has been blocked. Rejecting." )
w . WriteHeader ( http . StatusForbidden )
return
} else if err != nil {
log . Errorln ( "error determining if IP address is blocked: " , err )
}
2021-07-19 19:22:29 -07:00
// Limit concurrent chat connections
2021-08-13 15:26:44 -07:00
if int64 ( len ( s . clients ) ) >= s . maxSocketConnectionLimit {
log . Warnln ( "rejecting incoming client connection as it exceeds the max client count of" , s . maxSocketConnectionLimit )
2021-07-19 19:22:29 -07:00
_ , _ = w . Write ( [ ] byte ( events . ErrorMaxConnectionsExceeded ) )
return
}
2022-05-03 13:01:50 -07:00
// To allow dev web environments to connect.
upgrader . CheckOrigin = func ( r * http . Request ) bool {
return true
}
2021-07-19 19:22:29 -07:00
conn , err := upgrader . Upgrade ( w , r , nil )
if err != nil {
log . Debugln ( err )
return
}
accessToken := r . URL . Query ( ) . Get ( "accessToken" )
if accessToken == "" {
log . Errorln ( "Access token is required" )
// Return HTTP status code
2021-09-12 00:18:15 -07:00
_ = conn . Close ( )
2021-07-19 19:22:29 -07:00
return
}
// A user is required to use the websocket
user := user . GetUserByToken ( accessToken )
if user == nil {
2022-04-21 14:55:26 -07:00
// Send error that registration is required
2021-07-19 19:22:29 -07:00
_ = conn . WriteJSON ( events . EventPayload {
"type" : events . ErrorNeedsRegistration ,
} )
2021-09-12 00:18:15 -07:00
_ = conn . Close ( )
2021-07-19 19:22:29 -07:00
return
}
// User is disabled therefore we should disconnect.
if user . DisabledAt != nil {
2021-09-12 00:18:15 -07:00
log . Traceln ( "Disabled user" , user . ID , user . DisplayName , "rejected" )
2021-07-19 19:22:29 -07:00
_ = conn . WriteJSON ( events . EventPayload {
"type" : events . ErrorUserDisabled ,
} )
2021-09-12 00:18:15 -07:00
_ = conn . Close ( )
2021-07-19 19:22:29 -07:00
return
2021-02-18 23:05:52 -08:00
}
2021-03-03 20:44:13 -08:00
2021-07-19 19:22:29 -07:00
userAgent := r . UserAgent ( )
2021-08-12 14:55:21 -07:00
s . Addclient ( conn , user , accessToken , userAgent , ipAddress )
2020-07-28 21:30:03 -07:00
}
2021-07-19 19:22:29 -07:00
// Broadcast sends message to all connected clients.
2021-09-12 00:18:15 -07:00
func ( s * Server ) Broadcast ( payload events . EventPayload ) error {
2021-07-19 19:22:29 -07:00
data , err := json . Marshal ( payload )
if err != nil {
return err
}
2022-04-15 21:30:05 -05:00
s . mu . RLock ( )
defer s . mu . RUnlock ( )
2020-06-23 01:52:50 -05:00
2021-07-19 19:22:29 -07:00
for _ , client := range s . clients {
if client == nil {
continue
}
2020-06-23 01:52:50 -05:00
2021-07-19 19:22:29 -07:00
select {
case client . send <- data :
default :
2022-04-15 21:30:05 -05:00
go client . close ( )
2020-06-23 01:52:50 -05:00
}
2021-07-19 19:22:29 -07:00
}
2020-06-23 01:52:50 -05:00
2021-07-19 19:22:29 -07:00
return nil
2020-06-23 01:52:50 -05:00
}
2021-09-12 00:18:15 -07:00
// Send will send a single payload to a single connected client.
func ( s * Server ) Send ( payload events . EventPayload , client * Client ) {
2021-07-19 19:22:29 -07:00
data , err := json . Marshal ( payload )
if err != nil {
log . Errorln ( err )
return
}
2020-06-22 20:11:56 -05:00
2021-07-19 19:22:29 -07:00
client . send <- data
}
2020-06-22 20:11:56 -05:00
2022-03-06 20:34:49 -08:00
// DisconnectClients will forcefully disconnect all clients belonging to a user by ID.
func ( s * Server ) DisconnectClients ( clients [ ] * Client ) {
2021-07-19 19:22:29 -07:00
for _ , client := range clients {
2021-09-12 00:18:15 -07:00
log . Traceln ( "Disconnecting client" , client . User . ID , "owned by" , client . User . DisplayName )
2021-03-14 11:46:27 -07:00
2021-09-12 00:18:15 -07:00
go func ( client * Client ) {
2021-07-19 19:22:29 -07:00
event := events . UserDisabledEvent { }
event . SetDefaults ( )
2021-03-10 14:07:47 -05:00
2021-07-19 19:22:29 -07:00
// Send this disabled event specifically to this single connected client
// to let them know they've been banned.
_server . Send ( event . GetBroadcastPayload ( ) , client )
2020-10-13 16:45:52 -07:00
2021-07-19 19:22:29 -07:00
// Give the socket time to send out the above message.
// Unfortunately I don't know of any way to get a real callback to know when
// the message was successfully sent, so give it a couple seconds.
time . Sleep ( 2 * time . Second )
2021-02-18 23:05:52 -08:00
2021-07-19 19:22:29 -07:00
// Forcefully disconnect if still valid.
if client != nil {
client . close ( )
2021-01-31 23:57:50 +01:00
}
2021-07-19 19:22:29 -07:00
} ( client )
}
}
2020-06-22 20:11:56 -05:00
2021-11-02 19:27:41 -07:00
// SendConnectedClientInfoToUser will find all the connected clients assigned to a user
// and re-send each the connected client info.
func SendConnectedClientInfoToUser ( userID string ) error {
clients , err := GetClientsForUser ( userID )
if err != nil {
return err
}
// Get an updated reference to the user.
user := user . GetUserByID ( userID )
if user == nil {
return fmt . Errorf ( "user not found" )
}
if err != nil {
return err
}
for _ , client := range clients {
// Update the client's reference to its user.
client . User = user
// Send the update to the client.
client . sendConnectedClientInfo ( )
}
return nil
}
// SendActionToUser will send system action text to all connected clients
// assigned to a user ID.
func SendActionToUser ( userID string , text string ) error {
clients , err := GetClientsForUser ( userID )
if err != nil {
return err
}
for _ , client := range clients {
_server . sendActionToClient ( client , text )
}
return nil
}
2021-09-12 00:18:15 -07:00
func ( s * Server ) eventReceived ( event chatClientEvent ) {
2022-03-06 23:26:24 -08:00
c := event . client
u := c . User
// If established chat user only mode is enabled and the user is not old
// enough then reject this event and send them an informative message.
if u != nil && data . GetChatEstbalishedUsersOnlyMode ( ) && time . Since ( event . client . User . CreatedAt ) < config . GetDefaults ( ) . ChatEstablishedUserModeTimeDuration && ! u . IsModerator ( ) {
s . sendActionToClient ( c , "You have not been an established chat participant long enough to take part in chat. Please enjoy the stream and try again later." )
return
}
2021-07-19 19:22:29 -07:00
var typecheck map [ string ] interface { }
if err := json . Unmarshal ( event . data , & typecheck ) ; err != nil {
log . Debugln ( err )
}
2020-06-22 20:11:56 -05:00
2021-07-19 19:22:29 -07:00
eventType := typecheck [ "type" ]
switch eventType {
case events . MessageSent :
s . userMessageSent ( event )
case events . UserNameChanged :
s . userNameChanged ( event )
2022-08-09 19:56:45 -07:00
case events . UserColorChanged :
s . userColorChanged ( event )
2021-07-19 19:22:29 -07:00
default :
2022-09-21 13:03:16 -04:00
log . Debugln ( logSanitize ( fmt . Sprint ( eventType ) ) , "event not found:" , logSanitize ( fmt . Sprint ( typecheck ) ) )
2020-06-22 20:11:56 -05:00
}
}
2020-07-19 15:14:51 -07:00
2021-09-12 00:18:15 -07:00
func ( s * Server ) sendWelcomeMessageToClient ( c * Client ) {
2021-07-19 19:22:29 -07:00
// Add an artificial delay so people notice this message come in.
time . Sleep ( 7 * time . Second )
welcomeMessage := utils . RenderSimpleMarkdown ( data . GetServerWelcomeMessage ( ) )
2020-12-21 19:42:47 -08:00
2021-07-19 19:22:29 -07:00
if welcomeMessage != "" {
s . sendSystemMessageToClient ( c , welcomeMessage )
2020-12-21 19:42:47 -08:00
}
}
2021-09-12 00:18:15 -07:00
func ( s * Server ) sendAllWelcomeMessage ( ) {
2021-07-19 19:22:29 -07:00
welcomeMessage := utils . RenderSimpleMarkdown ( data . GetServerWelcomeMessage ( ) )
2020-07-19 15:14:51 -07:00
2021-07-19 19:22:29 -07:00
if welcomeMessage != "" {
clientMessage := events . SystemMessageEvent {
Event : events . Event { } ,
MessageEvent : events . MessageEvent {
Body : welcomeMessage ,
} ,
2021-03-21 17:10:56 -04:00
}
2021-07-19 19:22:29 -07:00
clientMessage . SetDefaults ( )
_ = s . Broadcast ( clientMessage . GetBroadcastPayload ( ) )
}
}
2021-09-12 00:18:15 -07:00
func ( s * Server ) sendSystemMessageToClient ( c * Client , message string ) {
2021-07-19 19:22:29 -07:00
clientMessage := events . SystemMessageEvent {
Event : events . Event { } ,
MessageEvent : events . MessageEvent {
Body : message ,
} ,
}
clientMessage . SetDefaults ( )
2021-09-13 10:26:28 +02:00
clientMessage . RenderBody ( )
2021-07-19 19:22:29 -07:00
s . Send ( clientMessage . GetBroadcastPayload ( ) , c )
}
2021-09-12 00:18:15 -07:00
func ( s * Server ) sendActionToClient ( c * Client , message string ) {
2021-07-19 19:22:29 -07:00
clientMessage := events . ActionEvent {
MessageEvent : events . MessageEvent {
Body : message ,
} ,
2021-11-02 19:27:41 -07:00
Event : events . Event {
Type : events . ChatActionSent ,
} ,
2021-07-19 19:22:29 -07:00
}
clientMessage . SetDefaults ( )
clientMessage . RenderBody ( )
s . Send ( clientMessage . GetBroadcastPayload ( ) , c )
2020-10-06 23:14:33 -07:00
}