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

191 lines
4.1 KiB
Go

package server
import (
"context"
"errors"
"fmt"
"io"
"net"
"net/http"
"reflect"
"runtime"
"strings"
"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
listener net.Listener
mux *http.ServeMux
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,
}
s.mux = http.NewServeMux()
s.mux.HandleFunc("/signals", s.signals)
s.server.Handler = s.mux
var err error
s.listener, err = net.Listen("tcp", address)
if err != nil {
return err
}
s.address = s.listener.Addr().String()
return nil
}
func (s *Server) Serve() {
err := s.server.Serve(s.listener)
if !errors.Is(err, http.ErrServerClosed) {
log.Error("signals server closed with error: %w", err)
}
}
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{}{}
}
func (s *Server) addEndpointWithResponse(handler func(string) string) {
endpoint := endpointName(functionName(handler))
log.Debug("adding endpoint", "name", endpoint)
s.mux.HandleFunc(endpoint, func(w http.ResponseWriter, r *http.Request) {
request, err := io.ReadAll(r.Body)
if err != nil {
log.Error("failed to read request: %w", err)
return
}
response := handler(string(request))
_, err = w.Write([]byte(response))
if err != nil {
log.Error("failed to write response: %w", err)
}
})
}
func (s *Server) addEndpointNoRequest(handler func() string) {
endpoint := endpointName(functionName(handler))
log.Debug("adding endpoint", "name", endpoint)
s.mux.HandleFunc(endpoint, func(w http.ResponseWriter, r *http.Request) {
response := handler()
_, err := w.Write([]byte(response))
if err != nil {
log.Error("failed to write response: %w", err)
}
})
}
func (s *Server) addUnsupportedEndpoint(name string) {
endpoint := endpointName(name)
log.Debug("marking unsupported endpoint", "name", endpoint)
s.mux.HandleFunc(endpoint, func(w http.ResponseWriter, r *http.Request) {
w.WriteHeader(http.StatusNotImplemented)
})
}
func (s *Server) RegisterMobileAPI() {
for _, endpoint := range EndpointsWithResponse {
s.addEndpointWithResponse(endpoint)
}
for _, endpoint := range EndpointsNoRequest {
s.addEndpointNoRequest(endpoint)
}
for _, endpoint := range EndpointsUnsupported {
s.addUnsupportedEndpoint(endpoint)
}
}
func functionName(fn any) string {
fullName := runtime.FuncForPC(reflect.ValueOf(fn).Pointer()).Name()
parts := strings.Split(fullName, "/")
lastPart := parts[len(parts)-1]
nameParts := strings.Split(lastPart, ".")
return nameParts[len(nameParts)-1]
}
func endpointName(functionName string) string {
const base = "statusgo"
return fmt.Sprintf("/%s/%s", base, functionName)
}