2020-06-23 04:11:56 +03:00
|
|
|
package chat
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
log "github.com/sirupsen/logrus"
|
|
|
|
"golang.org/x/net/websocket"
|
2020-06-23 09:52:50 +03:00
|
|
|
|
2020-10-05 20:07:09 +03:00
|
|
|
"github.com/owncast/owncast/config"
|
|
|
|
"github.com/owncast/owncast/models"
|
2020-06-23 04:11:56 +03:00
|
|
|
)
|
|
|
|
|
2020-06-23 23:11:01 +03:00
|
|
|
var (
|
|
|
|
_server *server
|
|
|
|
)
|
|
|
|
|
2020-06-23 04:11:56 +03:00
|
|
|
//Server represents the server which handles the chat
|
2020-06-23 23:11:01 +03:00
|
|
|
type server struct {
|
2020-08-06 07:01:06 +03:00
|
|
|
Clients map[string]*Client
|
2020-06-23 04:11:56 +03:00
|
|
|
|
2020-06-23 23:11:01 +03:00
|
|
|
pattern string
|
|
|
|
listener models.ChatListener
|
|
|
|
|
2020-06-23 04:11:56 +03:00
|
|
|
addCh chan *Client
|
|
|
|
delCh chan *Client
|
|
|
|
sendAllCh chan models.ChatMessage
|
|
|
|
pingCh chan models.PingMessage
|
|
|
|
doneCh chan bool
|
|
|
|
errCh chan error
|
|
|
|
}
|
|
|
|
|
|
|
|
//Add adds a client to the server
|
2020-06-23 23:11:01 +03:00
|
|
|
func (s *server) add(c *Client) {
|
2020-06-23 04:11:56 +03:00
|
|
|
s.addCh <- c
|
|
|
|
}
|
|
|
|
|
|
|
|
//Remove removes a client from the server
|
2020-06-23 23:11:01 +03:00
|
|
|
func (s *server) remove(c *Client) {
|
2020-06-23 04:11:56 +03:00
|
|
|
s.delCh <- c
|
|
|
|
}
|
|
|
|
|
|
|
|
//SendToAll sends a message to all of the connected clients
|
2020-06-23 23:11:01 +03:00
|
|
|
func (s *server) SendToAll(msg models.ChatMessage) {
|
2020-06-23 04:11:56 +03:00
|
|
|
s.sendAllCh <- msg
|
|
|
|
}
|
|
|
|
|
|
|
|
//Done marks the server as done
|
2020-06-23 23:11:01 +03:00
|
|
|
func (s *server) done() {
|
2020-06-23 04:11:56 +03:00
|
|
|
s.doneCh <- true
|
|
|
|
}
|
|
|
|
|
|
|
|
//Err handles an error
|
2020-06-23 23:11:01 +03:00
|
|
|
func (s *server) err(err error) {
|
2020-06-23 04:11:56 +03:00
|
|
|
s.errCh <- err
|
|
|
|
}
|
|
|
|
|
2020-06-23 23:11:01 +03:00
|
|
|
func (s *server) sendAll(msg models.ChatMessage) {
|
2020-06-23 04:11:56 +03:00
|
|
|
for _, c := range s.Clients {
|
|
|
|
c.Write(msg)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-23 23:11:01 +03:00
|
|
|
func (s *server) ping() {
|
2020-07-29 07:30:03 +03:00
|
|
|
ping := models.PingMessage{MessageType: PING}
|
2020-06-23 04:11:56 +03:00
|
|
|
for _, c := range s.Clients {
|
|
|
|
c.pingch <- ping
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-29 07:30:03 +03:00
|
|
|
func (s *server) usernameChanged(msg models.NameChangeEvent) {
|
|
|
|
for _, c := range s.Clients {
|
|
|
|
c.usernameChangeChannel <- msg
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-23 23:11:01 +03:00
|
|
|
func (s *server) onConnection(ws *websocket.Conn) {
|
|
|
|
client := NewClient(ws)
|
2020-06-23 09:52:50 +03:00
|
|
|
|
|
|
|
defer func() {
|
2020-10-07 09:14:33 +03:00
|
|
|
log.Tracef("The client was connected for %s and sent %d messages (%s)", time.Since(client.ConnectedAt), client.MessageCount, client.ClientID)
|
2020-06-23 09:52:50 +03:00
|
|
|
|
|
|
|
if err := ws.Close(); err != nil {
|
|
|
|
s.errCh <- err
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2020-06-23 23:11:01 +03:00
|
|
|
s.add(client)
|
2020-06-23 09:52:50 +03:00
|
|
|
client.Listen()
|
|
|
|
}
|
|
|
|
|
2020-06-23 04:11:56 +03:00
|
|
|
// Listen and serve.
|
|
|
|
// It serves client connection and broadcast request.
|
2020-06-23 23:11:01 +03:00
|
|
|
func (s *server) Listen() {
|
2020-06-23 09:52:50 +03:00
|
|
|
http.Handle(s.pattern, websocket.Handler(s.onConnection))
|
2020-06-23 04:11:56 +03:00
|
|
|
|
2020-07-07 07:27:31 +03:00
|
|
|
log.Tracef("Starting the websocket listener on: %s", s.pattern)
|
2020-06-23 04:11:56 +03:00
|
|
|
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
// add new a client
|
|
|
|
case c := <-s.addCh:
|
2020-07-29 07:30:03 +03:00
|
|
|
s.Clients[c.socketID] = c
|
2020-10-07 09:14:33 +03:00
|
|
|
s.listener.ClientAdded(c.GetViewerClientFromChatClient())
|
2020-07-20 01:14:51 +03:00
|
|
|
s.sendWelcomeMessageToClient(c)
|
2020-06-23 04:11:56 +03:00
|
|
|
|
|
|
|
// remove a client
|
|
|
|
case c := <-s.delCh:
|
2020-07-29 07:30:03 +03:00
|
|
|
delete(s.Clients, c.socketID)
|
2020-10-07 09:14:33 +03:00
|
|
|
s.listener.ClientRemoved(c.ClientID)
|
2020-06-23 04:11:56 +03:00
|
|
|
|
2020-10-14 02:45:52 +03:00
|
|
|
// message was recieved from a client and should be sanitized, validated
|
|
|
|
// and distributed to other clients.
|
2020-06-23 04:11:56 +03:00
|
|
|
case msg := <-s.sendAllCh:
|
2020-10-14 02:45:52 +03:00
|
|
|
// Will turn markdown into html, sanitize user-supplied raw html
|
|
|
|
// and standardize this message into something safe we can send everyone else.
|
|
|
|
msg.RenderAndSanitizeMessageBody()
|
|
|
|
|
2020-06-23 23:11:01 +03:00
|
|
|
s.listener.MessageSent(msg)
|
2020-06-23 04:11:56 +03:00
|
|
|
s.sendAll(msg)
|
2020-10-14 02:45:52 +03:00
|
|
|
|
|
|
|
// Store in the message history
|
2020-07-13 00:53:33 +03:00
|
|
|
addMessage(msg)
|
2020-06-23 04:11:56 +03:00
|
|
|
case ping := <-s.pingCh:
|
|
|
|
fmt.Println("PING?", ping)
|
|
|
|
|
|
|
|
case err := <-s.errCh:
|
2020-07-07 07:27:31 +03:00
|
|
|
log.Error("Error:", err.Error())
|
2020-06-23 04:11:56 +03:00
|
|
|
|
|
|
|
case <-s.doneCh:
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-07-20 01:14:51 +03:00
|
|
|
|
|
|
|
func (s *server) sendWelcomeMessageToClient(c *Client) {
|
|
|
|
go func() {
|
2020-07-20 10:22:32 +03:00
|
|
|
// Add an artificial delay so people notice this message come in.
|
|
|
|
time.Sleep(7 * time.Second)
|
2020-07-20 01:14:51 +03:00
|
|
|
|
|
|
|
initialChatMessageText := fmt.Sprintf("Welcome to %s! %s", config.Config.InstanceDetails.Title, config.Config.InstanceDetails.Summary)
|
2020-10-17 03:50:00 +03:00
|
|
|
initialMessage := models.ChatMessage{"owncast-server", config.Config.InstanceDetails.Name, initialChatMessageText, "initial-message-1", "SYSTEM", true, time.Now()}
|
2020-07-20 01:14:51 +03:00
|
|
|
c.Write(initialMessage)
|
|
|
|
}()
|
|
|
|
|
|
|
|
}
|
2020-10-07 09:14:33 +03:00
|
|
|
|
|
|
|
func (s *server) getClientForClientID(clientID string) *Client {
|
|
|
|
for _, client := range s.Clients {
|
|
|
|
if client.ClientID == clientID {
|
|
|
|
return client
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|