mirror of
https://github.com/AynaLivePlayer/AynaLivePlayer.git
synced 2025-12-06 10:22:50 +08:00
152 lines
3.0 KiB
Go
152 lines
3.0 KiB
Go
package webinfo
|
|
|
|
import (
|
|
"AynaLivePlayer/config"
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"github.com/gorilla/websocket"
|
|
"net/http"
|
|
"sync"
|
|
)
|
|
|
|
var upgrader = websocket.Upgrader{
|
|
ReadBufferSize: 1024,
|
|
WriteBufferSize: 1024,
|
|
CheckOrigin: func(r *http.Request) bool {
|
|
return true
|
|
},
|
|
}
|
|
|
|
type WebInfoServer struct {
|
|
Info OutInfo
|
|
Server *http.Server
|
|
Clients map[*Client]int
|
|
Running bool
|
|
lock sync.Mutex
|
|
}
|
|
|
|
type Client struct {
|
|
conn *websocket.Conn
|
|
Data chan []byte
|
|
Close chan byte
|
|
}
|
|
|
|
func NewWebInfoServer(port int) *WebInfoServer {
|
|
server := &WebInfoServer{
|
|
Clients: map[*Client]int{},
|
|
}
|
|
mux := http.NewServeMux()
|
|
mux.Handle("/", http.FileServer(http.Dir(config.GetAssetPath("webinfo"))))
|
|
mux.HandleFunc("/ws/info", server.handleInfo)
|
|
mux.HandleFunc("/api/info", server.getInfo)
|
|
server.Server = &http.Server{
|
|
Addr: fmt.Sprintf("localhost:%d", port),
|
|
Handler: mux,
|
|
}
|
|
return server
|
|
}
|
|
|
|
func (s *WebInfoServer) getInfo(w http.ResponseWriter, r *http.Request) {
|
|
w.Header().Set("Content-Type", "application/json")
|
|
d, _ := json.Marshal(s.Info)
|
|
_, err := w.Write(d)
|
|
if err != nil {
|
|
lg.Warnf("api get info error: %s", err)
|
|
return
|
|
}
|
|
}
|
|
|
|
func (s *WebInfoServer) handleInfo(w http.ResponseWriter, r *http.Request) {
|
|
lg.Debug("connection start")
|
|
conn, err := upgrader.Upgrade(w, r, nil)
|
|
if err != nil {
|
|
lg.Warnf("upgrade error: %s", err)
|
|
return
|
|
}
|
|
client := &Client{
|
|
conn: conn,
|
|
Data: make(chan []byte, 16),
|
|
Close: make(chan byte, 1),
|
|
}
|
|
s.addClient(client)
|
|
defer s.removeClient(client)
|
|
go func() {
|
|
for {
|
|
_, _, err := client.conn.ReadMessage()
|
|
if err != nil {
|
|
client.Close <- 1
|
|
}
|
|
}
|
|
}()
|
|
for {
|
|
lg.Trace("waiting for message")
|
|
select {
|
|
case data := <-client.Data:
|
|
writer, err := client.conn.NextWriter(websocket.TextMessage)
|
|
if err != nil {
|
|
lg.Warn("get writer error", err)
|
|
return
|
|
}
|
|
|
|
if _, err = writer.Write(data); err != nil {
|
|
lg.Warn("send error:", err)
|
|
return
|
|
}
|
|
if err = writer.Close(); err != nil {
|
|
lg.Warnf("can't close writer: %s", err)
|
|
return
|
|
}
|
|
case _ = <-client.Close:
|
|
lg.Debug("client close")
|
|
if err := client.conn.Close(); err != nil {
|
|
lg.Warnf("close connection encouter an error: %s", err)
|
|
}
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
func (s *WebInfoServer) SendInfo(update string, info OutInfo) {
|
|
for client := range s.Clients {
|
|
d, _ := json.Marshal(WebsocketData{Update: update, Data: info})
|
|
client.Data <- d
|
|
}
|
|
}
|
|
|
|
func (s *WebInfoServer) addClient(c *Client) {
|
|
s.lock.Lock()
|
|
s.Clients[c] = 1
|
|
s.lock.Unlock()
|
|
}
|
|
|
|
func (s *WebInfoServer) removeClient(c *Client) {
|
|
s.lock.Lock()
|
|
close(c.Data)
|
|
delete(s.Clients, c)
|
|
s.lock.Unlock()
|
|
}
|
|
|
|
func (s *WebInfoServer) Start() {
|
|
s.Running = true
|
|
go func() {
|
|
err := s.Server.ListenAndServe()
|
|
s.Running = false
|
|
if err == http.ErrServerClosed {
|
|
lg.Info("server closed")
|
|
return
|
|
}
|
|
if err != nil {
|
|
lg.Warnf("Failed to start webinfo server: %s", err)
|
|
return
|
|
}
|
|
}()
|
|
}
|
|
|
|
func (s *WebInfoServer) Stop() error {
|
|
s.lock.Lock()
|
|
s.Clients = map[*Client]int{}
|
|
s.lock.Unlock()
|
|
return s.Server.Shutdown(context.Background())
|
|
}
|