mirror of
https://github.com/asmogo/nws.git
synced 2025-01-18 18:11:33 +00:00
348 lines
12 KiB
Go
348 lines
12 KiB
Go
package exit
|
|
|
|
import (
|
|
"encoding/base32"
|
|
"encoding/hex"
|
|
"fmt"
|
|
"log/slog"
|
|
"net"
|
|
"strings"
|
|
|
|
"github.com/asmogo/nws/config"
|
|
"github.com/asmogo/nws/netstr"
|
|
"github.com/asmogo/nws/protocol"
|
|
"github.com/asmogo/nws/socks5"
|
|
"github.com/btcsuite/btcd/btcec/v2"
|
|
"github.com/btcsuite/btcd/btcec/v2/schnorr"
|
|
"github.com/nbd-wtf/go-nostr"
|
|
"github.com/nbd-wtf/go-nostr/nip04"
|
|
"github.com/nbd-wtf/go-nostr/nip19"
|
|
"github.com/puzpuzpuz/xsync/v3"
|
|
"golang.org/x/net/context"
|
|
)
|
|
|
|
const (
|
|
startingReverseProxyMessage = "starting exit node with https reverse proxy"
|
|
generateKeyMessage = "Generated new private key. Please set your environment using the new key, otherwise your key will be lost."
|
|
)
|
|
|
|
// Exit represents a structure that holds information related to an exit node.
|
|
type Exit struct {
|
|
|
|
// pool represents a pool of relays and manages the subscription to incoming events from relays.
|
|
pool *nostr.SimplePool
|
|
|
|
// config is a field in the Exit struct that holds information related to exit node configuration.
|
|
config *config.ExitConfig
|
|
|
|
// relays represents a slice of *nostr.Relay, which contains information about the relay nodes used by the Exit node.
|
|
// Todo -- check if this is deprecated
|
|
relays []*nostr.Relay
|
|
|
|
// nostrConnectionMap is a concurrent map used to store connections for the Exit node.
|
|
// It is used to establish and maintain connections between the Exit node and the backend host.
|
|
nostrConnectionMap *xsync.MapOf[string, *netstr.NostrConnection]
|
|
|
|
// mutexMap is a field in the Exit struct that represents a map used for synchronizing access to resources based on a string key.
|
|
mutexMap *MutexMap
|
|
|
|
// incomingChannel represents a channel used to receive incoming events from relays.
|
|
incomingChannel chan nostr.IncomingEvent
|
|
|
|
nprofile string
|
|
publicKey string
|
|
}
|
|
|
|
// NewExit creates a new Exit node with the provided context and config.
|
|
// This function will currently panic if there is an error while creating the Exit node.
|
|
func NewExit(ctx context.Context, exitNodeConfig *config.ExitConfig) *Exit {
|
|
// generate new private key if it is not set
|
|
if exitNodeConfig.NostrPrivateKey == "" {
|
|
// generate new private key
|
|
exitNodeConfig.NostrPrivateKey = nostr.GeneratePrivateKey()
|
|
slog.Warn(generateKeyMessage, "key", exitNodeConfig.NostrPrivateKey)
|
|
}
|
|
// get public key from private key
|
|
pubKey, err := nostr.GetPublicKey(exitNodeConfig.NostrPrivateKey)
|
|
slog.Info("using public key", "key", pubKey)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
// encode profile
|
|
profile, err := nip19.EncodeProfile(pubKey,
|
|
exitNodeConfig.NostrRelays)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
// create a new pool
|
|
pool := nostr.NewSimplePool(ctx)
|
|
|
|
exit := &Exit{
|
|
nostrConnectionMap: xsync.NewMapOf[string, *netstr.NostrConnection](),
|
|
pool: pool,
|
|
mutexMap: NewMutexMap(),
|
|
publicKey: pubKey,
|
|
nprofile: profile,
|
|
}
|
|
// start reverse proxy if https port is set
|
|
if exitNodeConfig.HttpsPort != 0 {
|
|
exitNodeConfig.BackendHost = fmt.Sprintf(":%d", exitNodeConfig.HttpsPort)
|
|
go func(cfg *config.ExitConfig) {
|
|
slog.Info(startingReverseProxyMessage, "port", cfg.HttpsPort)
|
|
err := exit.StartReverseProxy(cfg.HttpsTarget, cfg.HttpsPort)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
}(exitNodeConfig)
|
|
}
|
|
// set config
|
|
exit.config = exitNodeConfig
|
|
// add relays to the pool
|
|
for _, relayUrl := range exitNodeConfig.NostrRelays {
|
|
relay, err := exit.pool.EnsureRelay(relayUrl)
|
|
if err != nil {
|
|
fmt.Println(err)
|
|
continue
|
|
}
|
|
exit.relays = append(exit.relays, relay)
|
|
fmt.Printf("added relay connection to %s\n", relayUrl)
|
|
|
|
}
|
|
domain, err := exit.getDomain()
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
slog.Info("created exit node", "profile", profile, "domain", domain)
|
|
// setup subscriptions
|
|
err = exit.setSubscriptions(ctx)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
err = exit.announceExitNode(ctx)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return exit
|
|
}
|
|
|
|
// getDomain returns the domain string used by the Exit node for communication with the Nostr relays.
|
|
// It concatenates the relay URLs using base32 encoding with no padding, separated by dots.
|
|
// The resulting domain is then appended with the base32 encoded public key obtained using the configured Nostr private key.
|
|
// The final domain string is converted to lowercase and returned.
|
|
// If any errors occur during the process, they are returned along with an
|
|
func (e *Exit) getDomain() (string, error) {
|
|
var domain string
|
|
// first lets build the subdomains
|
|
for _, relayUrl := range e.config.NostrRelays {
|
|
if domain == "" {
|
|
domain = base32.HexEncoding.WithPadding(base32.NoPadding).EncodeToString([]byte(relayUrl))
|
|
} else {
|
|
domain = fmt.Sprintf("%s.%s", domain, base32.HexEncoding.WithPadding(base32.NoPadding).EncodeToString([]byte(relayUrl)))
|
|
}
|
|
}
|
|
// create base32 encoded public key
|
|
decoded, err := GetPublicKeyBase32(e.config.NostrPrivateKey)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
// use public key as host. add TLD
|
|
domain = strings.ToLower(fmt.Sprintf("%s.%s.nostr", domain, decoded))
|
|
return domain, nil
|
|
}
|
|
|
|
// GetPublicKeyBase32 decodes the private key string from hexadecimal format
|
|
// and returns the base32 encoded public key obtained using the provided private key.
|
|
// The base32 encoding has no padding. If there is an error decoding the private key
|
|
// or generating the public key, an error is returned.
|
|
//
|
|
// Parameters:
|
|
// - sk: The private key string in hexadecimal format
|
|
//
|
|
// Returns:
|
|
// - The base32 encoded public key as a string
|
|
// - Any error that occurred during the process
|
|
func GetPublicKeyBase32(sk string) (string, error) {
|
|
b, err := hex.DecodeString(sk)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
_, pk := btcec.PrivKeyFromBytes(b)
|
|
return base32.HexEncoding.WithPadding(base32.NoPadding).EncodeToString(schnorr.SerializePubKey(pk)), nil
|
|
}
|
|
|
|
// setSubscriptions sets up subscriptions for the Exit node to receive incoming events from the specified relays.
|
|
// It first obtains the public key using the configured Nostr private key.
|
|
// Then it calls the `handleSubscription` method to open a subscription to the relays with the specified filters.
|
|
// This method runs in a separate goroutine and continuously handles the incoming events by calling the `processMessage` method.
|
|
// If the context is canceled before the subscription is established, it returns the context error.
|
|
// If any errors occur during the process, they are returned.
|
|
// This method should be called once when starting the Exit node.
|
|
func (e *Exit) setSubscriptions(ctx context.Context) error {
|
|
pubKey, err := nostr.GetPublicKey(e.config.NostrPrivateKey)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
now := nostr.Now()
|
|
if err = e.handleSubscription(ctx, pubKey, now); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
|
|
}
|
|
|
|
// handleSubscription handles the subscription to incoming events from relays based on the provided filters.
|
|
// It sets up the incoming event channel and starts a goroutine to handle the events.
|
|
// It returns an error if there is any issue with the subscription.
|
|
func (e *Exit) handleSubscription(ctx context.Context, pubKey string, since nostr.Timestamp) error {
|
|
incomingEventChannel := e.pool.SubMany(ctx, e.config.NostrRelays, nostr.Filters{
|
|
{Kinds: []int{protocol.KindEphemeralEvent},
|
|
Since: &since,
|
|
Tags: nostr.TagMap{
|
|
"p": []string{pubKey},
|
|
},
|
|
},
|
|
})
|
|
e.incomingChannel = incomingEventChannel
|
|
return nil
|
|
}
|
|
|
|
// ListenAndServe handles incoming events from the subscription channel.
|
|
// It processes each event by calling the processMessage method, as long as the event is not nil.
|
|
// If the context is canceled (ctx.Done() receives a value), the method returns.
|
|
func (e *Exit) ListenAndServe(ctx context.Context) {
|
|
for {
|
|
select {
|
|
case event := <-e.incomingChannel:
|
|
slog.Debug("received event", "event", event)
|
|
if event.Relay == nil {
|
|
continue
|
|
}
|
|
go e.processMessage(ctx, event)
|
|
case <-ctx.Done():
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
// processMessage decrypts and unmarshals the incoming event message, and then
|
|
// routes the message to the appropriate handler based on its protocol type.
|
|
func (e *Exit) processMessage(ctx context.Context, msg nostr.IncomingEvent) {
|
|
sharedKey, err := nip04.ComputeSharedSecret(msg.PubKey, e.config.NostrPrivateKey)
|
|
if err != nil {
|
|
return
|
|
}
|
|
decodedMessage, err := nip04.Decrypt(msg.Content, sharedKey)
|
|
if err != nil {
|
|
return
|
|
}
|
|
protocolMessage, err := protocol.UnmarshalJSON([]byte(decodedMessage))
|
|
if err != nil {
|
|
slog.Error("could not unmarshal message")
|
|
return
|
|
}
|
|
destination, err := protocol.Parse(protocolMessage.Destination)
|
|
if err != nil {
|
|
return
|
|
}
|
|
if destination.TLD == "nostr" {
|
|
protocolMessage.Destination = e.config.BackendHost
|
|
}
|
|
switch protocolMessage.Type {
|
|
case protocol.MessageConnect:
|
|
e.handleConnect(ctx, msg, protocolMessage)
|
|
case protocol.MessageConnectReverse:
|
|
e.handleConnectReverse(protocolMessage)
|
|
case protocol.MessageTypeSocks5:
|
|
e.handleSocks5ProxyMessage(msg, protocolMessage)
|
|
}
|
|
}
|
|
|
|
// handleConnect handles the connection for the given message and protocol message.
|
|
// It locks the mutex for the protocol message key, encodes the receiver's profile,
|
|
// creates a new connection with the provided context and options, and establishes
|
|
// a connection to the backend host.
|
|
// If the connection cannot be established, it logs an error and returns.
|
|
// It then stores the connection in the nostrConnectionMap and creates two goroutines
|
|
// to proxy the data between the connection and the backend.
|
|
func (e *Exit) handleConnect(
|
|
ctx context.Context,
|
|
msg nostr.IncomingEvent,
|
|
protocolMessage *protocol.Message) {
|
|
e.mutexMap.Lock(protocolMessage.Key.String())
|
|
defer e.mutexMap.Unlock(protocolMessage.Key.String())
|
|
receiver, err := nip19.EncodeProfile(msg.PubKey, []string{msg.Relay.String()})
|
|
if err != nil {
|
|
return
|
|
}
|
|
connection := netstr.NewConnection(
|
|
ctx,
|
|
netstr.WithPrivateKey(e.config.NostrPrivateKey),
|
|
netstr.WithDst(receiver),
|
|
netstr.WithUUID(protocolMessage.Key),
|
|
)
|
|
|
|
var dst net.Conn
|
|
dst, err = net.Dial("tcp", protocolMessage.Destination)
|
|
if err != nil {
|
|
slog.Error("could not connect to backend", "error", err)
|
|
return
|
|
}
|
|
|
|
e.nostrConnectionMap.Store(protocolMessage.Key.String(), connection)
|
|
|
|
go socks5.Proxy(dst, connection, nil)
|
|
go socks5.Proxy(connection, dst, nil)
|
|
}
|
|
|
|
func (e *Exit) handleConnectReverse(protocolMessage *protocol.Message) {
|
|
|
|
e.mutexMap.Lock(protocolMessage.Key.String())
|
|
defer e.mutexMap.Unlock(protocolMessage.Key.String())
|
|
connection, err := net.Dial("tcp", protocolMessage.EntryPublicAddress)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
_, err = connection.Write([]byte(protocolMessage.Key.String()))
|
|
if err != nil {
|
|
return
|
|
}
|
|
// read single byte from the connection
|
|
readbuffer := make([]byte, 1)
|
|
_, err = connection.Read(readbuffer)
|
|
if err != nil {
|
|
return
|
|
}
|
|
if readbuffer[0] != 1 {
|
|
return
|
|
}
|
|
var dst net.Conn
|
|
dst, err = net.Dial("tcp", protocolMessage.Destination)
|
|
if err != nil {
|
|
slog.Error("could not connect to backend", "error", err)
|
|
return
|
|
}
|
|
|
|
go socks5.Proxy(dst, connection, nil)
|
|
go socks5.Proxy(connection, dst, nil)
|
|
}
|
|
|
|
// handleSocks5ProxyMessage handles the SOCKS5 proxy message by writing it to the destination connection.
|
|
// If the destination connection does not exist, the function returns without doing anything.
|
|
//
|
|
// Parameters:
|
|
// - msg: The incoming event containing the SOCKS5 proxy message.
|
|
// - protocolMessage: The protocol message associated with the incoming event.
|
|
func (e *Exit) handleSocks5ProxyMessage(
|
|
msg nostr.IncomingEvent,
|
|
protocolMessage *protocol.Message,
|
|
) {
|
|
e.mutexMap.Lock(protocolMessage.Key.String())
|
|
defer e.mutexMap.Unlock(protocolMessage.Key.String())
|
|
dst, ok := e.nostrConnectionMap.Load(protocolMessage.Key.String())
|
|
if !ok {
|
|
return
|
|
}
|
|
dst.WriteNostrEvent(msg)
|
|
}
|