mirror of
https://github.com/fiatjaf/khatru.git
synced 2026-06-09 22:28:56 +02:00
Compare commits
18 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
33545587b6 | ||
|
|
214371f8bd | ||
|
|
fbb40f3b74 | ||
|
|
d97a2f1cf2 | ||
|
|
c9a7d60543 | ||
|
|
2bb6d4d29a | ||
|
|
2292ce4a30 | ||
|
|
2ae219a34c | ||
|
|
8c9394993b | ||
|
|
850497956c | ||
|
|
28ce6cfb7a | ||
|
|
f47282c745 | ||
|
|
f72dea346f | ||
|
|
51632dcc9f | ||
|
|
6cc2477e89 | ||
|
|
581c4ece28 | ||
|
|
596bca93c3 | ||
|
|
650d9209c3 |
132
adding.go
132
adding.go
@@ -11,34 +11,46 @@ import (
|
|||||||
|
|
||||||
// AddEvent sends an event through then normal add pipeline, as if it was received from a websocket.
|
// AddEvent sends an event through then normal add pipeline, as if it was received from a websocket.
|
||||||
func (rl *Relay) AddEvent(ctx context.Context, evt *nostr.Event) (skipBroadcast bool, writeError error) {
|
func (rl *Relay) AddEvent(ctx context.Context, evt *nostr.Event) (skipBroadcast bool, writeError error) {
|
||||||
ctx, cancel := context.WithCancel(ctx)
|
|
||||||
defer cancel()
|
|
||||||
|
|
||||||
if evt == nil {
|
if evt == nil {
|
||||||
return false, errors.New("error: event is nil")
|
return false, errors.New("error: event is nil")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if nostr.IsEphemeralKind(evt.Kind) {
|
||||||
|
return false, rl.handleEphemeral(ctx, evt)
|
||||||
|
} else {
|
||||||
|
return rl.handleNormal(ctx, evt)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rl *Relay) handleNormal(ctx context.Context, evt *nostr.Event) (skipBroadcast bool, writeError error) {
|
||||||
for _, reject := range rl.RejectEvent {
|
for _, reject := range rl.RejectEvent {
|
||||||
if reject, msg := reject(ctx, evt); reject {
|
if reject, msg := reject(ctx, evt); reject {
|
||||||
if msg == "" {
|
if msg == "" {
|
||||||
return false, errors.New("blocked: no reason")
|
return true, errors.New("blocked: no reason")
|
||||||
} else {
|
} else {
|
||||||
return false, errors.New(nostr.NormalizeOKMessage(msg, "blocked"))
|
return true, errors.New(nostr.NormalizeOKMessage(msg, "blocked"))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if 20000 <= evt.Kind && evt.Kind < 30000 {
|
// will store
|
||||||
// do not store ephemeral events
|
// regular kinds are just saved directly
|
||||||
for _, oee := range rl.OnEphemeralEvent {
|
if nostr.IsRegularKind(evt.Kind) {
|
||||||
oee(ctx, evt)
|
for _, store := range rl.StoreEvent {
|
||||||
|
if err := store(ctx, evt); err != nil {
|
||||||
|
switch err {
|
||||||
|
case eventstore.ErrDupEvent:
|
||||||
|
return true, nil
|
||||||
|
default:
|
||||||
|
return false, fmt.Errorf("%s", nostr.NormalizeOKMessage(err.Error(), "error"))
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// will store
|
// otherwise it's a replaceable -- so we'll use the replacer functions if we have any
|
||||||
// regular kinds are just saved directly
|
if len(rl.ReplaceEvent) > 0 {
|
||||||
if nostr.IsRegularKind(evt.Kind) {
|
for _, repl := range rl.ReplaceEvent {
|
||||||
for _, store := range rl.StoreEvent {
|
if err := repl(ctx, evt); err != nil {
|
||||||
if err := store(ctx, evt); err != nil {
|
|
||||||
switch err {
|
switch err {
|
||||||
case eventstore.ErrDupEvent:
|
case eventstore.ErrDupEvent:
|
||||||
return true, nil
|
return true, nil
|
||||||
@@ -48,68 +60,54 @@ func (rl *Relay) AddEvent(ctx context.Context, evt *nostr.Event) (skipBroadcast
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// otherwise it's a replaceable -- so we'll use the replacer functions if we have any
|
// otherwise do it the manual way
|
||||||
if len(rl.ReplaceEvent) > 0 {
|
filter := nostr.Filter{Limit: 1, Kinds: []int{evt.Kind}, Authors: []string{evt.PubKey}}
|
||||||
for _, repl := range rl.ReplaceEvent {
|
if nostr.IsAddressableKind(evt.Kind) {
|
||||||
if err := repl(ctx, evt); err != nil {
|
// when addressable, add the "d" tag to the filter
|
||||||
switch err {
|
filter.Tags = nostr.TagMap{"d": []string{evt.Tags.GetD()}}
|
||||||
|
}
|
||||||
|
|
||||||
|
// now we fetch old events and delete them
|
||||||
|
shouldStore := true
|
||||||
|
for _, query := range rl.QueryEvents {
|
||||||
|
ch, err := query(ctx, filter)
|
||||||
|
if err != nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
for previous := range ch {
|
||||||
|
if isOlder(previous, evt) {
|
||||||
|
for _, del := range rl.DeleteEvent {
|
||||||
|
del(ctx, previous)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// we found a more recent event, so we won't delete it and also will not store this new one
|
||||||
|
shouldStore = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// store
|
||||||
|
if shouldStore {
|
||||||
|
for _, store := range rl.StoreEvent {
|
||||||
|
if saveErr := store(ctx, evt); saveErr != nil {
|
||||||
|
switch saveErr {
|
||||||
case eventstore.ErrDupEvent:
|
case eventstore.ErrDupEvent:
|
||||||
return true, nil
|
return true, nil
|
||||||
default:
|
default:
|
||||||
return false, fmt.Errorf("%s", nostr.NormalizeOKMessage(err.Error(), "error"))
|
return false, fmt.Errorf("%s", nostr.NormalizeOKMessage(saveErr.Error(), "error"))
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// otherwise do it the manual way
|
|
||||||
filter := nostr.Filter{Limit: 1, Kinds: []int{evt.Kind}, Authors: []string{evt.PubKey}}
|
|
||||||
if nostr.IsAddressableKind(evt.Kind) {
|
|
||||||
// when addressable, add the "d" tag to the filter
|
|
||||||
filter.Tags = nostr.TagMap{"d": []string{evt.Tags.GetD()}}
|
|
||||||
}
|
|
||||||
|
|
||||||
// now we fetch old events and delete them
|
|
||||||
shouldStore := true
|
|
||||||
for _, query := range rl.QueryEvents {
|
|
||||||
ch, err := query(ctx, filter)
|
|
||||||
if err != nil {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
for previous := range ch {
|
|
||||||
if isOlder(previous, evt) {
|
|
||||||
for _, del := range rl.DeleteEvent {
|
|
||||||
del(ctx, previous)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// we found a more recent event, so we won't delete it and also will not store this new one
|
|
||||||
shouldStore = false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// store
|
|
||||||
if shouldStore {
|
|
||||||
for _, store := range rl.StoreEvent {
|
|
||||||
if saveErr := store(ctx, evt); saveErr != nil {
|
|
||||||
switch saveErr {
|
|
||||||
case eventstore.ErrDupEvent:
|
|
||||||
return true, nil
|
|
||||||
default:
|
|
||||||
return false, fmt.Errorf("%s", nostr.NormalizeOKMessage(saveErr.Error(), "error"))
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, ons := range rl.OnEventSaved {
|
|
||||||
ons(ctx, evt)
|
|
||||||
}
|
|
||||||
|
|
||||||
// track event expiration if applicable
|
|
||||||
rl.expirationManager.trackEvent(evt)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for _, ons := range rl.OnEventSaved {
|
||||||
|
ons(ctx, evt)
|
||||||
|
}
|
||||||
|
|
||||||
|
// track event expiration if applicable
|
||||||
|
rl.expirationManager.trackEvent(evt)
|
||||||
|
|
||||||
return false, nil
|
return false, nil
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,12 +1,12 @@
|
|||||||
package blossom
|
package blossom
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"encoding/base64"
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/http"
|
"net/http"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/cloudwego/base64x"
|
|
||||||
"github.com/mailru/easyjson"
|
"github.com/mailru/easyjson"
|
||||||
"github.com/nbd-wtf/go-nostr"
|
"github.com/nbd-wtf/go-nostr"
|
||||||
)
|
)
|
||||||
@@ -17,7 +17,7 @@ func readAuthorization(r *http.Request) (*nostr.Event, error) {
|
|||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
eventj, err := base64x.StdEncoding.DecodeString(token[6:])
|
eventj, err := base64.StdEncoding.DecodeString(token[6:])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("invalid base64 token")
|
return nil, fmt.Errorf("invalid base64 token")
|
||||||
}
|
}
|
||||||
@@ -25,7 +25,7 @@ func readAuthorization(r *http.Request) (*nostr.Event, error) {
|
|||||||
if err := easyjson.Unmarshal(eventj, &evt); err != nil {
|
if err := easyjson.Unmarshal(eventj, &evt); err != nil {
|
||||||
return nil, fmt.Errorf("broken event")
|
return nil, fmt.Errorf("broken event")
|
||||||
}
|
}
|
||||||
if evt.Kind != 24242 || len(evt.ID) != 64 || !evt.CheckID() {
|
if evt.Kind != 24242 || !evt.CheckID() {
|
||||||
return nil, fmt.Errorf("invalid event")
|
return nil, fmt.Errorf("invalid event")
|
||||||
}
|
}
|
||||||
if ok, _ := evt.CheckSignature(); !ok {
|
if ok, _ := evt.CheckSignature(); !ok {
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ func (bs BlossomServer) handleUploadCheck(w http.ResponseWriter, r *http.Request
|
|||||||
blossomError(w, "missing \"Authorization\" header", 401)
|
blossomError(w, "missing \"Authorization\" header", 401)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if auth.Tags.GetFirst([]string{"t", "upload"}) == nil {
|
if auth.Tags.FindWithValue("t", "upload") == nil {
|
||||||
blossomError(w, "invalid \"Authorization\" event \"t\" tag", 403)
|
blossomError(w, "invalid \"Authorization\" event \"t\" tag", 403)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -59,7 +59,7 @@ func (bs BlossomServer) handleUpload(w http.ResponseWriter, r *http.Request) {
|
|||||||
blossomError(w, "missing \"Authorization\" header", 401)
|
blossomError(w, "missing \"Authorization\" header", 401)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if auth.Tags.GetFirst([]string{"t", "upload"}) == nil {
|
if auth.Tags.FindWithValue("t", "upload") == nil {
|
||||||
blossomError(w, "invalid \"Authorization\" event \"t\" tag", 403)
|
blossomError(w, "invalid \"Authorization\" event \"t\" tag", 403)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -73,7 +73,7 @@ func (bs BlossomServer) handleUpload(w http.ResponseWriter, r *http.Request) {
|
|||||||
|
|
||||||
// read first bytes of upload so we can find out the filetype
|
// read first bytes of upload so we can find out the filetype
|
||||||
b := make([]byte, min(50, size), size)
|
b := make([]byte, min(50, size), size)
|
||||||
if _, err = r.Body.Read(b); err != nil {
|
if n, err := r.Body.Read(b); err != nil && n != size {
|
||||||
blossomError(w, "failed to read initial bytes of upload body: "+err.Error(), 400)
|
blossomError(w, "failed to read initial bytes of upload body: "+err.Error(), 400)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -163,13 +163,13 @@ func (bs BlossomServer) handleGetBlob(w http.ResponseWriter, r *http.Request) {
|
|||||||
|
|
||||||
// if there is one, we check if it has the extra requirements
|
// if there is one, we check if it has the extra requirements
|
||||||
if auth != nil {
|
if auth != nil {
|
||||||
if auth.Tags.GetFirst([]string{"t", "get"}) == nil {
|
if auth.Tags.FindWithValue("t", "get") == nil {
|
||||||
blossomError(w, "invalid \"Authorization\" event \"t\" tag", 403)
|
blossomError(w, "invalid \"Authorization\" event \"t\" tag", 403)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if auth.Tags.GetFirst([]string{"x", hhash}) == nil &&
|
if auth.Tags.FindWithValue("x", hhash) == nil &&
|
||||||
auth.Tags.GetFirst([]string{"server", bs.ServiceURL}) == nil {
|
auth.Tags.FindWithValue("server", bs.ServiceURL) == nil {
|
||||||
blossomError(w, "invalid \"Authorization\" event \"x\" or \"server\" tag", 403)
|
blossomError(w, "invalid \"Authorization\" event \"x\" or \"server\" tag", 403)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -239,7 +239,7 @@ func (bs BlossomServer) handleList(w http.ResponseWriter, r *http.Request) {
|
|||||||
|
|
||||||
// if there is one, we check if it has the extra requirements
|
// if there is one, we check if it has the extra requirements
|
||||||
if auth != nil {
|
if auth != nil {
|
||||||
if auth.Tags.GetFirst([]string{"t", "list"}) == nil {
|
if auth.Tags.FindWithValue("t", "list") == nil {
|
||||||
blossomError(w, "invalid \"Authorization\" event \"t\" tag", 403)
|
blossomError(w, "invalid \"Authorization\" event \"t\" tag", 403)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -283,7 +283,7 @@ func (bs BlossomServer) handleDelete(w http.ResponseWriter, r *http.Request) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if auth != nil {
|
if auth != nil {
|
||||||
if auth.Tags.GetFirst([]string{"t", "delete"}) == nil {
|
if auth.Tags.FindWithValue("t", "delete") == nil {
|
||||||
blossomError(w, "invalid \"Authorization\" event \"t\" tag", 403)
|
blossomError(w, "invalid \"Authorization\" event \"t\" tag", 403)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -296,8 +296,8 @@ func (bs BlossomServer) handleDelete(w http.ResponseWriter, r *http.Request) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
hhash = hhash[1:]
|
hhash = hhash[1:]
|
||||||
if auth.Tags.GetFirst([]string{"x", hhash}) == nil &&
|
if auth.Tags.FindWithValue("x", hhash) == nil &&
|
||||||
auth.Tags.GetFirst([]string{"server", bs.ServiceURL}) == nil {
|
auth.Tags.FindWithValue("server", bs.ServiceURL) == nil {
|
||||||
blossomError(w, "invalid \"Authorization\" event \"x\" or \"server\" tag", 403)
|
blossomError(w, "invalid \"Authorization\" event \"x\" or \"server\" tag", 403)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -49,7 +49,7 @@ func New(rl *khatru.Relay, serviceURL string) *BlossomServer {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(strings.SplitN(r.URL.Path, ".", 2)[0]) == 65 {
|
if (len(r.URL.Path) == 65 || strings.Index(r.URL.Path, ".") == 65) && strings.Index(r.URL.Path[1:], "/") == -1 {
|
||||||
if r.Method == "HEAD" {
|
if r.Method == "HEAD" {
|
||||||
bs.handleHasBlob(w, r)
|
bs.handleHasBlob(w, r)
|
||||||
return
|
return
|
||||||
|
|||||||
@@ -6,6 +6,6 @@ import (
|
|||||||
|
|
||||||
// BroadcastEvent emits an event to all listeners whose filters' match, skipping all filters and actions
|
// BroadcastEvent emits an event to all listeners whose filters' match, skipping all filters and actions
|
||||||
// it also doesn't attempt to store the event or trigger any reactions or callbacks
|
// it also doesn't attempt to store the event or trigger any reactions or callbacks
|
||||||
func (rl *Relay) BroadcastEvent(evt *nostr.Event) {
|
func (rl *Relay) BroadcastEvent(evt *nostr.Event) int {
|
||||||
rl.notifyListeners(evt)
|
return rl.notifyListeners(evt)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -39,6 +39,7 @@ func (rl *Relay) handleDeleteRequest(ctx context.Context, evt *nostr.Event) erro
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ctx := context.WithValue(ctx, internalCallKey, struct{}{})
|
||||||
for _, query := range rl.QueryEvents {
|
for _, query := range rl.QueryEvents {
|
||||||
ch, err := query(ctx, f)
|
ch, err := query(ctx, f)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -66,6 +67,9 @@ func (rl *Relay) handleDeleteRequest(ctx context.Context, evt *nostr.Event) erro
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// if it was tracked to be expired that is not needed anymore
|
||||||
|
rl.expirationManager.removeEvent(target.ID)
|
||||||
} else {
|
} else {
|
||||||
// fail and stop here
|
// fail and stop here
|
||||||
return fmt.Errorf("blocked: %s", msg)
|
return fmt.Errorf("blocked: %s", msg)
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ export default {
|
|||||||
{ text: 'HTTP Integration', link: '/core/embed' },
|
{ text: 'HTTP Integration', link: '/core/embed' },
|
||||||
{ text: 'Request Routing', link: '/core/routing' },
|
{ text: 'Request Routing', link: '/core/routing' },
|
||||||
{ text: 'Management API', link: '/core/management' },
|
{ text: 'Management API', link: '/core/management' },
|
||||||
{ text: 'Media Storage', link: '/core/blossom' },
|
{ text: 'Media Storage (Blossom)', link: '/core/blossom' },
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -18,7 +18,8 @@ func main() {
|
|||||||
bl := blossom.New(relay, "http://localhost:3334")
|
bl := blossom.New(relay, "http://localhost:3334")
|
||||||
|
|
||||||
// create a database for keeping track of blob metadata
|
// create a database for keeping track of blob metadata
|
||||||
bl.Store = blossom.EventStoreBlobIndexWrapper{Store: db, ServiceURL: bl.ServiceURL}
|
// (do not use the same database used for the relay events)
|
||||||
|
bl.Store = blossom.EventStoreBlobIndexWrapper{Store: blobdb, ServiceURL: bl.ServiceURL}
|
||||||
|
|
||||||
// implement the required storage functions
|
// implement the required storage functions
|
||||||
bl.StoreBlob = append(bl.StoreBlob, func(ctx context.Context, sha256 string, body []byte) error {
|
bl.StoreBlob = append(bl.StoreBlob, func(ctx context.Context, sha256 string, body []byte) error {
|
||||||
|
|||||||
@@ -47,7 +47,7 @@ router.Route().
|
|||||||
return true
|
return true
|
||||||
case event.Kind <= 12 && event.Kind >= 9:
|
case event.Kind <= 12 && event.Kind >= 9:
|
||||||
return true
|
return true
|
||||||
case event.Tags.GetFirst([]string{"h", ""}) != nil:
|
case event.Tags.Find("h") != nil:
|
||||||
return true
|
return true
|
||||||
default:
|
default:
|
||||||
return false
|
return false
|
||||||
|
|||||||
26
ephemeral.go
Normal file
26
ephemeral.go
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
package khatru
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
|
||||||
|
"github.com/nbd-wtf/go-nostr"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (rl *Relay) handleEphemeral(ctx context.Context, evt *nostr.Event) error {
|
||||||
|
for _, reject := range rl.RejectEvent {
|
||||||
|
if reject, msg := reject(ctx, evt); reject {
|
||||||
|
if msg == "" {
|
||||||
|
return errors.New("blocked: no reason")
|
||||||
|
} else {
|
||||||
|
return errors.New(nostr.NormalizeOKMessage(msg, "blocked"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, oee := range rl.OnEphemeralEvent {
|
||||||
|
oee(ctx, evt)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
@@ -13,7 +13,7 @@ func main() {
|
|||||||
relay := khatru.NewRelay()
|
relay := khatru.NewRelay()
|
||||||
|
|
||||||
db := lmdb.LMDBBackend{Path: "/tmp/khatru-lmdb-tmp"}
|
db := lmdb.LMDBBackend{Path: "/tmp/khatru-lmdb-tmp"}
|
||||||
os.MkdirAll(db.Path, 0755)
|
os.MkdirAll(db.Path, 0o755)
|
||||||
if err := db.Init(); err != nil {
|
if err := db.Init(); err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,19 +15,22 @@ import (
|
|||||||
func main() {
|
func main() {
|
||||||
relay := khatru.NewRelay()
|
relay := khatru.NewRelay()
|
||||||
|
|
||||||
db := &badger.BadgerBackend{Path: "/tmp/khatru-badger-blossom-tmp"}
|
db := &badger.BadgerBackend{Path: "/tmp/khatru-badger-tmp"}
|
||||||
if err := db.Init(); err != nil {
|
if err := db.Init(); err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
relay.StoreEvent = append(relay.StoreEvent, db.SaveEvent)
|
relay.StoreEvent = append(relay.StoreEvent, db.SaveEvent)
|
||||||
relay.QueryEvents = append(relay.QueryEvents, db.QueryEvents)
|
relay.QueryEvents = append(relay.QueryEvents, db.QueryEvents)
|
||||||
relay.CountEvents = append(relay.CountEvents, db.CountEvents)
|
relay.CountEvents = append(relay.CountEvents, db.CountEvents)
|
||||||
relay.DeleteEvent = append(relay.DeleteEvent, db.DeleteEvent)
|
relay.DeleteEvent = append(relay.DeleteEvent, db.DeleteEvent)
|
||||||
relay.ReplaceEvent = append(relay.ReplaceEvent, db.ReplaceEvent)
|
relay.ReplaceEvent = append(relay.ReplaceEvent, db.ReplaceEvent)
|
||||||
|
|
||||||
|
bdb := &badger.BadgerBackend{Path: "/tmp/khatru-badger-blossom-tmp"}
|
||||||
|
if err := bdb.Init(); err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
bl := blossom.New(relay, "http://localhost:3334")
|
bl := blossom.New(relay, "http://localhost:3334")
|
||||||
bl.Store = blossom.EventStoreBlobIndexWrapper{Store: db, ServiceURL: bl.ServiceURL}
|
bl.Store = blossom.EventStoreBlobIndexWrapper{Store: bdb, ServiceURL: bl.ServiceURL}
|
||||||
bl.StoreBlob = append(bl.StoreBlob, func(ctx context.Context, sha256 string, body []byte) error {
|
bl.StoreBlob = append(bl.StoreBlob, func(ctx context.Context, sha256 string, body []byte) error {
|
||||||
fmt.Println("storing", sha256, len(body))
|
fmt.Println("storing", sha256, len(body))
|
||||||
return nil
|
return nil
|
||||||
|
|||||||
@@ -16,7 +16,7 @@ func main() {
|
|||||||
relay := khatru.NewRelay()
|
relay := khatru.NewRelay()
|
||||||
|
|
||||||
db := lmdb.LMDBBackend{Path: "/tmp/exclusive"}
|
db := lmdb.LMDBBackend{Path: "/tmp/exclusive"}
|
||||||
os.MkdirAll(db.Path, 0755)
|
os.MkdirAll(db.Path, 0o755)
|
||||||
if err := db.Init(); err != nil {
|
if err := db.Init(); err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -52,7 +52,7 @@ func main() {
|
|||||||
return slices.Contains(filter.Kinds, 1) && slices.Contains(filter.Tags["t"], "spam")
|
return slices.Contains(filter.Kinds, 1) && slices.Contains(filter.Tags["t"], "spam")
|
||||||
}).
|
}).
|
||||||
Event(func(event *nostr.Event) bool {
|
Event(func(event *nostr.Event) bool {
|
||||||
return event.Kind == 1 && event.Tags.GetFirst([]string{"t", "spam"}) != nil
|
return event.Kind == 1 && event.Tags.FindWithValue("t", "spam") != nil
|
||||||
}).
|
}).
|
||||||
Relay(r2)
|
Relay(r2)
|
||||||
|
|
||||||
|
|||||||
@@ -73,6 +73,7 @@ func (em *expirationManager) initialScan(ctx context.Context) {
|
|||||||
defer em.mu.Unlock()
|
defer em.mu.Unlock()
|
||||||
|
|
||||||
// query all events
|
// query all events
|
||||||
|
ctx = context.WithValue(ctx, internalCallKey, struct{}{})
|
||||||
for _, query := range em.relay.QueryEvents {
|
for _, query := range em.relay.QueryEvents {
|
||||||
ch, err := query(ctx, nostr.Filter{})
|
ch, err := query(ctx, nostr.Filter{})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -107,6 +108,7 @@ func (em *expirationManager) checkExpiredEvents(ctx context.Context) {
|
|||||||
|
|
||||||
heap.Pop(&em.events)
|
heap.Pop(&em.events)
|
||||||
|
|
||||||
|
ctx := context.WithValue(ctx, internalCallKey, struct{}{})
|
||||||
for _, query := range em.relay.QueryEvents {
|
for _, query := range em.relay.QueryEvents {
|
||||||
ch, err := query(ctx, nostr.Filter{IDs: []string{next.id}})
|
ch, err := query(ctx, nostr.Filter{IDs: []string{next.id}})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -133,3 +135,16 @@ func (em *expirationManager) trackEvent(evt *nostr.Event) {
|
|||||||
em.mu.Unlock()
|
em.mu.Unlock()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (em *expirationManager) removeEvent(id string) {
|
||||||
|
em.mu.Lock()
|
||||||
|
defer em.mu.Unlock()
|
||||||
|
|
||||||
|
// Find and remove the event from the heap
|
||||||
|
for i := 0; i < len(em.events); i++ {
|
||||||
|
if em.events[i].id == id {
|
||||||
|
heap.Remove(&em.events, i)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
17
go.mod
17
go.mod
@@ -1,14 +1,14 @@
|
|||||||
module github.com/fiatjaf/khatru
|
module github.com/fiatjaf/khatru
|
||||||
|
|
||||||
go 1.23.1
|
go 1.24.1
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/bep/debounce v1.2.1
|
github.com/bep/debounce v1.2.1
|
||||||
github.com/cloudwego/base64x v0.1.5
|
|
||||||
github.com/fasthttp/websocket v1.5.12
|
github.com/fasthttp/websocket v1.5.12
|
||||||
github.com/fiatjaf/eventstore v0.16.2
|
github.com/fiatjaf/eventstore v0.16.2
|
||||||
github.com/liamg/magic v0.0.1
|
github.com/liamg/magic v0.0.1
|
||||||
github.com/nbd-wtf/go-nostr v0.51.2
|
github.com/mailru/easyjson v0.9.0
|
||||||
|
github.com/nbd-wtf/go-nostr v0.51.8
|
||||||
github.com/puzpuzpuz/xsync/v3 v3.5.1
|
github.com/puzpuzpuz/xsync/v3 v3.5.1
|
||||||
github.com/rs/cors v1.11.1
|
github.com/rs/cors v1.11.1
|
||||||
github.com/stretchr/testify v1.10.0
|
github.com/stretchr/testify v1.10.0
|
||||||
@@ -22,10 +22,11 @@ require (
|
|||||||
github.com/aquasecurity/esquery v0.2.0 // indirect
|
github.com/aquasecurity/esquery v0.2.0 // indirect
|
||||||
github.com/btcsuite/btcd/btcec/v2 v2.3.4 // indirect
|
github.com/btcsuite/btcd/btcec/v2 v2.3.4 // indirect
|
||||||
github.com/btcsuite/btcd/chaincfg/chainhash v1.1.0 // indirect
|
github.com/btcsuite/btcd/chaincfg/chainhash v1.1.0 // indirect
|
||||||
github.com/bytedance/sonic v1.13.1 // indirect
|
github.com/bytedance/sonic v1.13.2 // indirect
|
||||||
github.com/bytedance/sonic/loader v0.2.4 // indirect
|
github.com/bytedance/sonic/loader v0.2.4 // indirect
|
||||||
github.com/cespare/xxhash/v2 v2.3.0 // indirect
|
github.com/cespare/xxhash/v2 v2.3.0 // indirect
|
||||||
github.com/coder/websocket v1.8.12 // indirect
|
github.com/cloudwego/base64x v0.1.5 // indirect
|
||||||
|
github.com/coder/websocket v1.8.13 // indirect
|
||||||
github.com/davecgh/go-spew v1.1.1 // indirect
|
github.com/davecgh/go-spew v1.1.1 // indirect
|
||||||
github.com/decred/dcrd/crypto/blake256 v1.1.0 // indirect
|
github.com/decred/dcrd/crypto/blake256 v1.1.0 // indirect
|
||||||
github.com/decred/dcrd/dcrec/secp256k1/v4 v4.4.0 // indirect
|
github.com/decred/dcrd/dcrec/secp256k1/v4 v4.4.0 // indirect
|
||||||
@@ -46,9 +47,7 @@ require (
|
|||||||
github.com/klauspost/compress v1.18.0 // indirect
|
github.com/klauspost/compress v1.18.0 // indirect
|
||||||
github.com/klauspost/cpuid/v2 v2.2.10 // indirect
|
github.com/klauspost/cpuid/v2 v2.2.10 // indirect
|
||||||
github.com/lib/pq v1.10.9 // indirect
|
github.com/lib/pq v1.10.9 // indirect
|
||||||
github.com/mailru/easyjson v0.9.0 // indirect
|
|
||||||
github.com/mattn/go-sqlite3 v1.14.24 // indirect
|
github.com/mattn/go-sqlite3 v1.14.24 // indirect
|
||||||
github.com/minio/simdjson-go v0.4.5 // indirect
|
|
||||||
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect
|
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect
|
||||||
github.com/modern-go/reflect2 v1.0.2 // indirect
|
github.com/modern-go/reflect2 v1.0.2 // indirect
|
||||||
github.com/pkg/errors v0.9.1 // indirect
|
github.com/pkg/errors v0.9.1 // indirect
|
||||||
@@ -59,14 +58,14 @@ require (
|
|||||||
github.com/tidwall/pretty v1.2.1 // indirect
|
github.com/tidwall/pretty v1.2.1 // indirect
|
||||||
github.com/twitchyliquid64/golang-asm v0.15.1 // indirect
|
github.com/twitchyliquid64/golang-asm v0.15.1 // indirect
|
||||||
github.com/valyala/bytebufferpool v1.0.0 // indirect
|
github.com/valyala/bytebufferpool v1.0.0 // indirect
|
||||||
github.com/valyala/fasthttp v1.58.0 // indirect
|
github.com/valyala/fasthttp v1.59.0 // indirect
|
||||||
go.opencensus.io v0.24.0 // indirect
|
go.opencensus.io v0.24.0 // indirect
|
||||||
go.opentelemetry.io/otel v1.32.0 // indirect
|
go.opentelemetry.io/otel v1.32.0 // indirect
|
||||||
go.opentelemetry.io/otel/metric v1.32.0 // indirect
|
go.opentelemetry.io/otel/metric v1.32.0 // indirect
|
||||||
go.opentelemetry.io/otel/trace v1.32.0 // indirect
|
go.opentelemetry.io/otel/trace v1.32.0 // indirect
|
||||||
golang.org/x/arch v0.15.0 // indirect
|
golang.org/x/arch v0.15.0 // indirect
|
||||||
golang.org/x/exp v0.0.0-20250305212735-054e65f0b394 // indirect
|
golang.org/x/exp v0.0.0-20250305212735-054e65f0b394 // indirect
|
||||||
golang.org/x/net v0.35.0 // indirect
|
golang.org/x/net v0.37.0 // indirect
|
||||||
golang.org/x/sys v0.31.0 // indirect
|
golang.org/x/sys v0.31.0 // indirect
|
||||||
google.golang.org/protobuf v1.36.2 // indirect
|
google.golang.org/protobuf v1.36.2 // indirect
|
||||||
gopkg.in/yaml.v3 v3.0.1 // indirect
|
gopkg.in/yaml.v3 v3.0.1 // indirect
|
||||||
|
|||||||
28
go.sum
28
go.sum
@@ -21,6 +21,8 @@ github.com/btcsuite/btcd/chaincfg/chainhash v1.1.0 h1:59Kx4K6lzOW5w6nFlA0v5+lk/6
|
|||||||
github.com/btcsuite/btcd/chaincfg/chainhash v1.1.0/go.mod h1:7SFka0XMvUgj3hfZtydOrQY2mwhPclbT2snogU7SQQc=
|
github.com/btcsuite/btcd/chaincfg/chainhash v1.1.0/go.mod h1:7SFka0XMvUgj3hfZtydOrQY2mwhPclbT2snogU7SQQc=
|
||||||
github.com/bytedance/sonic v1.13.1 h1:Jyd5CIvdFnkOWuKXr+wm4Nyk2h0yAFsr8ucJgEasO3g=
|
github.com/bytedance/sonic v1.13.1 h1:Jyd5CIvdFnkOWuKXr+wm4Nyk2h0yAFsr8ucJgEasO3g=
|
||||||
github.com/bytedance/sonic v1.13.1/go.mod h1:o68xyaF9u2gvVBuGHPlUVCy+ZfmNNO5ETf1+KgkJhz4=
|
github.com/bytedance/sonic v1.13.1/go.mod h1:o68xyaF9u2gvVBuGHPlUVCy+ZfmNNO5ETf1+KgkJhz4=
|
||||||
|
github.com/bytedance/sonic v1.13.2 h1:8/H1FempDZqC4VqjptGo14QQlJx8VdZJegxs6wwfqpQ=
|
||||||
|
github.com/bytedance/sonic v1.13.2/go.mod h1:o68xyaF9u2gvVBuGHPlUVCy+ZfmNNO5ETf1+KgkJhz4=
|
||||||
github.com/bytedance/sonic/loader v0.1.1/go.mod h1:ncP89zfokxS5LZrJxl5z0UJcsk4M4yY2JpfqGeCtNLU=
|
github.com/bytedance/sonic/loader v0.1.1/go.mod h1:ncP89zfokxS5LZrJxl5z0UJcsk4M4yY2JpfqGeCtNLU=
|
||||||
github.com/bytedance/sonic/loader v0.2.4 h1:ZWCw4stuXUsn1/+zQDqeE7JKP+QO47tz7QCNan80NzY=
|
github.com/bytedance/sonic/loader v0.2.4 h1:ZWCw4stuXUsn1/+zQDqeE7JKP+QO47tz7QCNan80NzY=
|
||||||
github.com/bytedance/sonic/loader v0.2.4/go.mod h1:N8A3vUdtUebEY2/VQC0MyhYeKUFosQU6FxH2JmUe6VI=
|
github.com/bytedance/sonic/loader v0.2.4/go.mod h1:N8A3vUdtUebEY2/VQC0MyhYeKUFosQU6FxH2JmUe6VI=
|
||||||
@@ -34,13 +36,13 @@ github.com/cloudwego/iasm v0.2.0/go.mod h1:8rXZaNYT2n95jn+zTI1sDr+IgcD2GVs0nlbbQ
|
|||||||
github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc=
|
github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc=
|
||||||
github.com/coder/websocket v1.8.12 h1:5bUXkEPPIbewrnkU8LTCLVaxi4N4J8ahufH2vlo4NAo=
|
github.com/coder/websocket v1.8.12 h1:5bUXkEPPIbewrnkU8LTCLVaxi4N4J8ahufH2vlo4NAo=
|
||||||
github.com/coder/websocket v1.8.12/go.mod h1:LNVeNrXQZfe5qhS9ALED3uA+l5pPqvwXg3CKoDBB2gs=
|
github.com/coder/websocket v1.8.12/go.mod h1:LNVeNrXQZfe5qhS9ALED3uA+l5pPqvwXg3CKoDBB2gs=
|
||||||
|
github.com/coder/websocket v1.8.13 h1:f3QZdXy7uGVz+4uCJy2nTZyM0yTBj8yANEHhqlXZ9FE=
|
||||||
|
github.com/coder/websocket v1.8.13/go.mod h1:LNVeNrXQZfe5qhS9ALED3uA+l5pPqvwXg3CKoDBB2gs=
|
||||||
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
||||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
github.com/decred/dcrd/crypto/blake256 v1.1.0 h1:zPMNGQCm0g4QTY27fOCorQW7EryeQ/U0x++OzVrdms8=
|
github.com/decred/dcrd/crypto/blake256 v1.1.0 h1:zPMNGQCm0g4QTY27fOCorQW7EryeQ/U0x++OzVrdms8=
|
||||||
github.com/decred/dcrd/crypto/blake256 v1.1.0/go.mod h1:2OfgNZ5wDpcsFmHmCK5gZTPcCXqlm2ArzUIkw9czNJo=
|
github.com/decred/dcrd/crypto/blake256 v1.1.0/go.mod h1:2OfgNZ5wDpcsFmHmCK5gZTPcCXqlm2ArzUIkw9czNJo=
|
||||||
github.com/decred/dcrd/dcrec/secp256k1/v4 v4.3.0 h1:rpfIENRNNilwHwZeG5+P150SMrnNEcHYvcCuK6dPZSg=
|
|
||||||
github.com/decred/dcrd/dcrec/secp256k1/v4 v4.3.0/go.mod h1:v57UDF4pDQJcEfFUCRop3lJL149eHGSe9Jvczhzjo/0=
|
|
||||||
github.com/decred/dcrd/dcrec/secp256k1/v4 v4.4.0 h1:NMZiJj8QnKe1LgsbDayM4UoHwbvwDRwnI3hwNaAHRnc=
|
github.com/decred/dcrd/dcrec/secp256k1/v4 v4.4.0 h1:NMZiJj8QnKe1LgsbDayM4UoHwbvwDRwnI3hwNaAHRnc=
|
||||||
github.com/decred/dcrd/dcrec/secp256k1/v4 v4.4.0/go.mod h1:ZXNYxsqcloTdSy/rNShjYzMhyjf0LaoftYK0p+A3h40=
|
github.com/decred/dcrd/dcrec/secp256k1/v4 v4.4.0/go.mod h1:ZXNYxsqcloTdSy/rNShjYzMhyjf0LaoftYK0p+A3h40=
|
||||||
github.com/dgraph-io/badger/v4 v4.5.0 h1:TeJE3I1pIWLBjYhIYCA1+uxrjWEoJXImFBMEBVSm16g=
|
github.com/dgraph-io/badger/v4 v4.5.0 h1:TeJE3I1pIWLBjYhIYCA1+uxrjWEoJXImFBMEBVSm16g=
|
||||||
@@ -127,24 +129,20 @@ github.com/mailru/easyjson v0.9.0/go.mod h1:1+xMtQp2MRNVL/V1bOzuP3aP8VNwRW55fQUt
|
|||||||
github.com/mattn/go-sqlite3 v1.14.22/go.mod h1:Uh1q+B4BYcTPb+yiD3kU8Ct7aC0hY9fxUwlHK0RXw+Y=
|
github.com/mattn/go-sqlite3 v1.14.22/go.mod h1:Uh1q+B4BYcTPb+yiD3kU8Ct7aC0hY9fxUwlHK0RXw+Y=
|
||||||
github.com/mattn/go-sqlite3 v1.14.24 h1:tpSp2G2KyMnnQu99ngJ47EIkWVmliIizyZBfPrBWDRM=
|
github.com/mattn/go-sqlite3 v1.14.24 h1:tpSp2G2KyMnnQu99ngJ47EIkWVmliIizyZBfPrBWDRM=
|
||||||
github.com/mattn/go-sqlite3 v1.14.24/go.mod h1:Uh1q+B4BYcTPb+yiD3kU8Ct7aC0hY9fxUwlHK0RXw+Y=
|
github.com/mattn/go-sqlite3 v1.14.24/go.mod h1:Uh1q+B4BYcTPb+yiD3kU8Ct7aC0hY9fxUwlHK0RXw+Y=
|
||||||
github.com/minio/simdjson-go v0.4.5 h1:r4IQwjRGmWCQ2VeMc7fGiilu1z5du0gJ/I/FsKwgo5A=
|
|
||||||
github.com/minio/simdjson-go v0.4.5/go.mod h1:eoNz0DcLQRyEDeaPr4Ru6JpjlZPzbA0IodxVJk8lO8E=
|
|
||||||
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
||||||
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w8PVh93nsPXa1VrQ6jlwL5oN8l14QlcNfg=
|
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w8PVh93nsPXa1VrQ6jlwL5oN8l14QlcNfg=
|
||||||
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
||||||
github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M=
|
github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M=
|
||||||
github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk=
|
github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk=
|
||||||
github.com/nbd-wtf/go-nostr v0.50.4 h1:KFMLxL07FPUzrCgllc2AKPP6INip+0MhAy6ZJxCwOyo=
|
github.com/nbd-wtf/go-nostr v0.51.7 h1:dGjtaaFQ1kA3H+vF8wt9a9WYl54K8C0JmVDf4cp+a4A=
|
||||||
github.com/nbd-wtf/go-nostr v0.50.4/go.mod h1:IoEUVJKvV2308WFhVu8f2OwGC32oEYpFYnV86EH8dqA=
|
github.com/nbd-wtf/go-nostr v0.51.7/go.mod h1:d6+DfvMWYG5pA3dmNMBJd6WCHVDDhkXbHqvfljf0Gzg=
|
||||||
github.com/nbd-wtf/go-nostr v0.51.2 h1:wQysG8omkF4LO7kcU6yoeCBBxD92SwUNab4TMeSuZZM=
|
github.com/nbd-wtf/go-nostr v0.51.8 h1:CIoS+YqChcm4e1L1rfMZ3/mIwTz4CwApM2qx7MHNzmE=
|
||||||
github.com/nbd-wtf/go-nostr v0.51.2/go.mod h1:9PcGOZ+e1VOaLvcK0peT4dbip+/eS+eTWXR3HuexQrA=
|
github.com/nbd-wtf/go-nostr v0.51.8/go.mod h1:d6+DfvMWYG5pA3dmNMBJd6WCHVDDhkXbHqvfljf0Gzg=
|
||||||
github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
|
github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
|
||||||
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
||||||
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
||||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
||||||
github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
|
github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
|
||||||
github.com/puzpuzpuz/xsync/v3 v3.5.0 h1:i+cMcpEDY1BkNm7lPDkCtE4oElsYLn+EKF8kAu2vXT4=
|
|
||||||
github.com/puzpuzpuz/xsync/v3 v3.5.0/go.mod h1:VjzYrABPabuM4KyBh1Ftq6u8nhwY5tBPKP9jpmh0nnA=
|
|
||||||
github.com/puzpuzpuz/xsync/v3 v3.5.1 h1:GJYJZwO6IdxN/IKbneznS6yPkVC+c3zyY/j19c++5Fg=
|
github.com/puzpuzpuz/xsync/v3 v3.5.1 h1:GJYJZwO6IdxN/IKbneznS6yPkVC+c3zyY/j19c++5Fg=
|
||||||
github.com/puzpuzpuz/xsync/v3 v3.5.1/go.mod h1:VjzYrABPabuM4KyBh1Ftq6u8nhwY5tBPKP9jpmh0nnA=
|
github.com/puzpuzpuz/xsync/v3 v3.5.1/go.mod h1:VjzYrABPabuM4KyBh1Ftq6u8nhwY5tBPKP9jpmh0nnA=
|
||||||
github.com/rs/cors v1.11.1 h1:eU3gRzXLRK57F5rKMGMZURNdIG4EoAmX8k94r9wXWHA=
|
github.com/rs/cors v1.11.1 h1:eU3gRzXLRK57F5rKMGMZURNdIG4EoAmX8k94r9wXWHA=
|
||||||
@@ -173,8 +171,8 @@ github.com/twitchyliquid64/golang-asm v0.15.1 h1:SU5vSMR7hnwNxj24w34ZyCi/FmDZTkS
|
|||||||
github.com/twitchyliquid64/golang-asm v0.15.1/go.mod h1:a1lVb/DtPvCB8fslRZhAngC2+aY1QWCk3Cedj/Gdt08=
|
github.com/twitchyliquid64/golang-asm v0.15.1/go.mod h1:a1lVb/DtPvCB8fslRZhAngC2+aY1QWCk3Cedj/Gdt08=
|
||||||
github.com/valyala/bytebufferpool v1.0.0 h1:GqA5TC/0021Y/b9FG4Oi9Mr3q7XYx6KllzawFIhcdPw=
|
github.com/valyala/bytebufferpool v1.0.0 h1:GqA5TC/0021Y/b9FG4Oi9Mr3q7XYx6KllzawFIhcdPw=
|
||||||
github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyCJ6HpOuEn7z0Csc=
|
github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyCJ6HpOuEn7z0Csc=
|
||||||
github.com/valyala/fasthttp v1.58.0 h1:GGB2dWxSbEprU9j0iMJHgdKYJVDyjrOwF9RE59PbRuE=
|
github.com/valyala/fasthttp v1.59.0 h1:Qu0qYHfXvPk1mSLNqcFtEk6DpxgA26hy6bmydotDpRI=
|
||||||
github.com/valyala/fasthttp v1.58.0/go.mod h1:SYXvHHaFp7QZHGKSHmoMipInhrI5StHrhDTYVEjK/Kw=
|
github.com/valyala/fasthttp v1.59.0/go.mod h1:GTxNb9Bc6r2a9D0TWNSPwDz78UxnTGBViY3xZNEqyYU=
|
||||||
github.com/xyproto/randomstring v1.0.5 h1:YtlWPoRdgMu3NZtP45drfy1GKoojuR7hmRcnhZqKjWU=
|
github.com/xyproto/randomstring v1.0.5 h1:YtlWPoRdgMu3NZtP45drfy1GKoojuR7hmRcnhZqKjWU=
|
||||||
github.com/xyproto/randomstring v1.0.5/go.mod h1:rgmS5DeNXLivK7YprL0pY+lTuhNQW3iGxZ18UQApw/E=
|
github.com/xyproto/randomstring v1.0.5/go.mod h1:rgmS5DeNXLivK7YprL0pY+lTuhNQW3iGxZ18UQApw/E=
|
||||||
go.opencensus.io v0.24.0 h1:y73uSU6J157QMP2kn2r30vwW1A2W2WFwSCGnAVxeaD0=
|
go.opencensus.io v0.24.0 h1:y73uSU6J157QMP2kn2r30vwW1A2W2WFwSCGnAVxeaD0=
|
||||||
@@ -192,8 +190,6 @@ golang.org/x/arch v0.15.0/go.mod h1:JmwW7aLIoRUKgaTzhkiEFxvcEiQGyOg9BMonBJUS7EE=
|
|||||||
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
||||||
golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
|
golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
|
||||||
golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
|
golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
|
||||||
golang.org/x/exp v0.0.0-20250210185358-939b2ce775ac h1:l5+whBCLH3iH2ZNHYLbAe58bo7yrN4mVcnkHDYz5vvs=
|
|
||||||
golang.org/x/exp v0.0.0-20250210185358-939b2ce775ac/go.mod h1:hH+7mtFmImwwcMvScyxUhjuVHR3HGaDPMn9rMSUUbxo=
|
|
||||||
golang.org/x/exp v0.0.0-20250305212735-054e65f0b394 h1:nDVHiLt8aIbd/VzvPWN6kSOPE7+F/fNFDSXLVYkE/Iw=
|
golang.org/x/exp v0.0.0-20250305212735-054e65f0b394 h1:nDVHiLt8aIbd/VzvPWN6kSOPE7+F/fNFDSXLVYkE/Iw=
|
||||||
golang.org/x/exp v0.0.0-20250305212735-054e65f0b394/go.mod h1:sIifuuw/Yco/y6yb6+bDNfyeQ/MdPUy/hKEMYQV17cM=
|
golang.org/x/exp v0.0.0-20250305212735-054e65f0b394/go.mod h1:sIifuuw/Yco/y6yb6+bDNfyeQ/MdPUy/hKEMYQV17cM=
|
||||||
golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
|
golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
|
||||||
@@ -205,8 +201,8 @@ golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73r
|
|||||||
golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
||||||
golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
||||||
golang.org/x/net v0.0.0-20201110031124-69a78807bb2b/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU=
|
golang.org/x/net v0.0.0-20201110031124-69a78807bb2b/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU=
|
||||||
golang.org/x/net v0.35.0 h1:T5GQRQb2y08kTAByq9L4/bz8cipCdA8FbRTXewonqY8=
|
golang.org/x/net v0.37.0 h1:1zLorHbz+LYj7MQlSf1+2tPIIgibq2eL5xkrGk6f+2c=
|
||||||
golang.org/x/net v0.35.0/go.mod h1:EglIi67kWsHKlRzzVMUD93VMSWGFOMSZgxFjparz1Qk=
|
golang.org/x/net v0.37.0/go.mod h1:ivrbrMbzFq5J41QOQh0siUuly180yBYtLp+CKbEaFx8=
|
||||||
golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
|
golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
|
||||||
golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
|
|||||||
78
handlers.go
78
handlers.go
@@ -6,9 +6,11 @@ import (
|
|||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
"errors"
|
"errors"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
"unsafe"
|
||||||
|
|
||||||
"github.com/bep/debounce"
|
"github.com/bep/debounce"
|
||||||
"github.com/fasthttp/websocket"
|
"github.com/fasthttp/websocket"
|
||||||
@@ -16,6 +18,7 @@ import (
|
|||||||
"github.com/nbd-wtf/go-nostr/nip42"
|
"github.com/nbd-wtf/go-nostr/nip42"
|
||||||
"github.com/nbd-wtf/go-nostr/nip45"
|
"github.com/nbd-wtf/go-nostr/nip45"
|
||||||
"github.com/nbd-wtf/go-nostr/nip45/hyperloglog"
|
"github.com/nbd-wtf/go-nostr/nip45/hyperloglog"
|
||||||
|
"github.com/nbd-wtf/go-nostr/nip70"
|
||||||
"github.com/nbd-wtf/go-nostr/nip77"
|
"github.com/nbd-wtf/go-nostr/nip77"
|
||||||
"github.com/nbd-wtf/go-nostr/nip77/negentropy"
|
"github.com/nbd-wtf/go-nostr/nip77/negentropy"
|
||||||
"github.com/puzpuzpuz/xsync/v3"
|
"github.com/puzpuzpuz/xsync/v3"
|
||||||
@@ -118,7 +121,7 @@ func (rl *Relay) HandleWebsocket(w http.ResponseWriter, r *http.Request) {
|
|||||||
smp := nostr.NewMessageParser()
|
smp := nostr.NewMessageParser()
|
||||||
|
|
||||||
for {
|
for {
|
||||||
typ, message, err := ws.conn.ReadMessage()
|
typ, msgb, err := ws.conn.ReadMessage()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if websocket.IsUnexpectedCloseError(
|
if websocket.IsUnexpectedCloseError(
|
||||||
err,
|
err,
|
||||||
@@ -139,11 +142,14 @@ func (rl *Relay) HandleWebsocket(w http.ResponseWriter, r *http.Request) {
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
// parse messages sequentially otherwise the world breaks
|
// this is safe because ReadMessage() will always create a new slice
|
||||||
|
message := unsafe.String(unsafe.SliceData(msgb), len(msgb))
|
||||||
|
|
||||||
|
// parse messages sequentially otherwise sonic breaks
|
||||||
envelope, err := smp.ParseMessage(message)
|
envelope, err := smp.ParseMessage(message)
|
||||||
|
|
||||||
// then delegate to the goroutine
|
// then delegate to the goroutine
|
||||||
go func(message []byte) {
|
go func(message string) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if err == nostr.UnknownLabel && rl.Negentropy {
|
if err == nostr.UnknownLabel && rl.Negentropy {
|
||||||
envelope = nip77.ParseNegMessage(message)
|
envelope = nip77.ParseNegMessage(message)
|
||||||
@@ -172,28 +178,31 @@ func (rl *Relay) HandleWebsocket(w http.ResponseWriter, r *http.Request) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// check NIP-70 protected
|
// check NIP-70 protected
|
||||||
for _, v := range env.Event.Tags {
|
if nip70.IsProtected(env.Event) {
|
||||||
if len(v) == 1 && v[0] == "-" {
|
authed := GetAuthed(ctx)
|
||||||
msg := "must be published by event author"
|
if authed == "" {
|
||||||
authed := GetAuthed(ctx)
|
RequestAuth(ctx)
|
||||||
if authed == "" {
|
ws.WriteJSON(nostr.OKEnvelope{
|
||||||
RequestAuth(ctx)
|
EventID: env.Event.ID,
|
||||||
ws.WriteJSON(nostr.OKEnvelope{
|
OK: false,
|
||||||
EventID: env.Event.ID,
|
Reason: "auth-required: must be published by authenticated event author",
|
||||||
OK: false,
|
})
|
||||||
Reason: "auth-required: " + msg,
|
return
|
||||||
})
|
} else if authed != env.Event.PubKey {
|
||||||
return
|
ws.WriteJSON(nostr.OKEnvelope{
|
||||||
}
|
EventID: env.Event.ID,
|
||||||
if authed != env.Event.PubKey {
|
OK: false,
|
||||||
ws.WriteJSON(nostr.OKEnvelope{
|
Reason: "blocked: must be published by event author",
|
||||||
EventID: env.Event.ID,
|
})
|
||||||
OK: false,
|
return
|
||||||
Reason: "blocked: " + msg,
|
|
||||||
})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
} else if nip70.HasEmbeddedProtected(env.Event) {
|
||||||
|
ws.WriteJSON(nostr.OKEnvelope{
|
||||||
|
EventID: env.Event.ID,
|
||||||
|
OK: false,
|
||||||
|
Reason: "blocked: can't repost nip70 protected",
|
||||||
|
})
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
srl := rl
|
srl := rl
|
||||||
@@ -208,9 +217,12 @@ func (rl *Relay) HandleWebsocket(w http.ResponseWriter, r *http.Request) {
|
|||||||
if env.Event.Kind == 5 {
|
if env.Event.Kind == 5 {
|
||||||
// this always returns "blocked: " whenever it returns an error
|
// this always returns "blocked: " whenever it returns an error
|
||||||
writeErr = srl.handleDeleteRequest(ctx, &env.Event)
|
writeErr = srl.handleDeleteRequest(ctx, &env.Event)
|
||||||
|
} else if nostr.IsEphemeralKind(env.Event.Kind) {
|
||||||
|
// this will also always return a prefixed reason
|
||||||
|
writeErr = srl.handleEphemeral(ctx, &env.Event)
|
||||||
} else {
|
} else {
|
||||||
// this will also always return a prefixed reason
|
// this will also always return a prefixed reason
|
||||||
skipBroadcast, writeErr = srl.AddEvent(ctx, &env.Event)
|
skipBroadcast, writeErr = srl.handleNormal(ctx, &env.Event)
|
||||||
}
|
}
|
||||||
|
|
||||||
var reason string
|
var reason string
|
||||||
@@ -220,9 +232,20 @@ func (rl *Relay) HandleWebsocket(w http.ResponseWriter, r *http.Request) {
|
|||||||
ovw(ctx, &env.Event)
|
ovw(ctx, &env.Event)
|
||||||
}
|
}
|
||||||
if !skipBroadcast {
|
if !skipBroadcast {
|
||||||
srl.notifyListeners(&env.Event)
|
n := srl.notifyListeners(&env.Event)
|
||||||
|
|
||||||
|
// the number of notified listeners matters in ephemeral events
|
||||||
|
if nostr.IsEphemeralKind(env.Event.Kind) {
|
||||||
|
if n == 0 {
|
||||||
|
ok = false
|
||||||
|
reason = "mute: no one was listening for this"
|
||||||
|
} else {
|
||||||
|
reason = "broadcasted to " + strconv.Itoa(n) + " listeners"
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
ok = false
|
||||||
reason = writeErr.Error()
|
reason = writeErr.Error()
|
||||||
if strings.HasPrefix(reason, "auth-required:") {
|
if strings.HasPrefix(reason, "auth-required:") {
|
||||||
RequestAuth(ctx)
|
RequestAuth(ctx)
|
||||||
@@ -237,14 +260,13 @@ func (rl *Relay) HandleWebsocket(w http.ResponseWriter, r *http.Request) {
|
|||||||
|
|
||||||
var total int64
|
var total int64
|
||||||
var hll *hyperloglog.HyperLogLog
|
var hll *hyperloglog.HyperLogLog
|
||||||
uneligibleForHLL := false
|
|
||||||
|
|
||||||
srl := rl
|
srl := rl
|
||||||
if rl.getSubRelayFromFilter != nil {
|
if rl.getSubRelayFromFilter != nil {
|
||||||
srl = rl.getSubRelayFromFilter(env.Filter)
|
srl = rl.getSubRelayFromFilter(env.Filter)
|
||||||
}
|
}
|
||||||
|
|
||||||
if offset := nip45.HyperLogLogEventPubkeyOffsetForFilter(env.Filter); offset != -1 && !uneligibleForHLL {
|
if offset := nip45.HyperLogLogEventPubkeyOffsetForFilter(env.Filter); offset != -1 {
|
||||||
total, hll = srl.handleCountRequestWithHLL(ctx, ws, env.Filter, offset)
|
total, hll = srl.handleCountRequestWithHLL(ctx, ws, env.Filter, offset)
|
||||||
} else {
|
} else {
|
||||||
total = srl.handleCountRequest(ctx, ws, env.Filter)
|
total = srl.handleCountRequest(ctx, ws, env.Filter)
|
||||||
|
|||||||
@@ -132,15 +132,20 @@ func (rl *Relay) removeClientAndListeners(ws *WebSocket) {
|
|||||||
delete(rl.clients, ws)
|
delete(rl.clients, ws)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (rl *Relay) notifyListeners(event *nostr.Event) {
|
// returns how many listeners were notified
|
||||||
|
func (rl *Relay) notifyListeners(event *nostr.Event) int {
|
||||||
|
count := 0
|
||||||
|
listenersloop:
|
||||||
for _, listener := range rl.listeners {
|
for _, listener := range rl.listeners {
|
||||||
if listener.filter.Matches(event) {
|
if listener.filter.Matches(event) {
|
||||||
for _, pb := range rl.PreventBroadcast {
|
for _, pb := range rl.PreventBroadcast {
|
||||||
if pb(listener.ws, event) {
|
if pb(listener.ws, event) {
|
||||||
return
|
continue listenersloop
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
listener.ws.WriteJSON(nostr.EventEnvelope{SubscriptionID: &listener.id, Event: *event})
|
listener.ws.WriteJSON(nostr.EventEnvelope{SubscriptionID: &listener.id, Event: *event})
|
||||||
|
count++
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return count
|
||||||
}
|
}
|
||||||
|
|||||||
8
nip86.go
8
nip86.go
@@ -3,6 +3,7 @@ package khatru
|
|||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"crypto/sha256"
|
"crypto/sha256"
|
||||||
|
"encoding/base64"
|
||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
@@ -12,7 +13,6 @@ import (
|
|||||||
"reflect"
|
"reflect"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/cloudwego/base64x"
|
|
||||||
"github.com/nbd-wtf/go-nostr"
|
"github.com/nbd-wtf/go-nostr"
|
||||||
"github.com/nbd-wtf/go-nostr/nip86"
|
"github.com/nbd-wtf/go-nostr/nip86"
|
||||||
)
|
)
|
||||||
@@ -72,7 +72,7 @@ func (rl *Relay) HandleNIP86(w http.ResponseWriter, r *http.Request) {
|
|||||||
goto respond
|
goto respond
|
||||||
}
|
}
|
||||||
|
|
||||||
evtj, err := base64x.StdEncoding.DecodeString(spl[1])
|
evtj, err := base64.StdEncoding.DecodeString(spl[1])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
resp.Error = "invalid base64 auth"
|
resp.Error = "invalid base64 auth"
|
||||||
goto respond
|
goto respond
|
||||||
@@ -86,10 +86,10 @@ func (rl *Relay) HandleNIP86(w http.ResponseWriter, r *http.Request) {
|
|||||||
goto respond
|
goto respond
|
||||||
}
|
}
|
||||||
|
|
||||||
if uTag := evt.Tags.GetFirst([]string{"u", ""}); uTag == nil || rl.getBaseURL(r) != (*uTag)[1] {
|
if uTag := evt.Tags.Find("u"); uTag == nil || rl.getBaseURL(r) != uTag[1] {
|
||||||
resp.Error = "invalid 'u' tag"
|
resp.Error = "invalid 'u' tag"
|
||||||
goto respond
|
goto respond
|
||||||
} else if pht := evt.Tags.GetFirst([]string{"payload", hex.EncodeToString(payloadHash[:])}); pht == nil {
|
} else if pht := evt.Tags.FindWithValue("payload", hex.EncodeToString(payloadHash[:])); pht == nil {
|
||||||
resp.Error = "invalid auth event payload hash"
|
resp.Error = "invalid auth event payload hash"
|
||||||
goto respond
|
goto respond
|
||||||
} else if evt.CreatedAt < nostr.Now()-30 {
|
} else if evt.CreatedAt < nostr.Now()-30 {
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/nbd-wtf/go-nostr"
|
"github.com/nbd-wtf/go-nostr"
|
||||||
|
"github.com/nbd-wtf/go-nostr/nip70"
|
||||||
)
|
)
|
||||||
|
|
||||||
// PreventTooManyIndexableTags returns a function that can be used as a RejectFilter that will reject
|
// PreventTooManyIndexableTags returns a function that can be used as a RejectFilter that will reject
|
||||||
@@ -107,3 +108,10 @@ func PreventTimestampsInTheFuture(threshold time.Duration) func(context.Context,
|
|||||||
func RejectEventsWithBase64Media(ctx context.Context, evt *nostr.Event) (bool, string) {
|
func RejectEventsWithBase64Media(ctx context.Context, evt *nostr.Event) (bool, string) {
|
||||||
return strings.Contains(evt.Content, "data:image/") || strings.Contains(evt.Content, "data:video/"), "event with base64 media"
|
return strings.Contains(evt.Content, "data:image/") || strings.Contains(evt.Content, "data:video/"), "event with base64 media"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func OnlyAllowNIP70ProtectedEvents(ctx context.Context, event *nostr.Event) (reject bool, msg string) {
|
||||||
|
if nip70.IsProtected(*event) {
|
||||||
|
return false, ""
|
||||||
|
}
|
||||||
|
return true, "blocked: we only accept events protected with the nip70 \"-\" tag"
|
||||||
|
}
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ package policies
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
|
||||||
"slices"
|
"slices"
|
||||||
|
|
||||||
"github.com/fiatjaf/khatru"
|
"github.com/fiatjaf/khatru"
|
||||||
|
|||||||
7
utils.go
7
utils.go
@@ -10,6 +10,7 @@ const (
|
|||||||
wsKey = iota
|
wsKey = iota
|
||||||
subscriptionIdKey
|
subscriptionIdKey
|
||||||
nip86HeaderAuthKey
|
nip86HeaderAuthKey
|
||||||
|
internalCallKey
|
||||||
)
|
)
|
||||||
|
|
||||||
func RequestAuth(ctx context.Context) {
|
func RequestAuth(ctx context.Context) {
|
||||||
@@ -40,6 +41,12 @@ func GetAuthed(ctx context.Context) string {
|
|||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// IsInternalCall returns true when a call to QueryEvents, for example, is being made because of a deletion
|
||||||
|
// or expiration request.
|
||||||
|
func IsInternalCall(ctx context.Context) bool {
|
||||||
|
return ctx.Value(internalCallKey) != nil
|
||||||
|
}
|
||||||
|
|
||||||
func GetIP(ctx context.Context) string {
|
func GetIP(ctx context.Context) string {
|
||||||
conn := GetConnection(ctx)
|
conn := GetConnection(ctx)
|
||||||
if conn == nil {
|
if conn == nil {
|
||||||
|
|||||||
10
websocket.go
10
websocket.go
@@ -33,12 +33,14 @@ type WebSocket struct {
|
|||||||
|
|
||||||
func (ws *WebSocket) WriteJSON(any any) error {
|
func (ws *WebSocket) WriteJSON(any any) error {
|
||||||
ws.mutex.Lock()
|
ws.mutex.Lock()
|
||||||
defer ws.mutex.Unlock()
|
err := ws.conn.WriteJSON(any)
|
||||||
return ws.conn.WriteJSON(any)
|
ws.mutex.Unlock()
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ws *WebSocket) WriteMessage(t int, b []byte) error {
|
func (ws *WebSocket) WriteMessage(t int, b []byte) error {
|
||||||
ws.mutex.Lock()
|
ws.mutex.Lock()
|
||||||
defer ws.mutex.Unlock()
|
err := ws.conn.WriteMessage(t, b)
|
||||||
return ws.conn.WriteMessage(t, b)
|
ws.mutex.Unlock()
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user