go-nostr/nip46/nip46.go

91 lines
2.0 KiB
Go
Raw Permalink Normal View History

2024-01-09 16:55:00 -03:00
package nip46
import (
"encoding/json"
"fmt"
2024-03-01 15:54:28 -03:00
"regexp"
2024-01-09 16:55:00 -03:00
"github.com/nbd-wtf/go-nostr"
"github.com/nbd-wtf/go-nostr/nip04"
2024-05-20 09:20:39 -03:00
"github.com/nbd-wtf/go-nostr/nip44"
2024-01-09 16:55:00 -03:00
)
2024-03-01 15:54:28 -03:00
var BUNKER_REGEX = regexp.MustCompile(`^bunker:\/\/([0-9a-f]{64})\??([?\/\w:.=&%]*)$`)
2024-01-09 16:55:00 -03:00
type Request struct {
ID string `json:"id"`
Method string `json:"method"`
Params []string `json:"params"`
}
type Response struct {
ID string `json:"id"`
Error string `json:"error,omitempty"`
Result string `json:"result,omitempty"`
}
type Signer interface {
GetSession(clientPubkey string) (Session, bool)
HandleRequest(event *nostr.Event) (req Request, resp Response, eventResponse nostr.Event, err error)
2024-01-09 16:55:00 -03:00
}
type Session struct {
SharedKey []byte // nip04
ConversationKey [32]byte // nip44
2024-01-09 16:55:00 -03:00
}
type RelayReadWrite struct {
Read bool `json:"read"`
Write bool `json:"write"`
}
func (s Session) ParseRequest(event *nostr.Event) (Request, error) {
var req Request
2024-05-20 09:20:39 -03:00
plain, err := nip44.Decrypt(event.Content, s.ConversationKey)
2024-01-09 16:55:00 -03:00
if err != nil {
2024-05-20 09:20:39 -03:00
plain, err = nip04.Decrypt(event.Content, s.SharedKey)
if err != nil {
return req, fmt.Errorf("failed to decrypt event from %s: %w", event.PubKey, err)
}
2024-01-09 16:55:00 -03:00
}
err = json.Unmarshal([]byte(plain), &req)
return req, err
}
func (s Session) MakeResponse(
id string,
requester string,
result string,
err error,
) (resp Response, evt nostr.Event, error error) {
if err != nil {
resp = Response{
ID: id,
Error: err.Error(),
}
} else if result != "" {
resp = Response{
ID: id,
Result: result,
}
}
jresp, _ := json.Marshal(resp)
ciphertext, err := nip04.Encrypt(string(jresp), s.SharedKey)
if err != nil {
return resp, evt, fmt.Errorf("failed to encrypt result: %w", err)
}
evt.Content = ciphertext
evt.CreatedAt = nostr.Now()
evt.Kind = nostr.KindNostrConnect
evt.Tags = nostr.Tags{nostr.Tag{"p", requester}}
return resp, evt, nil
}
2024-03-01 15:54:28 -03:00
func IsValidBunkerURL(input string) bool {
return BUNKER_REGEX.MatchString(input)
}