import { NostrEvent, NostrFilter, NSchema as n } from '@nostrify/nostrify'; import { z } from 'zod'; import { AppController } from '@/app.ts'; import { AdminSigner } from '@/signers/AdminSigner.ts'; import { booleanParamSchema } from '@/schema.ts'; import { Conf } from '@/config.ts'; import { Storages } from '@/storages.ts'; import { hydrateEvents } from '@/storages/hydrate.ts'; import { createEvent, paginated, paginationSchema, parseBody } from '@/utils/api.ts'; import { renderNameRequest } from '@/views/ditto.ts'; import { getZapSplits } from '@/utils/zap-split.ts'; import { updateListAdminEvent } from '@/utils/api.ts'; import { addTag } from '@/utils/tags.ts'; import { deleteTag } from '@/utils/tags.ts'; const markerSchema = z.enum(['read', 'write']); const relaySchema = z.object({ url: z.string().url(), marker: markerSchema.optional(), }); type RelayEntity = z.infer; export const adminRelaysController: AppController = async (c) => { const store = await Storages.db(); const [event] = await store.query([ { kinds: [10002], authors: [Conf.pubkey], limit: 1 }, ]); if (!event) { return c.json([]); } return c.json(renderRelays(event)); }; export const adminSetRelaysController: AppController = async (c) => { const store = await Storages.db(); const relays = relaySchema.array().parse(await c.req.json()); const event = await new AdminSigner().signEvent({ kind: 10002, tags: relays.map(({ url, marker }) => marker ? ['r', url, marker] : ['r', url]), content: '', created_at: Math.floor(Date.now() / 1000), }); await store.event(event); return c.json(renderRelays(event)); }; /** Render Ditto API relays from a NIP-65 event. */ function renderRelays(event: NostrEvent): RelayEntity[] { return event.tags.reduce((acc, [name, url, marker]) => { if (name === 'r') { const relay: RelayEntity = { url, marker: markerSchema.safeParse(marker).success ? marker as 'read' | 'write' : undefined, }; acc.push(relay); } return acc; }, [] as RelayEntity[]); } const nameRequestSchema = z.object({ name: z.string().email(), reason: z.string().max(500).optional(), }); export const nameRequestController: AppController = async (c) => { const store = await Storages.db(); const signer = c.get('signer')!; const pubkey = await signer.getPublicKey(); const { name, reason } = nameRequestSchema.parse(await c.req.json()); const [existing] = await store.query([{ kinds: [3036], authors: [pubkey], '#r': [name], limit: 1 }]); if (existing) { return c.json({ error: 'Name request already exists' }, 400); } const event = await createEvent({ kind: 3036, content: reason, tags: [ ['r', name], ['L', 'nip05.domain'], ['l', name.split('@')[1], 'nip05.domain'], ['p', Conf.pubkey], ], }, c); await hydrateEvents({ events: [event], store: await Storages.db() }); const nameRequest = await renderNameRequest(event); return c.json(nameRequest); }; const nameRequestsSchema = z.object({ approved: booleanParamSchema.optional(), rejected: booleanParamSchema.optional(), }); export const nameRequestsController: AppController = async (c) => { const store = await Storages.db(); const signer = c.get('signer')!; const pubkey = await signer.getPublicKey(); const params = paginationSchema.parse(c.req.query()); const { approved, rejected } = nameRequestsSchema.parse(c.req.query()); const filter: NostrFilter = { kinds: [30383], authors: [Conf.pubkey], '#k': ['3036'], '#p': [pubkey], ...params, }; if (approved) { filter['#n'] = ['approved']; } if (rejected) { filter['#n'] = ['rejected']; } const orig = await store.query([filter]); const ids = new Set(); for (const event of orig) { const d = event.tags.find(([name]) => name === 'd')?.[1]; if (d) { ids.add(d); } } if (!ids.size) { return c.json([]); } const events = await store.query([{ kinds: [3036], ids: [...ids], authors: [pubkey] }]) .then((events) => hydrateEvents({ store, events: events, signal: c.req.raw.signal })); const nameRequests = await Promise.all( events.map((event) => renderNameRequest(event)), ); return paginated(c, orig, nameRequests); }; const zapSplitSchema = z.record( n.id(), z.object({ amount: z.number().int().min(1).max(100), message: z.string().max(500), }), ); export const updateZapSplitsController: AppController = async (c) => { const body = await parseBody(c.req.raw); const result = zapSplitSchema.safeParse(body); const store = c.get('store'); if (!result.success) { return c.json({ error: result.error }, 400); } const zap_split = await getZapSplits(store, Conf.pubkey); if (!zap_split) { return c.json({ error: 'Zap split not activated, restart the server.' }, 404); } const { data } = result; const pubkeys = Object.keys(data); if (pubkeys.length < 1) { return c.json(200); } await updateListAdminEvent( { kinds: [30078], authors: [Conf.pubkey], '#d': ['pub.ditto.zapSplits'], limit: 1 }, (tags) => pubkeys.reduce((accumulator, pubkey) => { return addTag(accumulator, ['p', pubkey, data[pubkey].amount.toString(), data[pubkey].message]); }, tags), c, ); return c.json(200); }; const deleteZapSplitSchema = z.array(n.id()).min(1); export const deleteZapSplitsController: AppController = async (c) => { const body = await parseBody(c.req.raw); const result = deleteZapSplitSchema.safeParse(body); const store = c.get('store'); if (!result.success) { return c.json({ error: result.error }, 400); } const zap_split = await getZapSplits(store, Conf.pubkey); if (!zap_split) { return c.json({ error: 'Zap split not activated, restart the server.' }, 404); } const { data } = result; await updateListAdminEvent( { kinds: [30078], authors: [Conf.pubkey], '#d': ['pub.ditto.zapSplits'], limit: 1 }, (tags) => data.reduce((accumulator, currentValue) => { return deleteTag(accumulator, ['p', currentValue]); }, tags), c, ); return c.json(200); };