Files
zooid/zooid/groups.go
T
2025-11-13 15:17:34 -08:00

337 lines
6.8 KiB
Go

package zooid
import (
"fiatjaf.com/nostr"
"fiatjaf.com/nostr/nip29"
"slices"
)
// Utils
func GetGroupIDFromEvent(event nostr.Event) string {
var tagName string
if slices.Contains(nip29.MetadataEventKinds, event.Kind) {
tagName = "d"
} else {
tagName = "h"
}
tag := event.Tags.Find(tagName)
if tag != nil {
return tag[1]
}
return ""
}
// Struct definition
type GroupStore struct {
Config *Config
Events *EventStore
Management *ManagementStore
}
// Metadata
func (g *GroupStore) GetMetadata(h string) (nostr.Event, bool) {
filter := nostr.Filter{
Kinds: []nostr.Kind{nostr.KindSimpleGroupMetadata},
Tags: nostr.TagMap{
"d": []string{h},
},
}
for event := range g.Events.QueryEvents(filter, 1) {
return event, true
}
return nostr.Event{}, false
}
func (g *GroupStore) UpdateMetadata(event nostr.Event) error {
tags := nostr.Tags{}
for _, tag := range event.Tags {
if len(tag) >= 2 && tag[0] == "h" {
tags = append(tags, nostr.Tag{"d", tag[1]})
} else {
tags = append(tags, tag)
}
}
metadataEvent := nostr.Event{
Kind: nostr.KindSimpleGroupMetadata,
CreatedAt: event.CreatedAt,
Tags: tags,
}
return g.Events.SignAndStoreEvent(&metadataEvent, true)
}
// Deletion
func (g *GroupStore) DeleteGroup(h string) {
filters := []nostr.Filter{
{
Kinds: nip29.MetadataEventKinds,
Tags: nostr.TagMap{
"d": []string{h},
},
},
{
Tags: nostr.TagMap{
"h": []string{h},
},
},
}
for _, filter := range filters {
for event := range g.Events.QueryEvents(filter, 0) {
if event.Kind != nostr.KindSimpleGroupDeleteGroup {
g.Events.DeleteEvent(event.ID)
}
}
}
}
// Admins
func (g *GroupStore) IsAdmin(h string, pubkey nostr.PubKey) bool {
return g.Management.IsAdmin(pubkey)
}
func (g *GroupStore) GetAdmins(h string) []nostr.PubKey {
return g.Management.GetAdmins()
}
func (g *GroupStore) UpdateAdminsList(h string) error {
tags := nostr.Tags{
nostr.Tag{"-"},
nostr.Tag{"d", h},
}
for _, pubkey := range g.GetAdmins(h) {
tags = append(tags, nostr.Tag{"p", pubkey.Hex()})
}
event := nostr.Event{
Kind: nostr.KindSimpleGroupAdmins,
CreatedAt: nostr.Now(),
Tags: tags,
}
return g.Events.SignAndStoreEvent(&event, true)
}
// Membership
func (g *GroupStore) AddMember(h string, pubkey nostr.PubKey) error {
event := nostr.Event{
Kind: nostr.KindSimpleGroupPutUser,
CreatedAt: nostr.Now(),
Tags: nostr.Tags{
nostr.Tag{"p", pubkey.Hex()},
nostr.Tag{"h", h},
},
}
return g.Events.SignAndStoreEvent(&event, true)
}
func (g *GroupStore) RemoveMember(h string, pubkey nostr.PubKey) error {
event := nostr.Event{
Kind: nostr.KindSimpleGroupRemoveUser,
CreatedAt: nostr.Now(),
Tags: nostr.Tags{
nostr.Tag{"p", pubkey.Hex()},
nostr.Tag{"h", h},
},
}
return g.Events.SignAndStoreEvent(&event, true)
}
func (g *GroupStore) IsMember(h string, pubkey nostr.PubKey) bool {
filter := nostr.Filter{
Kinds: []nostr.Kind{nostr.KindSimpleGroupPutUser, nostr.KindSimpleGroupRemoveUser},
Tags: nostr.TagMap{
"p": []string{pubkey.Hex()},
"h": []string{h},
},
}
for event := range g.Events.QueryEvents(filter, 1) {
if event.Kind == nostr.KindSimpleGroupPutUser {
return true
}
if event.Kind == nostr.KindSimpleGroupRemoveUser {
return false
}
}
return false
}
func (g *GroupStore) GetMembers(h string) []nostr.PubKey {
filter := nostr.Filter{
Kinds: []nostr.Kind{nostr.KindSimpleGroupPutUser, nostr.KindSimpleGroupRemoveUser},
Tags: nostr.TagMap{
"h": []string{h},
},
}
members := make(map[nostr.PubKey]struct{})
for _, event := range Reversed(slices.Collect(g.Events.QueryEvents(filter, 0))) {
for tag := range event.Tags.FindAll("p") {
if pubkey, err := nostr.PubKeyFromHex(tag[1]); err == nil {
if event.Kind == nostr.KindSimpleGroupPutUser {
members[pubkey] = struct{}{}
} else {
delete(members, pubkey)
}
}
}
}
return Keys(members)
}
func (g *GroupStore) UpdateMembersList(h string) error {
tags := nostr.Tags{
nostr.Tag{"-"},
nostr.Tag{"d", h},
}
for _, pubkey := range g.GetMembers(h) {
tags = append(tags, nostr.Tag{"p", pubkey.Hex()})
}
event := nostr.Event{
Kind: nostr.KindSimpleGroupMembers,
CreatedAt: nostr.Now(),
Tags: tags,
}
return g.Events.SignAndStoreEvent(&event, true)
}
// Other stuff
func (g *GroupStore) HasAccess(h string, pubkey nostr.PubKey) bool {
return g.Config.CanManage(pubkey) || g.IsAdmin(h, pubkey) || g.IsMember(h, pubkey)
}
func (g *GroupStore) IsGroupEvent(event nostr.Event) bool {
if slices.Contains(nip29.MetadataEventKinds, event.Kind) {
return true
}
if slices.Contains(nip29.ModerationEventKinds, event.Kind) {
return true
}
joinKinds := []nostr.Kind{
nostr.KindSimpleGroupJoinRequest,
nostr.KindSimpleGroupLeaveRequest,
}
if slices.Contains(joinKinds, event.Kind) {
return true
}
return GetGroupIDFromEvent(event) != ""
}
func (g *GroupStore) CanRead(pubkey nostr.PubKey, event nostr.Event) bool {
if !g.Config.Groups.Enabled {
return false
}
h := GetGroupIDFromEvent(event)
meta, found := g.GetMetadata(h)
if !found {
return false
}
if HasTag(meta.Tags, "hidden") && !g.HasAccess(h, pubkey) {
return false
}
if event.Kind == nostr.KindSimpleGroupMetadata {
return true
}
if event.Kind == nostr.KindSimpleGroupDeleteGroup {
return true
}
if HasTag(meta.Tags, "private") && !g.HasAccess(h, pubkey) {
return false
}
return true
}
func (g *GroupStore) CheckWrite(event nostr.Event) string {
if !g.Config.Groups.Enabled {
return "invalid: groups are not enabled"
}
if slices.Contains(nip29.MetadataEventKinds, event.Kind) {
return "invalid: group metadata cannot be set directly"
}
h := GetGroupIDFromEvent(event)
meta, found := g.GetMetadata(h)
if event.Kind == nostr.KindSimpleGroupCreateGroup {
if found {
return "invalid: that group already exists"
}
} else if !found {
return "invalid: group not found"
}
if slices.Contains(nip29.ModerationEventKinds, event.Kind) && !g.Config.CanManage(event.PubKey) {
return "restricted: you are not authorized to manage groups"
}
if HasTag(meta.Tags, "hidden") && !g.HasAccess(h, event.PubKey) {
return "invalid: group not found"
}
if event.Kind == nostr.KindSimpleGroupJoinRequest {
if g.IsMember(h, event.PubKey) {
return "duplicate: already a member"
} else {
return ""
}
}
if event.Kind == nostr.KindSimpleGroupLeaveRequest {
if !g.IsMember(h, event.PubKey) {
return "duplicate: not currently a member"
} else {
return ""
}
}
if HasTag(meta.Tags, "closed") && !g.HasAccess(h, event.PubKey) {
return "restricted: you are not a member of that group"
}
return ""
}
// Middleware
func (g *GroupStore) Enable(instance *Instance) {
instance.Relay.Info.SupportedNIPs = append(instance.Relay.Info.SupportedNIPs, 29)
}