grain/server/handlers/event.go

156 lines
4.4 KiB
Go
Raw Normal View History

package handlers
import (
"context"
"encoding/json"
"fmt"
2024-07-30 15:27:38 +00:00
"grain/config"
"grain/server/db"
"grain/server/handlers/kinds"
"grain/server/handlers/response"
"grain/server/utils"
relay "grain/server/types"
"golang.org/x/net/websocket"
)
2024-07-25 13:57:24 +00:00
func HandleEvent(ws *websocket.Conn, message []interface{}) {
if len(message) != 2 {
fmt.Println("Invalid EVENT message format")
2024-07-27 14:06:34 +00:00
response.SendNotice(ws, "", "Invalid EVENT message format")
return
}
eventData, ok := message[1].(map[string]interface{})
if !ok {
fmt.Println("Invalid event data format")
2024-07-27 14:06:34 +00:00
response.SendNotice(ws, "", "Invalid event data format")
return
}
eventBytes, err := json.Marshal(eventData)
if err != nil {
fmt.Println("Error marshaling event data:", err)
2024-07-27 14:06:34 +00:00
response.SendNotice(ws, "", "Error marshaling event data")
return
}
var evt relay.Event
err = json.Unmarshal(eventBytes, &evt)
if err != nil {
fmt.Println("Error unmarshaling event data:", err)
2024-07-27 14:06:34 +00:00
response.SendNotice(ws, "", "Error unmarshaling event data")
return
}
2024-07-27 14:06:34 +00:00
eventSize := len(eventBytes) // Calculate event size
HandleKind(context.TODO(), evt, ws, eventSize)
fmt.Println("Event processed:", evt.ID)
}
2024-07-27 14:06:34 +00:00
func HandleKind(ctx context.Context, evt relay.Event, ws *websocket.Conn, eventSize int) {
if !utils.CheckSignature(evt) {
2024-07-25 14:19:42 +00:00
response.SendOK(ws, evt.ID, false, "invalid: signature verification failed")
return
}
collection := db.GetCollection(evt.Kind)
2024-07-30 15:27:38 +00:00
rateLimiter := config.GetRateLimiter()
sizeLimiter := config.GetSizeLimiter()
2024-07-27 14:06:34 +00:00
2024-08-04 18:02:53 +00:00
if config.GetConfig().DomainWhitelist.Enabled {
domains := config.GetConfig().DomainWhitelist.Domains
pubkeys, err := utils.FetchPubkeysFromDomains(domains)
if err != nil {
fmt.Println("Error fetching pubkeys from domains:", err)
response.SendNotice(ws, "", "Error fetching pubkeys from domains")
return
}
for _, pubkey := range pubkeys {
config.GetConfig().PubkeyWhitelist.Pubkeys = append(config.GetConfig().PubkeyWhitelist.Pubkeys, pubkey)
}
}
// Check against manual blacklist
if blacklisted, msg := utils.CheckBlacklist(evt.PubKey, evt.Content); blacklisted {
response.SendOK(ws, evt.ID, false, msg)
return
}
2024-08-03 20:18:34 +00:00
// Check if the kind is whitelisted
if config.GetConfig().KindWhitelist.Enabled && !utils.IsKindWhitelisted(evt.Kind) {
2024-08-03 20:18:34 +00:00
response.SendOK(ws, evt.ID, false, "not allowed: event kind is not whitelisted")
return
}
// Check pubkey/npub whitelist only if the kind is not whitelisted
if config.GetConfig().PubkeyWhitelist.Enabled && !utils.IsPubKeyWhitelisted(evt.PubKey) {
2024-08-03 20:18:34 +00:00
response.SendOK(ws, evt.ID, false, "not allowed: pubkey or npub is not whitelisted")
2024-08-03 18:27:58 +00:00
return
}
2024-07-27 14:06:34 +00:00
category := determineCategory(evt.Kind)
2024-07-25 13:57:24 +00:00
2024-07-26 14:02:34 +00:00
if allowed, msg := rateLimiter.AllowEvent(evt.Kind, category); !allowed {
response.SendOK(ws, evt.ID, false, msg)
2024-07-25 13:57:24 +00:00
return
}
2024-07-26 20:46:01 +00:00
if allowed, msg := sizeLimiter.AllowSize(evt.Kind, eventSize); !allowed {
response.SendOK(ws, evt.ID, false, msg)
return
}
var err error
switch {
case evt.Kind == 0:
err = kinds.HandleKind0(ctx, evt, collection, ws)
case evt.Kind == 1:
2024-07-27 14:06:34 +00:00
err = kinds.HandleKind1(ctx, evt, collection, ws)
2024-07-24 20:54:11 +00:00
case evt.Kind == 2:
2024-07-27 14:06:34 +00:00
err = kinds.HandleKind2(ctx, evt, ws)
case evt.Kind == 3:
err = kinds.HandleReplaceableKind(ctx, evt, collection, ws)
2024-07-27 19:47:37 +00:00
case evt.Kind == 5:
err = kinds.HandleKind5(ctx, evt, db.GetClient(), ws)
2024-07-24 20:54:11 +00:00
case evt.Kind >= 4 && evt.Kind < 45:
2024-07-27 14:06:34 +00:00
err = kinds.HandleRegularKind(ctx, evt, collection, ws)
2024-07-24 20:54:11 +00:00
case evt.Kind >= 1000 && evt.Kind < 10000:
2024-07-27 14:06:34 +00:00
err = kinds.HandleRegularKind(ctx, evt, collection, ws)
case evt.Kind >= 10000 && evt.Kind < 20000:
err = kinds.HandleReplaceableKind(ctx, evt, collection, ws)
case evt.Kind >= 20000 && evt.Kind < 30000:
fmt.Println("Ephemeral event received and ignored:", evt.ID)
case evt.Kind >= 30000 && evt.Kind < 40000:
err = kinds.HandleParameterizedReplaceableKind(ctx, evt, collection, ws)
default:
2024-07-25 14:05:33 +00:00
err = kinds.HandleUnknownKind(ctx, evt, collection, ws)
}
if err != nil {
2024-07-25 14:19:42 +00:00
response.SendOK(ws, evt.ID, false, fmt.Sprintf("error: %v", err))
return
}
2024-07-25 14:19:42 +00:00
response.SendOK(ws, evt.ID, true, "")
2024-07-25 13:57:24 +00:00
}
2024-07-27 14:06:34 +00:00
func determineCategory(kind int) string {
switch {
case kind == 0, kind == 3, kind >= 10000 && kind < 20000:
return "replaceable"
case kind == 1, kind >= 4 && kind < 45, kind >= 1000 && kind < 10000:
return "regular"
case kind == 2:
return "deprecated"
case kind >= 20000 && kind < 30000:
return "ephemeral"
case kind >= 30000 && kind < 40000:
return "parameterized_replaceable"
default:
return "unknown"
}
}
2024-08-03 18:27:58 +00:00