go-nostr/event_aux.go

257 lines
5.3 KiB
Go
Raw Permalink Normal View History

package nostr
import (
"encoding/json"
"errors"
"fmt"
2022-11-11 10:08:36 -03:00
"strings"
"time"
"github.com/valyala/fastjson"
"golang.org/x/exp/slices"
)
2022-11-11 10:08:36 -03:00
type Tag []string
// StartsWith checks if a tag contains a prefix.
// for example,
// ["p", "abcdef...", "wss://relay.com"]
// would match against
// ["p", "abcdef..."]
// or even
// ["p", "abcdef...", "wss://"]
func (tag Tag) StartsWith(prefix []string) bool {
prefixLen := len(prefix)
if prefixLen > len(tag) {
return false
}
// check initial elements for equality
for i := 0; i < prefixLen-1; i++ {
if prefix[i] != tag[i] {
return false
}
}
// check last element just for a prefix
return strings.HasPrefix(tag[prefixLen-1], prefix[prefixLen-1])
}
type Tags []Tag
// GetFirst gets the first tag in tags that matches tagPrefix, see [Tag.StartsWith]
func (tags Tags) GetFirst(tagPrefix []string) *Tag {
for _, v := range tags {
if v.StartsWith(tagPrefix) {
return &v
}
}
return nil
}
// GetLast gets the last tag in tags that matches tagPrefix, see [Tag.StartsWith]
func (tags Tags) GetLast(tagPrefix []string) *Tag {
for i := len(tags) - 1; i >= 0; i-- {
v := tags[i]
if v.StartsWith(tagPrefix) {
return &v
}
}
return nil
}
func (tags Tags) GetAll(tagPrefix []string) Tags {
result := make(Tags, 0, len(tags))
for _, v := range tags {
if v.StartsWith(tagPrefix) {
result = append(result, v)
}
}
return result
}
func (tags Tags) FilterOut(tagPrefix []string) Tags {
filtered := make(Tags, 0, len(tags))
for _, v := range tags {
if !v.StartsWith(tagPrefix) {
filtered = append(filtered, v)
}
}
return filtered
}
func (tags Tags) AppendUnique(tag Tag) Tags {
if tags.GetFirst(tag) == nil {
return append(tags.FilterOut(tag), tag)
} else {
return tags
}
}
func (t *Tags) Scan(src interface{}) error {
var jtags []byte = make([]byte, 0)
switch v := src.(type) {
case []byte:
jtags = v
case string:
jtags = []byte(v)
default:
return errors.New("couldn't scan tags, it's not a json string")
}
json.Unmarshal(jtags, &t)
return nil
}
func (tags Tags) ContainsAny(tagName string, values []string) bool {
for _, tag := range tags {
if len(tag) < 2 {
continue
}
if tag[0] != tagName {
continue
}
if slices.Contains(values, tag[1]) {
return true
}
}
return false
}
func (evt *Event) UnmarshalJSON(payload []byte) error {
var fastjsonParser fastjson.Parser
parsed, err := fastjsonParser.ParseBytes(payload)
if err != nil {
return fmt.Errorf("failed to parse event: %w", err)
}
obj, err := parsed.Object()
if err != nil {
return fmt.Errorf("event is not an object")
}
// prepare this to receive any extra property that may serialized along with the event
evt.extra = make(map[string]any)
var visiterr error
obj.Visit(func(k []byte, v *fastjson.Value) {
key := string(k)
switch key {
case "id":
id, err := v.StringBytes()
if err != nil {
visiterr = fmt.Errorf("invalid 'id' field: %w", err)
}
evt.ID = string(id)
case "pubkey":
id, err := v.StringBytes()
if err != nil {
visiterr = fmt.Errorf("invalid 'pubkey' field: %w", err)
}
evt.PubKey = string(id)
case "created_at":
val, err := v.Int64()
if err != nil {
visiterr = fmt.Errorf("invalid 'created_at' field: %w", err)
}
evt.CreatedAt = time.Unix(val, 0)
case "kind":
kind, err := v.Int64()
if err != nil {
visiterr = fmt.Errorf("invalid 'kind' field: %w", err)
}
evt.Kind = int(kind)
case "tags":
evt.Tags, err = fastjsonArrayToTags(v)
if err != nil {
visiterr = fmt.Errorf("invalid '%s' field: %w", key, err)
}
case "content":
id, err := v.StringBytes()
if err != nil {
visiterr = fmt.Errorf("invalid 'content' field: %w", err)
}
evt.Content = string(id)
case "sig":
id, err := v.StringBytes()
if err != nil {
visiterr = fmt.Errorf("invalid 'sig' field: %w", err)
}
evt.Sig = string(id)
default:
var anyValue any
json.Unmarshal(v.MarshalTo(nil), anyValue)
evt.extra[key] = anyValue
}
})
if visiterr != nil {
return visiterr
}
return nil
}
func (evt Event) MarshalJSON() ([]byte, error) {
var arena fastjson.Arena
o := arena.NewObject()
o.Set("id", arena.NewString(evt.ID))
o.Set("pubkey", arena.NewString(evt.PubKey))
o.Set("created_at", arena.NewNumberInt(int(evt.CreatedAt.Unix())))
o.Set("kind", arena.NewNumberInt(evt.Kind))
o.Set("tags", tagsToFastjsonArray(&arena, evt.Tags))
o.Set("content", arena.NewString(evt.Content))
o.Set("sig", arena.NewString(evt.Sig))
for k, v := range evt.extra {
b, _ := json.Marshal(v)
if val, err := fastjson.ParseBytes(b); err == nil {
o.Set(k, val)
}
}
return o.MarshalTo(nil), nil
}
func fastjsonArrayToTags(v *fastjson.Value) (Tags, error) {
arr, err := v.Array()
if err != nil {
return nil, err
}
2022-11-11 10:08:36 -03:00
sll := make(Tags, len(arr))
for i, v := range arr {
subarr, err := v.Array()
if err != nil {
return nil, err
}
2022-11-11 10:08:36 -03:00
sl := make(Tag, len(subarr))
for j, subv := range subarr {
sb, err := subv.StringBytes()
if err != nil {
return nil, err
}
sl[j] = string(sb)
}
sll[i] = sl
}
2022-11-11 10:08:36 -03:00
return sll, nil
}
func tagsToFastjsonArray(arena *fastjson.Arena, tags Tags) *fastjson.Value {
jtags := arena.NewArray()
for i, v := range tags {
arr := arena.NewArray()
for j, subv := range v {
arr.SetArrayItem(j, arena.NewString(subv))
}
jtags.SetArrayItem(i, arr)
}
return jtags
}