freifunkmanager/websocket/client.go

147 lines
2.7 KiB
Go
Raw Normal View History

2017-05-08 19:13:29 +02:00
package websocket
import (
"io"
2017-05-30 16:39:14 +02:00
"time"
2017-05-08 19:13:29 +02:00
2017-05-29 22:55:38 +02:00
"github.com/genofire/golang-lib/log"
2017-05-30 16:39:14 +02:00
"github.com/genofire/golang-lib/worker"
2017-05-08 19:13:29 +02:00
"golang.org/x/net/websocket"
)
const channelBufSize = 100
type Client struct {
ip string
ws *websocket.Conn
ch chan *Message
writeQuit chan bool
readQuit chan bool
2017-05-08 19:13:29 +02:00
}
func NewClient(ip string, ws *websocket.Conn) *Client {
if ws == nil {
log.Log.Panic("ws cannot be nil")
2017-05-08 19:13:29 +02:00
}
return &Client{
ws: ws,
ch: make(chan *Message, channelBufSize),
writeQuit: make(chan bool),
readQuit: make(chan bool),
ip: ip,
2017-05-08 19:13:29 +02:00
}
}
func (c *Client) Write(msg *Message) {
select {
case c.ch <- msg:
default:
2017-05-12 21:32:10 +02:00
clientsMutex.Lock()
delete(clients, c.ip)
2017-05-12 21:32:10 +02:00
clientsMutex.Unlock()
log.HTTP(c.ws.Request()).Error("client disconnected")
2017-05-08 19:13:29 +02:00
}
}
func (c *Client) Close() {
c.writeQuit <- true
c.readQuit <- true
log.HTTP(c.ws.Request()).Info("client disconnecting...")
}
// Listen Write and Read request via chanel
func (c *Client) Listen() {
go c.listenWrite()
2017-07-06 12:17:10 +02:00
if stats != nil {
c.Write(&Message{Type: MessageTypeStats, Body: stats})
}
2017-05-30 16:39:14 +02:00
c.publishAllData()
c.listenRead()
2017-05-08 19:13:29 +02:00
}
2017-05-30 16:39:14 +02:00
func (c *Client) publishAllData() {
2017-05-08 19:13:29 +02:00
for _, node := range nodes.List {
c.Write(&Message{Type: MessageTypeSystemNode, Node: node})
2017-05-08 19:13:29 +02:00
}
for _, node := range nodes.Current {
c.Write(&Message{Type: MessageTypeCurrentNode, Node: node})
2017-05-08 19:13:29 +02:00
}
2017-07-06 12:17:10 +02:00
if commands != nil {
for _, cmd := range commands.List {
c.Write(&Message{Type: MessageTypeCommand, Command: cmd})
}
2017-05-30 16:39:14 +02:00
}
2017-05-08 19:13:29 +02:00
}
2017-05-15 21:59:48 +02:00
func (c *Client) handleMessage(msg *Message) {
switch msg.Type {
case MessageTypeSystemNode:
nodes.UpdateNode(msg.Node)
2017-05-30 16:39:14 +02:00
break
case MessageTypeCommand:
2017-07-06 12:17:10 +02:00
if commands == nil {
break
}
2017-05-30 16:39:14 +02:00
cmd := commands.AddCommand(msg.Command)
w := worker.NewWorker(time.Millisecond*300, func() {
2017-07-07 08:12:25 +02:00
cmd.Lock()
2017-05-30 16:39:14 +02:00
SendAll(Message{Type: MessageTypeCommand, Command: cmd})
2017-07-07 08:12:25 +02:00
cmd.Unlock()
2017-05-30 16:39:14 +02:00
})
go w.Start()
go cmd.Run(func() {
w.Close()
SendAll(Message{Type: MessageTypeCommand, Command: cmd})
})
break
}
2017-05-08 19:13:29 +02:00
}
// Listen write request via chanel
func (c *Client) listenWrite() {
for {
select {
case msg := <-c.ch:
websocket.JSON.Send(c.ws, msg)
case <-c.writeQuit:
2017-07-06 12:17:10 +02:00
clientsMutex.Lock()
close(c.ch)
close(c.writeQuit)
delete(clients, c.ip)
2017-05-12 21:32:10 +02:00
clientsMutex.Unlock()
2017-05-08 19:13:29 +02:00
return
}
}
}
// Listen read request via chanel
func (c *Client) listenRead() {
for {
select {
case <-c.readQuit:
2017-05-12 21:32:10 +02:00
clientsMutex.Lock()
2017-07-06 12:17:10 +02:00
close(c.readQuit)
delete(clients, c.ip)
2017-05-12 21:32:10 +02:00
clientsMutex.Unlock()
2017-05-08 19:13:29 +02:00
return
default:
var msg Message
err := websocket.JSON.Receive(c.ws, &msg)
if err == io.EOF {
close(c.readQuit)
c.writeQuit <- true
return
2017-05-08 19:13:29 +02:00
} else if err != nil {
log.HTTP(c.ws.Request()).Error(err)
} else {
2017-05-15 21:59:48 +02:00
c.handleMessage(&msg)
2017-05-08 19:13:29 +02:00
}
}
}
}