2024-06-08 17:58:01 +00:00
|
|
|
package adapter
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
. "forge.lightcrystal.systems/lightcrystal/underbbs/models"
|
|
|
|
nostr "github.com/nbd-wtf/go-nostr"
|
2024-09-28 17:39:03 +00:00
|
|
|
"log"
|
|
|
|
"os"
|
2024-06-08 17:58:01 +00:00
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
type NostrAdapter struct {
|
2024-09-28 17:39:03 +00:00
|
|
|
data *chan SocketData
|
2024-06-08 17:58:01 +00:00
|
|
|
nickname string
|
|
|
|
privkey string
|
|
|
|
relays []*nostr.Relay
|
|
|
|
}
|
|
|
|
|
2024-09-28 17:39:03 +00:00
|
|
|
func (self *NostrAdapter) send(data SocketData) {
|
|
|
|
if self.data != nil {
|
|
|
|
*self.data <- data
|
|
|
|
} else {
|
|
|
|
fmt.Fprintln(os.Stdout, string(data.ToDatagram()))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-07 03:13:18 +00:00
|
|
|
func (self *NostrAdapter) Name() string {
|
2024-07-07 20:54:33 +00:00
|
|
|
return self.nickname
|
2024-07-07 03:13:18 +00:00
|
|
|
}
|
|
|
|
|
2024-09-28 17:39:03 +00:00
|
|
|
func (self *NostrAdapter) Init(settings Settings, data *chan SocketData) error {
|
2024-06-08 17:58:01 +00:00
|
|
|
self.nickname = settings.Nickname
|
|
|
|
self.privkey = *settings.PrivKey
|
|
|
|
self.data = data
|
|
|
|
|
|
|
|
ctx := context.Background()
|
|
|
|
|
|
|
|
for _, r := range settings.Relays {
|
|
|
|
pr, _ := nostr.RelayConnect(ctx, strings.Trim(r, " "))
|
|
|
|
if pr == nil {
|
|
|
|
return errors.New("Relay connection could not be completed")
|
|
|
|
}
|
|
|
|
self.relays = append(self.relays, pr)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-12-07 02:25:58 +00:00
|
|
|
func (self *NostrAdapter) Stop() {
|
|
|
|
// nostr has native streaming so we don't need to do anything in this method
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-12-06 04:05:01 +00:00
|
|
|
func (self *NostrAdapter) Subscribe(filter string, target *string) []error {
|
2024-06-08 17:58:01 +00:00
|
|
|
var filters nostr.Filters
|
|
|
|
err := json.Unmarshal([]byte(filter), &filters)
|
|
|
|
if err != nil {
|
|
|
|
return []error{err}
|
|
|
|
}
|
|
|
|
|
|
|
|
errs := make([]error, 0)
|
|
|
|
|
2024-09-28 17:39:03 +00:00
|
|
|
log.Print("unmarshalled filter from json; iterating through relays to subscribe...")
|
2024-06-08 17:58:01 +00:00
|
|
|
|
|
|
|
for _, r := range self.relays {
|
|
|
|
sub, err := r.Subscribe(context.Background(), filters)
|
|
|
|
if err != nil {
|
|
|
|
errs = append(errs, err)
|
|
|
|
} else {
|
|
|
|
go func() {
|
|
|
|
for ev := range sub.Events {
|
|
|
|
// try sequentially to encode into an underbbs object
|
|
|
|
// and send it to the appropriate channel
|
2024-06-08 20:40:14 +00:00
|
|
|
m, err := self.nostrEventToMsg(ev)
|
2024-06-08 17:58:01 +00:00
|
|
|
if err == nil {
|
2024-09-28 17:39:03 +00:00
|
|
|
self.send(m)
|
2024-06-08 17:58:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(errs) > 0 {
|
2024-09-28 17:39:03 +00:00
|
|
|
log.Print("subscription operation completed with errors")
|
2024-06-08 17:58:01 +00:00
|
|
|
return errs
|
|
|
|
}
|
2024-09-28 17:39:03 +00:00
|
|
|
log.Print("subscription operation completed without errors")
|
2024-06-08 17:58:01 +00:00
|
|
|
return nil
|
|
|
|
}
|
2024-06-22 16:39:02 +00:00
|
|
|
|
2024-08-17 00:36:01 +00:00
|
|
|
func (self *NostrAdapter) Fetch(etype string, ids []string) error {
|
2024-06-08 17:58:01 +00:00
|
|
|
return nil
|
|
|
|
}
|
2024-06-22 16:39:02 +00:00
|
|
|
|
2024-11-02 01:13:21 +00:00
|
|
|
func (self *NostrAdapter) Do(action string, data map[string]string) error {
|
2024-06-08 17:58:01 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *NostrAdapter) DefaultSubscriptionFilter() string {
|
|
|
|
return "[{\"kinds\":[1]}]"
|
|
|
|
}
|
|
|
|
|
2024-06-08 20:40:14 +00:00
|
|
|
func (self *NostrAdapter) nostrEventToMsg(evt *nostr.Event) (Message, error) {
|
2024-06-08 17:58:01 +00:00
|
|
|
m := Message{
|
2024-06-30 00:18:31 +00:00
|
|
|
Datagram: Datagram{
|
|
|
|
Protocol: "nostr",
|
|
|
|
Adapter: self.nickname,
|
|
|
|
Type: "message",
|
|
|
|
},
|
2024-06-08 17:58:01 +00:00
|
|
|
}
|
|
|
|
if evt == nil {
|
|
|
|
return m, errors.New("no event")
|
|
|
|
}
|
|
|
|
switch evt.Kind {
|
|
|
|
case nostr.KindTextNote:
|
2024-06-30 00:18:31 +00:00
|
|
|
m.Id = evt.ID
|
2024-06-27 03:13:44 +00:00
|
|
|
m.Author = evt.PubKey
|
2024-07-16 19:43:35 +00:00
|
|
|
m.Created = evt.CreatedAt.Time().UnixMilli()
|
2024-06-08 17:58:01 +00:00
|
|
|
m.Content = evt.Content
|
|
|
|
return m, nil
|
|
|
|
default:
|
|
|
|
return m, errors.New(fmt.Sprintf("unsupported event kind: %d", evt.Kind))
|
|
|
|
}
|
|
|
|
}
|