2024-09-12 14:35:25 +01:00
|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2024-09-27 16:02:18 +01:00
|
|
|
"io"
|
2024-09-12 14:35:25 +01:00
|
|
|
"net"
|
|
|
|
"net/http"
|
2024-09-27 16:02:18 +01:00
|
|
|
"strconv"
|
2024-09-12 14:35:25 +01:00
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
|
2024-09-27 16:02:18 +01:00
|
|
|
"github.com/ethereum/go-ethereum/log"
|
|
|
|
|
2024-09-12 14:35:25 +01:00
|
|
|
"github.com/gorilla/websocket"
|
|
|
|
|
2024-09-27 16:02:18 +01:00
|
|
|
"github.com/pkg/errors"
|
2024-09-12 14:35:25 +01:00
|
|
|
|
|
|
|
"github.com/status-im/status-go/signal"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Server struct {
|
|
|
|
server *http.Server
|
2024-09-27 16:02:18 +01:00
|
|
|
listener net.Listener
|
|
|
|
mux *http.ServeMux
|
2024-09-12 14:35:25 +01:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2024-09-27 16:02:18 +01:00
|
|
|
func (s *Server) Port() (int, error) {
|
|
|
|
_, portString, err := net.SplitHostPort(s.address)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
return strconv.Atoi(portString)
|
|
|
|
}
|
|
|
|
|
2024-09-12 14:35:25 +01:00
|
|
|
func (s *Server) Setup() {
|
|
|
|
signal.SetMobileSignalHandler(s.signalHandler)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) signalHandler(data []byte) {
|
|
|
|
s.lock.Lock()
|
|
|
|
defer s.lock.Unlock()
|
|
|
|
|
2024-12-06 17:29:11 +08:00
|
|
|
deleteConnection := func(connection *websocket.Conn) {
|
|
|
|
delete(s.connections, connection)
|
|
|
|
err := connection.Close()
|
|
|
|
if err != nil {
|
|
|
|
log.Error("failed to close connection", "error", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-12 14:35:25 +01:00
|
|
|
for connection := range s.connections {
|
2024-12-06 17:29:11 +08:00
|
|
|
err := connection.SetWriteDeadline(time.Now().Add(5 * time.Second))
|
|
|
|
if err != nil {
|
|
|
|
log.Error("failed to set write deadline", "error", err)
|
|
|
|
deleteConnection(connection)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
err = connection.WriteMessage(websocket.TextMessage, data)
|
2024-09-12 14:35:25 +01:00
|
|
|
if err != nil {
|
2024-12-06 17:29:11 +08:00
|
|
|
log.Error("failed to write signal message", "error", err)
|
|
|
|
deleteConnection(connection)
|
2024-09-12 14:35:25 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) Listen(address string) error {
|
|
|
|
if s.server != nil {
|
|
|
|
return errors.New("server already started")
|
|
|
|
}
|
|
|
|
|
2024-09-27 16:02:18 +01:00
|
|
|
_, _, err := net.SplitHostPort(address)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "invalid address")
|
|
|
|
}
|
|
|
|
|
2024-09-12 14:35:25 +01:00
|
|
|
s.server = &http.Server{
|
|
|
|
Addr: address,
|
|
|
|
ReadHeaderTimeout: 5 * time.Second,
|
|
|
|
}
|
|
|
|
|
2024-09-27 16:02:18 +01:00
|
|
|
s.mux = http.NewServeMux()
|
|
|
|
s.mux.HandleFunc("/signals", s.signals)
|
|
|
|
s.server.Handler = s.mux
|
2024-09-12 14:35:25 +01:00
|
|
|
|
2024-09-27 16:02:18 +01:00
|
|
|
s.listener, err = net.Listen("tcp", address)
|
2024-09-12 14:35:25 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2024-09-27 16:02:18 +01:00
|
|
|
s.address = s.listener.Addr().String()
|
2024-09-12 14:35:25 +01:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-09-27 16:02:18 +01:00
|
|
|
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)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-12 14:35:25 +01:00
|
|
|
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
|
|
|
|
}
|
2024-12-06 17:29:11 +08:00
|
|
|
log.Debug("new websocket connection")
|
2024-09-12 14:35:25 +01:00
|
|
|
|
|
|
|
s.connections[connection] = struct{}{}
|
|
|
|
}
|
2024-09-27 16:02:18 +01:00
|
|
|
|
|
|
|
func (s *Server) addEndpointWithResponse(name string, handler func(string) string) {
|
|
|
|
log.Debug("adding endpoint", "name", name)
|
|
|
|
s.mux.HandleFunc(name, 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))
|
|
|
|
|
|
|
|
s.setHeaders(name, w)
|
|
|
|
|
|
|
|
_, err = w.Write([]byte(response))
|
|
|
|
if err != nil {
|
|
|
|
log.Error("failed to write response: %w", err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) addEndpointNoRequest(name string, handler func() string) {
|
|
|
|
log.Debug("adding endpoint", "name", name)
|
|
|
|
s.mux.HandleFunc(name, func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
response := handler()
|
|
|
|
|
|
|
|
s.setHeaders(name, w)
|
|
|
|
|
|
|
|
_, err := w.Write([]byte(response))
|
|
|
|
if err != nil {
|
|
|
|
log.Error("failed to write response: %w", err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) addUnsupportedEndpoint(name string) {
|
|
|
|
log.Debug("marking unsupported endpoint", "name", name)
|
|
|
|
s.mux.HandleFunc(name, func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
w.WriteHeader(http.StatusNotImplemented)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) RegisterMobileAPI() {
|
|
|
|
for name, endpoint := range EndpointsWithRequest {
|
|
|
|
s.addEndpointWithResponse(name, endpoint)
|
|
|
|
}
|
|
|
|
for name, endpoint := range EndpointsWithoutRequest {
|
|
|
|
s.addEndpointNoRequest(name, endpoint)
|
|
|
|
}
|
|
|
|
for _, name := range EndpointsUnsupported {
|
|
|
|
s.addUnsupportedEndpoint(name)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) setHeaders(name string, w http.ResponseWriter) {
|
|
|
|
if _, ok := EndpointsDeprecated[name]; ok {
|
|
|
|
w.Header().Set("Deprecation", "true")
|
|
|
|
}
|
|
|
|
w.Header().Set("Content-Type", "application/json")
|
|
|
|
}
|