mirror of
https://github.com/asmogo/nws.git
synced 2025-01-18 18:11:33 +00:00
436 lines
14 KiB
Go
436 lines
14 KiB
Go
package netstr
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"encoding/base32"
|
|
"encoding/base64"
|
|
"encoding/hex"
|
|
"errors"
|
|
"fmt"
|
|
"log/slog"
|
|
"net"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/asmogo/nws/protocol"
|
|
"github.com/btcsuite/btcd/btcec/v2/schnorr"
|
|
"github.com/google/uuid"
|
|
"github.com/nbd-wtf/go-nostr"
|
|
"github.com/nbd-wtf/go-nostr/nip04"
|
|
"github.com/nbd-wtf/go-nostr/nip19"
|
|
"github.com/samber/lo"
|
|
)
|
|
|
|
// NostrConnection implements the net.Conn interface.
|
|
// It is used to establish a connection to Nostr relays.
|
|
// It provides methods for reading and writing data.
|
|
type NostrConnection struct {
|
|
// uuid is a field of type uuid.UUID in the NostrConnection struct.
|
|
uuid uuid.UUID
|
|
// ctx is a field of type context.Context in the NostrConnection struct.
|
|
ctx context.Context
|
|
// cancel is a field of type context.CancelFunc in the NostrConnection struct.
|
|
cancel context.CancelFunc
|
|
|
|
// readBuffer is a field of type `bytes.Buffer` in the `NostrConnection` struct.
|
|
// It is used to store the decrypted message from incoming events.
|
|
// The `handleSubscription` method continuously listens for events on the subscription channel,
|
|
// decrypts the event content, and writes the decrypted message to the `readBuffer`.
|
|
readBuffer bytes.Buffer
|
|
|
|
// private key of the connection
|
|
privateKey string
|
|
|
|
// NostrConnection represents a connection object.
|
|
pool *nostr.SimplePool
|
|
|
|
// dst is a field that represents the destination address for the Nostr connection configuration.
|
|
dst string
|
|
|
|
// subscriptionChan is a channel of type protocol.IncomingEvent.
|
|
// It is used to write incoming events which will be read and processed by the Read method.
|
|
subscriptionChan chan nostr.IncomingEvent
|
|
|
|
// readIDs represents the list of event IDs that have been read by the NostrConnection object.
|
|
readIDs []string
|
|
|
|
// writeIDs is a field of type []string in the NostrConnection struct.
|
|
// It stores the IDs of the events that have been written to the connection.
|
|
// This field is used to check if an event has already been written and avoid duplicate writes.
|
|
writeIDs []string
|
|
|
|
// sentBytes is a field that stores the bytes of data that have been sent by the connection.
|
|
sentBytes [][]byte
|
|
|
|
// sub represents a boolean value indicating if a connection should subscribe to a response when writing.
|
|
sub bool
|
|
defaultRelays []string
|
|
targetPublicKey string
|
|
}
|
|
|
|
var errContextCanceled = errors.New("context canceled")
|
|
|
|
// WriteNostrEvent writes the incoming event to the subscription channel of the NostrConnection.
|
|
// The subscription channel is used by the Read method to read events and handle them.
|
|
// Parameters:
|
|
// - event: The incoming event to be written to the subscription channel.
|
|
func (nc *NostrConnection) WriteNostrEvent(event nostr.IncomingEvent) {
|
|
nc.subscriptionChan <- event
|
|
}
|
|
|
|
// NewConnection creates a new NostrConnection object with the provided context and options.
|
|
// It initializes the config with default values, processes the options to customize the config,
|
|
// and creates a new NostrConnection object using the config.
|
|
// If an uuid is provided in the options, it is assigned to the NostrConnection object.
|
|
// The NostrConnection object is then returned.
|
|
func NewConnection(ctx context.Context, opts ...NostrConnOption) *NostrConnection {
|
|
ctx, c := context.WithCancel(ctx)
|
|
nostrConnection := &NostrConnection{
|
|
pool: nostr.NewSimplePool(ctx),
|
|
ctx: ctx,
|
|
cancel: c,
|
|
subscriptionChan: make(chan nostr.IncomingEvent),
|
|
readIDs: make([]string, 0),
|
|
sentBytes: make([][]byte, 0),
|
|
}
|
|
for _, opt := range opts {
|
|
opt(nostrConnection)
|
|
}
|
|
|
|
return nostrConnection
|
|
}
|
|
|
|
// Read reads data from the connection. The data is decrypted and returned in the provided byte slice.
|
|
// If there is no data available, Read blocks until data arrives or the context is canceled.
|
|
// If the context is canceled before data is received, Read returns an error.
|
|
//
|
|
// The number of bytes read is returned as n and any error encountered is returned as err.
|
|
// The content of the decrypted message is then copied to the provided byte slice b.
|
|
func (nc *NostrConnection) Read(b []byte) (int, error) {
|
|
return nc.handleNostrRead(b)
|
|
}
|
|
|
|
// handleNostrRead reads the incoming events from the subscription channel and processes them.
|
|
// It checks if the event has already been read, decrypts the content using the shared key,
|
|
// unmarshals the decoded message and copies the content into the provided byte slice.
|
|
// It returns the number of bytes copied and any error encountered.
|
|
// If the context is canceled, it returns an error with "context canceled" message.
|
|
func (nc *NostrConnection) handleNostrRead(buffer []byte) (int, error) {
|
|
for {
|
|
select {
|
|
case event := <-nc.subscriptionChan:
|
|
if event.Relay == nil {
|
|
return 0, nil
|
|
}
|
|
// check if we have already read this event
|
|
if lo.Contains(nc.readIDs, event.ID) {
|
|
continue
|
|
}
|
|
nc.readIDs = append(nc.readIDs, event.ID)
|
|
sharedKey, err := nip04.ComputeSharedSecret(event.PubKey, nc.privateKey)
|
|
if err != nil {
|
|
return 0, fmt.Errorf("could not compute shared key: %w", err)
|
|
}
|
|
decodedMessage, err := nip04.Decrypt(event.Content, sharedKey)
|
|
if err != nil {
|
|
return 0, fmt.Errorf("could not decrypt message: %w", err)
|
|
}
|
|
message, err := protocol.UnmarshalJSON([]byte(decodedMessage))
|
|
if err != nil {
|
|
return 0, fmt.Errorf("could not unmarshal message: %w", err)
|
|
}
|
|
slog.Debug("reading",
|
|
slog.String("event", event.ID),
|
|
slog.String("content", base64.StdEncoding.EncodeToString(message.Data)),
|
|
)
|
|
n := copy(buffer, message.Data)
|
|
return n, nil
|
|
case <-nc.ctx.Done():
|
|
return 0, errContextCanceled
|
|
default:
|
|
time.Sleep(time.Millisecond * 100)
|
|
}
|
|
}
|
|
}
|
|
|
|
// Write writes data to the connection.
|
|
// It delegates the writing logic to handleNostrWrite method.
|
|
// The number of bytes written and error (if any) are returned.
|
|
func (nc *NostrConnection) Write(b []byte) (int, error) {
|
|
return nc.handleNostrWrite(b)
|
|
}
|
|
|
|
// Go lang
|
|
func (nc *NostrConnection) handleNostrWrite(buffer []byte) (int, error) {
|
|
if nc.ctx.Err() != nil {
|
|
return 0, fmt.Errorf("context canceled: %w", nc.ctx.Err())
|
|
}
|
|
publicKey, relays, err := nc.parseDestination()
|
|
if err != nil {
|
|
return 0, fmt.Errorf("could not parse destination: %w", err)
|
|
}
|
|
signer, err := protocol.NewEventSigner(nc.privateKey)
|
|
if err != nil {
|
|
return 0, fmt.Errorf("could not create event signer: %w", err)
|
|
}
|
|
signedEvent, err := nc.createSignedEvent(signer, buffer, publicKey, relays)
|
|
if err != nil {
|
|
return 0, fmt.Errorf("could not create signed event: %w", err)
|
|
}
|
|
err = nc.publishEventToRelays(signedEvent, relays)
|
|
if err != nil {
|
|
return 0, fmt.Errorf("could not publish event to relays: %w", err)
|
|
}
|
|
nc.appendSentBytes(buffer)
|
|
slog.Debug("writing",
|
|
slog.String("event", signedEvent.ID),
|
|
slog.String("content", base64.StdEncoding.EncodeToString(buffer)),
|
|
)
|
|
return len(buffer), nil
|
|
}
|
|
|
|
func (nc *NostrConnection) createSignedEvent(
|
|
signer *protocol.EventSigner,
|
|
b []byte,
|
|
publicKey string,
|
|
relays []string,
|
|
) (nostr.Event, error) {
|
|
opts := []protocol.MessageOption{
|
|
protocol.WithUUID(nc.uuid),
|
|
protocol.WithType(protocol.MessageTypeSocks5),
|
|
protocol.WithDestination(nc.dst),
|
|
protocol.WithData(b),
|
|
}
|
|
signedEvent, err := signer.CreateSignedEvent(
|
|
publicKey,
|
|
protocol.KindEphemeralEvent,
|
|
nostr.Tags{nostr.Tag{"p", publicKey}},
|
|
opts...,
|
|
)
|
|
if err != nil {
|
|
return signedEvent, fmt.Errorf("could not create signed event: %w", err)
|
|
}
|
|
if lo.Contains(nc.writeIDs, signedEvent.ID) {
|
|
slog.Info("event already sent", slog.String("event", signedEvent.ID))
|
|
return signedEvent, nil
|
|
}
|
|
nc.writeIDs = append(nc.writeIDs, signedEvent.ID)
|
|
if nc.sub {
|
|
nc.sub = false
|
|
now := nostr.Now()
|
|
incomingEventChannel := nc.pool.SubMany(nc.ctx, relays,
|
|
nostr.Filters{
|
|
{
|
|
Kinds: []int{protocol.KindEphemeralEvent},
|
|
Authors: []string{publicKey},
|
|
Since: &now,
|
|
Tags: nostr.TagMap{
|
|
"p": []string{signedEvent.PubKey},
|
|
},
|
|
},
|
|
},
|
|
)
|
|
nc.subscriptionChan = incomingEventChannel
|
|
}
|
|
return signedEvent, nil
|
|
}
|
|
|
|
func (nc *NostrConnection) publishEventToRelays(ev nostr.Event, relays []string) error {
|
|
for _, responseRelay := range relays {
|
|
var relay *nostr.Relay
|
|
relay, err := nc.pool.EnsureRelay(responseRelay)
|
|
if err != nil {
|
|
return fmt.Errorf("could not ensure relay: %w", err)
|
|
}
|
|
err = relay.Publish(nc.ctx, ev)
|
|
if err != nil {
|
|
return fmt.Errorf("could not publish event to relay: %w", err)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (nc *NostrConnection) appendSentBytes(b []byte) {
|
|
nc.sentBytes = append(nc.sentBytes, b)
|
|
}
|
|
|
|
// parseDestination takes a destination string and returns a public key and relays.
|
|
// The destination can be "npub" or "nprofile".
|
|
// If the prefix is "npub", the public key is extracted.
|
|
// If the prefix is "nprofile", the public key and relays are extracted.
|
|
// Returns the public key, relays (if any), and any error encountered.
|
|
func (nc *NostrConnection) parseDestination() (string, []string, error) {
|
|
// check if destination ends with .nostr
|
|
if strings.HasPrefix(nc.dst, "npub") || strings.HasPrefix(nc.dst, "nprofile") {
|
|
// destination can be npub or nprofile
|
|
prefix, pubKey, err := nip19.Decode(nc.dst)
|
|
|
|
if err != nil {
|
|
return "", nil, fmt.Errorf("could not decode destination: %w", err)
|
|
}
|
|
|
|
var relays []string
|
|
var publicKey string
|
|
|
|
switch prefix {
|
|
case "npub":
|
|
publicKey = pubKey.(string)
|
|
case "nprofile":
|
|
profilePointer := pubKey.(nostr.ProfilePointer)
|
|
publicKey = profilePointer.PublicKey
|
|
relays = profilePointer.Relays
|
|
}
|
|
return publicKey, relays, nil
|
|
}
|
|
return nc.parseDestinationDomain()
|
|
}
|
|
|
|
func (nc *NostrConnection) parseDestinationDomain() (string, []string, error) {
|
|
url, err := protocol.Parse(nc.dst)
|
|
if err != nil {
|
|
return "", nil, err
|
|
}
|
|
if !url.IsDomain {
|
|
// try to parse as ip
|
|
ip := net.ParseIP(url.Name)
|
|
if ip != nil {
|
|
return nc.targetPublicKey, nc.defaultRelays, nil
|
|
}
|
|
return "", nil, fmt.Errorf("destination is not a domain")
|
|
|
|
}
|
|
if url.TLD != "nostr" {
|
|
// parse public key
|
|
/*pubKey,err := nostr.GetPublicKey(nc.privateKey)
|
|
if err != nil {
|
|
return "", nil, err
|
|
}*/
|
|
return nc.targetPublicKey, nc.defaultRelays, nil
|
|
}
|
|
subdomains := make([]string, 0)
|
|
split := strings.Split(url.SubName, ".")
|
|
for _, subdomain := range split {
|
|
decodedSubDomain, err := base32.HexEncoding.WithPadding(base32.NoPadding).DecodeString(strings.ToUpper(subdomain))
|
|
if err != nil {
|
|
continue
|
|
}
|
|
subdomains = append(subdomains, string(decodedSubDomain))
|
|
}
|
|
|
|
// base32 decode the subdomain
|
|
decodedPubKey, err := base32.HexEncoding.WithPadding(base32.NoPadding).DecodeString(strings.ToUpper(url.Name))
|
|
|
|
if err != nil {
|
|
return "", nil, err
|
|
}
|
|
pk, err := schnorr.ParsePubKey(decodedPubKey)
|
|
if err != nil {
|
|
return "", nil, err
|
|
}
|
|
// todo -- check if this is correct
|
|
return hex.EncodeToString(pk.SerializeCompressed())[2:], subdomains, nil
|
|
}
|
|
|
|
func (nc *NostrConnection) Close() error {
|
|
nc.cancel()
|
|
return nil
|
|
}
|
|
|
|
func (nc *NostrConnection) LocalAddr() net.Addr {
|
|
return &net.TCPAddr{IP: net.ParseIP("127.0.0.1"), Port: 9333}
|
|
}
|
|
|
|
func (nc *NostrConnection) RemoteAddr() net.Addr {
|
|
return &net.TCPAddr{IP: net.ParseIP("127.0.0.1"), Port: 0}
|
|
}
|
|
|
|
func (nc *NostrConnection) SetDeadline(_ time.Time) error {
|
|
return nil
|
|
}
|
|
|
|
func (nc *NostrConnection) SetReadDeadline(_ time.Time) error {
|
|
return nil
|
|
}
|
|
|
|
func (nc *NostrConnection) SetWriteDeadline(_ time.Time) error {
|
|
return nil
|
|
}
|
|
|
|
// NostrConnOption is a functional option type for configuring NostrConnConfig.
|
|
type NostrConnOption func(*NostrConnection)
|
|
|
|
// WithPrivateKey sets the private key for the NostrConnConfig.
|
|
func WithPrivateKey(privateKey string) NostrConnOption {
|
|
return func(config *NostrConnection) {
|
|
config.privateKey = privateKey
|
|
}
|
|
}
|
|
|
|
// WithPrivateKey sets the private key for the NostrConnConfig.
|
|
func WithDefaultRelays(defaultRelays []string) NostrConnOption {
|
|
return func(config *NostrConnection) {
|
|
config.defaultRelays = defaultRelays
|
|
}
|
|
}
|
|
|
|
// WithTargetPublicKey sets the private key for the NostrConnConfig.
|
|
func WithTargetPublicKey(pubKey string) NostrConnOption {
|
|
return func(config *NostrConnection) {
|
|
config.targetPublicKey = pubKey
|
|
}
|
|
}
|
|
|
|
// WithSub is a function that returns a NostrConnOption. When this option is applied
|
|
// to a NostrConnConfig, it sets the 'sub' field to true, indicating that
|
|
// the connection will handle subscriptions.
|
|
func WithSub(...bool) NostrConnOption {
|
|
return func(connection *NostrConnection) {
|
|
connection.sub = true
|
|
go connection.handleSubscription()
|
|
}
|
|
}
|
|
|
|
// WithDst is a NostrConnOption function that sets the destination address for the Nostr connection configuration.
|
|
// It takes a string parameter `dst` and updates the `config.dst` field accordingly.
|
|
func WithDst(dst string) NostrConnOption {
|
|
return func(connection *NostrConnection) {
|
|
connection.dst = dst
|
|
}
|
|
}
|
|
|
|
// WithUUID sets the UUID option for creating a NostrConnConfig.
|
|
// It assigns the provided UUID to the config's uuid field.
|
|
func WithUUID(uuid uuid.UUID) NostrConnOption {
|
|
return func(connection *NostrConnection) {
|
|
connection.uuid = uuid
|
|
}
|
|
}
|
|
|
|
// handleSubscription handles the subscription channel for incoming events.
|
|
// It continuously listens for events on the subscription channel and performs necessary operations.
|
|
// If the event has a valid relay, it computes the shared key and decrypts the event content.
|
|
// The decrypted message is then written to the read buffer.
|
|
// If the context is canceled, the method returns.
|
|
func (nc *NostrConnection) handleSubscription() {
|
|
for {
|
|
select {
|
|
case event := <-nc.subscriptionChan:
|
|
if event.Relay == nil {
|
|
continue
|
|
}
|
|
sharedKey, err := nip04.ComputeSharedSecret(event.PubKey, nc.privateKey)
|
|
if err != nil {
|
|
continue
|
|
}
|
|
decodedMessage, err := nip04.Decrypt(event.Content, sharedKey)
|
|
if err != nil {
|
|
continue
|
|
}
|
|
nc.readBuffer.Write([]byte(decodedMessage))
|
|
case <-nc.ctx.Done():
|
|
return
|
|
}
|
|
}
|
|
}
|