2021-07-19 19:22:29 -07:00
package chat
import (
"encoding/json"
"fmt"
"strings"
2021-08-12 22:40:10 -07:00
"time"
2021-07-19 19:22:29 -07:00
2022-08-09 19:56:45 -07:00
"github.com/owncast/owncast/config"
2021-07-19 19:22:29 -07:00
"github.com/owncast/owncast/core/chat/events"
"github.com/owncast/owncast/core/webhooks"
2024-11-15 19:20:58 -08:00
"github.com/owncast/owncast/persistence/configrepository"
2024-07-01 18:58:50 -07:00
"github.com/owncast/owncast/persistence/userrepository"
2022-12-28 21:30:06 -08:00
"github.com/owncast/owncast/utils"
2021-07-19 19:22:29 -07:00
log "github.com/sirupsen/logrus"
)
2021-09-12 00:18:15 -07:00
func ( s * Server ) userNameChanged ( eventData chatClientEvent ) {
2021-07-19 19:22:29 -07:00
var receivedEvent events . NameChangeEvent
if err := json . Unmarshal ( eventData . data , & receivedEvent ) ; err != nil {
log . Errorln ( "error unmarshalling to NameChangeEvent" , err )
return
}
2024-11-15 19:20:58 -08:00
configRepository := configrepository . Get ( )
2021-07-19 19:22:29 -07:00
proposedUsername := receivedEvent . NewName
2022-04-21 14:55:26 -07:00
// Check if name is on the blocklist
2024-11-15 19:20:58 -08:00
blocklist := configRepository . GetForbiddenUsernameList ( )
2021-07-19 19:22:29 -07:00
2022-08-14 18:09:16 -07:00
// Names have a max length
2022-12-28 21:30:06 -08:00
proposedUsername = utils . MakeSafeStringOfLength ( proposedUsername , config . MaxChatDisplayNameLength )
2022-08-14 18:09:16 -07:00
2021-07-19 19:22:29 -07:00
for _ , blockedName := range blocklist {
normalizedName := strings . TrimSpace ( blockedName )
normalizedName = strings . ToLower ( normalizedName )
if strings . Contains ( normalizedName , proposedUsername ) {
// Denied.
2022-09-21 13:03:16 -04:00
log . Debugln ( logSanitize ( eventData . client . User . DisplayName ) , "blocked from changing name to" , logSanitize ( proposedUsername ) , "due to blocked name" , normalizedName )
2021-07-19 19:22:29 -07:00
message := fmt . Sprintf ( "You cannot change your name to **%s**." , proposedUsername )
s . sendActionToClient ( eventData . client , message )
// Resend the client's user so their username is in sync.
eventData . client . sendConnectedClientInfo ( )
return
}
}
2024-07-01 18:58:50 -07:00
userRepository := userrepository . Get ( )
2022-04-21 14:55:26 -07:00
// Check if the name is not already assigned to a registered user.
2024-07-01 18:58:50 -07:00
if available , err := userRepository . IsDisplayNameAvailable ( proposedUsername ) ; err != nil {
2022-04-21 14:55:26 -07:00
log . Errorln ( "error checking if name is available" , err )
return
} else if ! available {
2024-07-02 13:55:25 -05:00
message := fmt . Sprintf ( "The name **%s** has already been registered. If this is your name, please authenticate." , proposedUsername )
2022-04-21 14:55:26 -07:00
s . sendActionToClient ( eventData . client , message )
// Resend the client's user so their username is in sync.
eventData . client . sendConnectedClientInfo ( )
return
}
2024-07-01 18:58:50 -07:00
savedUser := userRepository . GetUserByToken ( eventData . client . accessToken )
2021-07-19 19:22:29 -07:00
oldName := savedUser . DisplayName
2023-07-11 02:16:36 -04:00
// Check that the new name is different from old.
if proposedUsername == oldName {
eventData . client . sendConnectedClientInfo ( )
return
}
2021-07-19 19:22:29 -07:00
// Save the new name
2024-07-01 18:58:50 -07:00
if err := userRepository . ChangeUsername ( eventData . client . User . ID , proposedUsername ) ; err != nil {
2022-04-21 14:55:26 -07:00
log . Errorln ( "error changing username" , err )
}
2021-07-19 19:22:29 -07:00
// Update the connected clients associated user with the new name
2021-08-13 00:05:13 -07:00
now := time . Now ( )
2021-07-19 19:22:29 -07:00
eventData . client . User = savedUser
2021-08-13 00:05:13 -07:00
eventData . client . User . NameChangedAt = & now
2021-07-19 19:22:29 -07:00
// Send chat event letting everyone about about the name change
2022-08-14 18:09:16 -07:00
savedUser . DisplayName = proposedUsername
2021-07-19 19:22:29 -07:00
broadcastEvent := events . NameChangeBroadcast {
Oldname : oldName ,
}
broadcastEvent . User = savedUser
broadcastEvent . SetDefaults ( )
payload := broadcastEvent . GetBroadcastPayload ( )
if err := s . Broadcast ( payload ) ; err != nil {
log . Errorln ( "error broadcasting NameChangeEvent" , err )
return
}
// Send chat user name changed webhook
receivedEvent . User = savedUser
2022-12-13 19:17:32 -08:00
receivedEvent . ClientID = eventData . client . Id
2021-07-19 19:22:29 -07:00
webhooks . SendChatEventUsernameChanged ( receivedEvent )
2022-05-26 13:52:04 -07:00
// Resend the client's user so their username is in sync.
eventData . client . sendConnectedClientInfo ( )
2021-07-19 19:22:29 -07:00
}
2022-08-09 19:56:45 -07:00
func ( s * Server ) userColorChanged ( eventData chatClientEvent ) {
2024-07-01 18:58:50 -07:00
userRepository := userrepository . Get ( )
2022-08-09 19:56:45 -07:00
var receivedEvent events . ColorChangeEvent
if err := json . Unmarshal ( eventData . data , & receivedEvent ) ; err != nil {
log . Errorln ( "error unmarshalling to ColorChangeEvent" , err )
return
}
// Verify this color is valid
if receivedEvent . NewColor > config . MaxUserColor {
log . Errorln ( "invalid color requested when changing user display color" )
return
}
// Save the new color
2024-07-01 18:58:50 -07:00
if err := userRepository . ChangeUserColor ( eventData . client . User . ID , receivedEvent . NewColor ) ; err != nil {
2022-08-09 19:56:45 -07:00
log . Errorln ( "error changing user display color" , err )
}
2022-11-23 19:52:39 +01:00
// Resend client's user info with new color, otherwise the name change dialog would still show the old color
eventData . client . User . DisplayColor = receivedEvent . NewColor
eventData . client . sendConnectedClientInfo ( )
2022-08-09 19:56:45 -07:00
}
2021-09-12 00:18:15 -07:00
func ( s * Server ) userMessageSent ( eventData chatClientEvent ) {
2024-07-01 18:58:50 -07:00
userRepository := userrepository . Get ( )
2021-07-19 19:22:29 -07:00
var event events . UserMessageEvent
if err := json . Unmarshal ( eventData . data , & event ) ; err != nil {
log . Errorln ( "error unmarshalling to UserMessageEvent" , err )
return
}
event . SetDefaults ( )
2022-12-13 19:17:32 -08:00
event . ClientID = eventData . client . Id
2021-07-19 19:22:29 -07:00
// Ignore empty messages
if event . Empty ( ) {
return
}
2021-08-12 22:40:10 -07:00
// Ignore if the stream has been offline
if ! getStatus ( ) . Online && getStatus ( ) . LastDisconnectTime != nil {
disconnectedTime := getStatus ( ) . LastDisconnectTime . Time
if time . Since ( disconnectedTime ) > 5 * time . Minute {
return
}
2021-07-27 19:42:05 +02:00
}
2024-07-01 18:58:50 -07:00
event . User = userRepository . GetUserByToken ( eventData . client . accessToken )
2021-07-19 19:22:29 -07:00
// Guard against nil users
if event . User == nil {
return
}
payload := event . GetBroadcastPayload ( )
if err := s . Broadcast ( payload ) ; err != nil {
log . Errorln ( "error broadcasting UserMessageEvent payload" , err )
return
}
// Send chat message sent webhook
webhooks . SendChatEvent ( & event )
2022-03-06 17:26:52 -08:00
chatMessagesSentCounter . Inc ( )
2021-07-19 19:22:29 -07:00
SaveUserMessage ( event )
2021-10-25 00:31:45 -07:00
eventData . client . MessageCount ++
2021-07-19 19:22:29 -07:00
}
2022-09-21 13:03:16 -04:00
func logSanitize ( userValue string ) string {
// strip carriage return and newline from user-submitted values to prevent log injection
2022-09-21 10:24:16 -07:00
sanitizedValue := strings . ReplaceAll ( userValue , "\n" , "" )
sanitizedValue = strings . ReplaceAll ( sanitizedValue , "\r" , "" )
2022-09-21 13:03:16 -04:00
return fmt . Sprintf ( "userSuppliedValue(%s)" , sanitizedValue )
}