Fix a memory leak with the chat aspect. (#23)
Essentially, the for loop wasn't being returned out of and that caused the read listener to never be let up and released to the gc
This commit is contained in:
parent
487bd12444
commit
fe96739f60
@ -3,6 +3,7 @@ package chat
|
|||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
|
"time"
|
||||||
|
|
||||||
log "github.com/sirupsen/logrus"
|
log "github.com/sirupsen/logrus"
|
||||||
"golang.org/x/net/websocket"
|
"golang.org/x/net/websocket"
|
||||||
@ -15,6 +16,9 @@ const channelBufSize = 100
|
|||||||
|
|
||||||
//Client represents a chat client.
|
//Client represents a chat client.
|
||||||
type Client struct {
|
type Client struct {
|
||||||
|
ConnectedAt time.Time
|
||||||
|
MessageCount int
|
||||||
|
|
||||||
id string
|
id string
|
||||||
ws *websocket.Conn
|
ws *websocket.Conn
|
||||||
server *Server
|
server *Server
|
||||||
@ -39,7 +43,7 @@ func NewClient(ws *websocket.Conn, server *Server) *Client {
|
|||||||
pingch := make(chan models.PingMessage)
|
pingch := make(chan models.PingMessage)
|
||||||
clientID := utils.GenerateClientIDFromRequest(ws.Request())
|
clientID := utils.GenerateClientIDFromRequest(ws.Request())
|
||||||
|
|
||||||
return &Client{clientID, ws, server, ch, pingch, doneCh}
|
return &Client{time.Now(), 0, clientID, ws, server, ch, pingch, doneCh}
|
||||||
}
|
}
|
||||||
|
|
||||||
//GetConnection gets the connection for the client
|
//GetConnection gets the connection for the client
|
||||||
@ -106,9 +110,11 @@ func (c *Client) listenRead() {
|
|||||||
|
|
||||||
if err := websocket.JSON.Receive(c.ws, &msg); err == io.EOF {
|
if err := websocket.JSON.Receive(c.ws, &msg); err == io.EOF {
|
||||||
c.doneCh <- true
|
c.doneCh <- true
|
||||||
|
return
|
||||||
} else if err != nil {
|
} else if err != nil {
|
||||||
c.server.Err(err)
|
c.server.Err(err)
|
||||||
} else {
|
} else {
|
||||||
|
c.MessageCount++
|
||||||
c.server.SendToAll(msg)
|
c.server.SendToAll(msg)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -5,19 +5,19 @@ import (
|
|||||||
"net/http"
|
"net/http"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
log "github.com/sirupsen/logrus"
|
||||||
|
"golang.org/x/net/websocket"
|
||||||
|
|
||||||
"github.com/gabek/owncast/core"
|
"github.com/gabek/owncast/core"
|
||||||
"github.com/gabek/owncast/models"
|
"github.com/gabek/owncast/models"
|
||||||
log "github.com/sirupsen/logrus"
|
|
||||||
|
|
||||||
"golang.org/x/net/websocket"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
//Server represents the server which handles the chat
|
//Server represents the server which handles the chat
|
||||||
type Server struct {
|
type Server struct {
|
||||||
Pattern string
|
|
||||||
Messages []models.ChatMessage
|
Messages []models.ChatMessage
|
||||||
Clients map[string]*Client
|
Clients map[string]*Client
|
||||||
|
|
||||||
|
pattern string
|
||||||
addCh chan *Client
|
addCh chan *Client
|
||||||
delCh chan *Client
|
delCh chan *Client
|
||||||
sendAllCh chan models.ChatMessage
|
sendAllCh chan models.ChatMessage
|
||||||
@ -27,7 +27,7 @@ type Server struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
//NewServer creates a new chat server
|
//NewServer creates a new chat server
|
||||||
func NewServer(pattern string) *Server {
|
func NewServer() *Server {
|
||||||
messages := []models.ChatMessage{}
|
messages := []models.ChatMessage{}
|
||||||
clients := make(map[string]*Client)
|
clients := make(map[string]*Client)
|
||||||
addCh := make(chan *Client)
|
addCh := make(chan *Client)
|
||||||
@ -46,9 +46,9 @@ func NewServer(pattern string) *Server {
|
|||||||
messages = append(messages, models.ChatMessage{"Blathers", "Blathers is an owl with brown feathers. His face is white and he has a yellow beak. His arms are wing shaped and he has yellow talons. His eyes are very big with small black irises. He also has big pink cheek circles on his cheeks. His belly appears to be checkered in diamonds with light brown and white squares, similar to an argyle vest, which is traditionally associated with academia. His green bowtie further alludes to his academic nature.", "https://vignette.wikia.nocookie.net/animalcrossing/images/b/b3/NH-character-Blathers.png/revision/latest?cb=20200229053519", "demo-message-6", "ChatMessage"})
|
messages = append(messages, models.ChatMessage{"Blathers", "Blathers is an owl with brown feathers. His face is white and he has a yellow beak. His arms are wing shaped and he has yellow talons. His eyes are very big with small black irises. He also has big pink cheek circles on his cheeks. His belly appears to be checkered in diamonds with light brown and white squares, similar to an argyle vest, which is traditionally associated with academia. His green bowtie further alludes to his academic nature.", "https://vignette.wikia.nocookie.net/animalcrossing/images/b/b3/NH-character-Blathers.png/revision/latest?cb=20200229053519", "demo-message-6", "ChatMessage"})
|
||||||
|
|
||||||
server := &Server{
|
server := &Server{
|
||||||
pattern,
|
|
||||||
messages,
|
messages,
|
||||||
clients,
|
clients,
|
||||||
|
"/entry", //hardcoded due to the UI requiring this and it is not configurable
|
||||||
addCh,
|
addCh,
|
||||||
delCh,
|
delCh,
|
||||||
sendAllCh,
|
sendAllCh,
|
||||||
@ -116,26 +116,27 @@ func (s *Server) ping() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *Server) onConnection(ws *websocket.Conn) {
|
||||||
|
client := NewClient(ws, s)
|
||||||
|
|
||||||
|
defer func() {
|
||||||
|
log.Printf("The client was connected for %s and sent %d messages (%s)", time.Since(client.ConnectedAt), client.MessageCount, client.id)
|
||||||
|
|
||||||
|
if err := ws.Close(); err != nil {
|
||||||
|
s.errCh <- err
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
s.Add(client)
|
||||||
|
client.Listen()
|
||||||
|
}
|
||||||
|
|
||||||
// Listen and serve.
|
// Listen and serve.
|
||||||
// It serves client connection and broadcast request.
|
// It serves client connection and broadcast request.
|
||||||
func (s *Server) Listen() {
|
func (s *Server) Listen() {
|
||||||
// websocket handler
|
http.Handle(s.pattern, websocket.Handler(s.onConnection))
|
||||||
onConnected := func(ws *websocket.Conn) {
|
|
||||||
defer func() {
|
|
||||||
err := ws.Close()
|
|
||||||
if err != nil {
|
|
||||||
s.errCh <- err
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
|
|
||||||
client := NewClient(ws, s)
|
log.Printf("Starting the websocket listener on: %s", s.pattern)
|
||||||
s.Add(client)
|
|
||||||
client.Listen()
|
|
||||||
}
|
|
||||||
|
|
||||||
http.Handle(s.Pattern, websocket.Handler(onConnected))
|
|
||||||
|
|
||||||
log.Printf("Starting the websocket listener on: %s", s.Pattern)
|
|
||||||
|
|
||||||
for {
|
for {
|
||||||
select {
|
select {
|
||||||
|
@ -15,7 +15,7 @@ import (
|
|||||||
//Start starts the router for the http, ws, and rtmp
|
//Start starts the router for the http, ws, and rtmp
|
||||||
func Start() error {
|
func Start() error {
|
||||||
// websocket server
|
// websocket server
|
||||||
chatServer := chat.NewServer("/entry")
|
chatServer := chat.NewServer()
|
||||||
go chatServer.Listen()
|
go chatServer.Listen()
|
||||||
|
|
||||||
// start the rtmp server
|
// start the rtmp server
|
||||||
|
@ -66,9 +66,9 @@ function setupWebsocket() {
|
|||||||
|
|
||||||
// Uncomment to point to somewhere other than goth.land
|
// Uncomment to point to somewhere other than goth.land
|
||||||
const protocol = location.protocol == "https:" ? "wss" : "ws"
|
const protocol = location.protocol == "https:" ? "wss" : "ws"
|
||||||
// var ws = new WebSocket(protocol + "://" + location.host + "/entry")
|
var ws = new WebSocket(protocol + "://" + location.host + "/entry")
|
||||||
|
|
||||||
var ws = new WebSocket("wss://goth.land/entry")
|
// var ws = new WebSocket("wss://goth.land/entry")
|
||||||
|
|
||||||
ws.onmessage = (e) => {
|
ws.onmessage = (e) => {
|
||||||
const model = JSON.parse(e.data)
|
const model = JSON.parse(e.data)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user