status-go/cmd/statusd/server/signals_server.go

118 lines
2.2 KiB
Go
Raw Normal View History

package server
import (
"context"
"errors"
"net"
"net/http"
"sync"
"time"
"github.com/gorilla/websocket"
"github.com/ethereum/go-ethereum/log"
"github.com/status-im/status-go/signal"
)
type Server struct {
server *http.Server
lock sync.Mutex
connections map[*websocket.Conn]struct{}
address string
}
func NewServer() *Server {
return &Server{
connections: make(map[*websocket.Conn]struct{}, 1),
}
}
func (s *Server) Address() string {
return s.address
}
func (s *Server) Setup() {
signal.SetMobileSignalHandler(s.signalHandler)
}
func (s *Server) signalHandler(data []byte) {
s.lock.Lock()
defer s.lock.Unlock()
for connection := range s.connections {
err := connection.WriteMessage(websocket.TextMessage, data)
if err != nil {
log.Error("failed to write message: %w", err)
}
}
}
func (s *Server) Listen(address string) error {
if s.server != nil {
return errors.New("server already started")
}
s.server = &http.Server{
Addr: address,
ReadHeaderTimeout: 5 * time.Second,
}
2024-09-16 12:38:50 +00:00
mux := http.NewServeMux()
mux.HandleFunc("/signals", s.signals)
s.server.Handler = mux
listener, err := net.Listen("tcp", address)
if err != nil {
return err
}
s.address = listener.Addr().String()
go func() {
err := s.server.Serve(listener)
if !errors.Is(err, http.ErrServerClosed) {
log.Error("signals server closed with error: %w", err)
}
}()
return nil
}
func (s *Server) Stop(ctx context.Context) {
for connection := range s.connections {
err := connection.Close()
if err != nil {
log.Error("failed to close connection: %w", err)
}
delete(s.connections, connection)
}
err := s.server.Shutdown(ctx)
if err != nil {
log.Error("failed to shutdown signals server: %w", err)
}
s.server = nil
s.address = ""
}
func (s *Server) signals(w http.ResponseWriter, r *http.Request) {
s.lock.Lock()
defer s.lock.Unlock()
upgrader := websocket.Upgrader{
CheckOrigin: func(r *http.Request) bool {
return true // Accepting all requests
},
}
connection, err := upgrader.Upgrade(w, r, nil)
if err != nil {
log.Error("failed to upgrade connection: %w", err)
return
}
s.connections[connection] = struct{}{}
}