163 lines
3.7 KiB
Go
163 lines
3.7 KiB
Go
package _default
|
|
|
|
import (
|
|
"context"
|
|
"git.lumeweb.com/LumeWeb/libs5-go/build"
|
|
s5net "git.lumeweb.com/LumeWeb/libs5-go/net"
|
|
"git.lumeweb.com/LumeWeb/libs5-go/service"
|
|
"github.com/julienschmidt/httprouter"
|
|
"go.sia.tech/jape"
|
|
"go.uber.org/zap"
|
|
"net"
|
|
"net/url"
|
|
"nhooyr.io/websocket"
|
|
"strings"
|
|
)
|
|
|
|
var _ service.Service = (*HTTPServiceDefault)(nil)
|
|
|
|
type P2PNodesResponse struct {
|
|
Nodes []P2PNodeResponse `json:"nodes"`
|
|
}
|
|
|
|
type P2PNodeResponse struct {
|
|
Id string `json:"id"`
|
|
Uris []string `json:"uris"`
|
|
}
|
|
|
|
type HTTPServiceDefault struct {
|
|
service.ServiceBase
|
|
}
|
|
|
|
func NewHTTP(params service.ServiceParams) *HTTPServiceDefault {
|
|
return &HTTPServiceDefault{
|
|
ServiceBase: service.NewServiceBase(params.Logger, params.Config, params.Db),
|
|
}
|
|
}
|
|
|
|
func (h *HTTPServiceDefault) GetHttpRouter(inject map[string]jape.Handler) *httprouter.Router {
|
|
routes := map[string]jape.Handler{
|
|
"GET /s5/version": h.versionHandler,
|
|
"GET /s5/p2p": h.p2pHandler,
|
|
"GET /s5/p2p/nodes": h.p2pNodesHandler,
|
|
}
|
|
|
|
for k, v := range inject {
|
|
routes[k] = v
|
|
}
|
|
|
|
return jape.Mux(routes)
|
|
}
|
|
|
|
func (h *HTTPServiceDefault) Start(ctx context.Context) error {
|
|
return nil
|
|
}
|
|
|
|
func (h *HTTPServiceDefault) Stop(ctx context.Context) error {
|
|
return nil
|
|
}
|
|
|
|
func (h *HTTPServiceDefault) Init(ctx context.Context) error {
|
|
return nil
|
|
}
|
|
|
|
func (h *HTTPServiceDefault) versionHandler(ctx jape.Context) {
|
|
_, _ = ctx.ResponseWriter.Write([]byte(build.Version))
|
|
}
|
|
func (h *HTTPServiceDefault) p2pHandler(ctx jape.Context) {
|
|
c, err := websocket.Accept(ctx.ResponseWriter, ctx.Request, nil)
|
|
if err != nil {
|
|
h.Logger().Error("error accepting websocket connection", zap.Error(err))
|
|
return
|
|
}
|
|
|
|
peer, err := s5net.CreateTransportPeer("wss", &s5net.TransportPeerConfig{
|
|
Socket: c,
|
|
Uris: []*url.URL{},
|
|
})
|
|
|
|
if err != nil {
|
|
h.Logger().Error("error creating transport peer", zap.Error(err))
|
|
err := c.Close(websocket.StatusInternalError, "the sky is falling")
|
|
if err != nil {
|
|
h.Logger().Error("error closing websocket connection", zap.Error(err))
|
|
}
|
|
return
|
|
}
|
|
|
|
// Check for reverse proxy headers
|
|
realIP := ctx.Request.Header.Get("X-Real-IP")
|
|
forwardedFor := ctx.Request.Header.Get("X-Forwarded-For")
|
|
|
|
var clientIP net.IP
|
|
if realIP != "" {
|
|
clientIP = net.ParseIP(realIP)
|
|
} else if forwardedFor != "" {
|
|
// X-Forwarded-For can contain multiple IP addresses separated by commas
|
|
// We take the first IP in the list as the client's IP
|
|
parts := strings.Split(forwardedFor, ",")
|
|
clientIP = net.ParseIP(parts[0])
|
|
}
|
|
|
|
blockConnection := func(ip net.Addr) bool {
|
|
// If we have a valid client IP from headers, use that for the loopback check
|
|
if clientIP != nil {
|
|
return clientIP.IsLoopback()
|
|
}
|
|
// Otherwise, fall back to the peer's IP
|
|
switch v := ip.(type) {
|
|
case *net.IPNet:
|
|
return v.IP.IsLoopback()
|
|
case *net.TCPAddr:
|
|
return v.IP.IsLoopback()
|
|
default:
|
|
return false
|
|
}
|
|
}
|
|
|
|
if blockConnection(peer.GetIP()) {
|
|
err := peer.End()
|
|
if err != nil {
|
|
h.Logger().Error("error ending peer", zap.Error(err))
|
|
}
|
|
return
|
|
}
|
|
|
|
if clientIP != nil {
|
|
peer.SetIP(&net.IPAddr{IP: clientIP})
|
|
}
|
|
|
|
h.Services().P2P().ConnectionTracker().Add(1)
|
|
|
|
go func() {
|
|
err := h.Services().P2P().OnNewPeer(peer, false)
|
|
if err != nil {
|
|
h.Logger().Error("error handling new peer", zap.Error(err))
|
|
}
|
|
h.Services().P2P().ConnectionTracker().Done()
|
|
}()
|
|
}
|
|
|
|
func (h *HTTPServiceDefault) p2pNodesHandler(ctx jape.Context) {
|
|
localId, err := h.Services().P2P().NodeId().ToString()
|
|
|
|
if ctx.Check("error getting local node id", err) != nil {
|
|
return
|
|
}
|
|
|
|
uris := h.Services().P2P().SelfConnectionUris()
|
|
|
|
nodeList := make([]P2PNodeResponse, len(uris))
|
|
|
|
for i, uri := range uris {
|
|
nodeList[i] = P2PNodeResponse{
|
|
Id: localId,
|
|
Uris: []string{uri.String()},
|
|
}
|
|
}
|
|
|
|
ctx.Encode(P2PNodesResponse{
|
|
Nodes: nodeList,
|
|
})
|
|
}
|