Merge branch 'rm-userdata' into 'main'
Remove UserData type, hydrate the event instead See merge request soapbox-pub/ditto!104
This commit is contained in:
commit
789ce80990
|
@ -1,8 +1,8 @@
|
||||||
import { Conf } from '@/config.ts';
|
import { Conf } from '@/config.ts';
|
||||||
import { type Event, type Filter, matchFilters, stringifyStable, z } from '@/deps.ts';
|
import { type Event, type Filter, matchFilters, stringifyStable, z } from '@/deps.ts';
|
||||||
import { nostrIdSchema } from '@/schemas/nostr.ts';
|
|
||||||
import { type EventData } from '@/types.ts';
|
|
||||||
import { isReplaceableKind } from '@/kinds.ts';
|
import { isReplaceableKind } from '@/kinds.ts';
|
||||||
|
import { nostrIdSchema } from '@/schemas/nostr.ts';
|
||||||
|
import { type DittoEvent } from '@/storages/types.ts';
|
||||||
|
|
||||||
/** Additional properties that may be added by Ditto to events. */
|
/** Additional properties that may be added by Ditto to events. */
|
||||||
type Relation = 'author' | 'author_stats' | 'event_stats';
|
type Relation = 'author' | 'author_stats' | 'event_stats';
|
||||||
|
@ -22,8 +22,8 @@ type AuthorMicrofilter = { kinds: [0]; authors: [Event['pubkey']] };
|
||||||
/** Filter to get one specific event. */
|
/** Filter to get one specific event. */
|
||||||
type MicroFilter = IdMicrofilter | AuthorMicrofilter;
|
type MicroFilter = IdMicrofilter | AuthorMicrofilter;
|
||||||
|
|
||||||
function matchDittoFilter(filter: DittoFilter, event: Event, data: EventData): boolean {
|
function matchDittoFilter(filter: DittoFilter, event: DittoEvent): boolean {
|
||||||
if (filter.local && !(data.user || event.pubkey === Conf.pubkey)) {
|
if (filter.local && !(event.user || event.pubkey === Conf.pubkey)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -34,9 +34,9 @@ function matchDittoFilter(filter: DittoFilter, event: Event, data: EventData): b
|
||||||
* Similar to nostr-tools `matchFilters`, but supports Ditto's custom keys.
|
* Similar to nostr-tools `matchFilters`, but supports Ditto's custom keys.
|
||||||
* Database calls are needed to look up the extra data, so it's passed in as an argument.
|
* Database calls are needed to look up the extra data, so it's passed in as an argument.
|
||||||
*/
|
*/
|
||||||
function matchDittoFilters(filters: DittoFilter[], event: Event, data: EventData): boolean {
|
function matchDittoFilters(filters: DittoFilter[], event: DittoEvent): boolean {
|
||||||
for (const filter of filters) {
|
for (const filter of filters) {
|
||||||
if (matchDittoFilter(filter, event, data)) {
|
if (matchDittoFilter(filter, event)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,7 +2,6 @@ import { Conf } from '@/config.ts';
|
||||||
import { encryptAdmin } from '@/crypto.ts';
|
import { encryptAdmin } from '@/crypto.ts';
|
||||||
import { addRelays } from '@/db/relays.ts';
|
import { addRelays } from '@/db/relays.ts';
|
||||||
import { deleteAttachedMedia } from '@/db/unattached-media.ts';
|
import { deleteAttachedMedia } from '@/db/unattached-media.ts';
|
||||||
import { findUser } from '@/db/users.ts';
|
|
||||||
import { Debug, type Event, LNURL } from '@/deps.ts';
|
import { Debug, type Event, LNURL } from '@/deps.ts';
|
||||||
import { isEphemeralKind } from '@/kinds.ts';
|
import { isEphemeralKind } from '@/kinds.ts';
|
||||||
import { isLocallyFollowed } from '@/queries.ts';
|
import { isLocallyFollowed } from '@/queries.ts';
|
||||||
|
@ -10,13 +9,13 @@ import { updateStats } from '@/stats.ts';
|
||||||
import { client, eventsDB, memorelay, reqmeister } from '@/storages.ts';
|
import { client, eventsDB, memorelay, reqmeister } from '@/storages.ts';
|
||||||
import { Sub } from '@/subs.ts';
|
import { Sub } from '@/subs.ts';
|
||||||
import { getTagSet } from '@/tags.ts';
|
import { getTagSet } from '@/tags.ts';
|
||||||
import { type EventData } from '@/types.ts';
|
|
||||||
import { eventAge, isRelay, nostrDate, nostrNow, Time } from '@/utils.ts';
|
import { eventAge, isRelay, nostrDate, nostrNow, Time } from '@/utils.ts';
|
||||||
import { fetchWorker } from '@/workers/fetch.ts';
|
import { fetchWorker } from '@/workers/fetch.ts';
|
||||||
import { TrendsWorker } from '@/workers/trends.ts';
|
import { TrendsWorker } from '@/workers/trends.ts';
|
||||||
import { verifySignatureWorker } from '@/workers/verify.ts';
|
import { verifySignatureWorker } from '@/workers/verify.ts';
|
||||||
import { signAdminEvent } from '@/sign.ts';
|
import { signAdminEvent } from '@/sign.ts';
|
||||||
import { lnurlCache } from '@/utils/lnurl.ts';
|
import { lnurlCache } from '@/utils/lnurl.ts';
|
||||||
|
import { DittoEvent } from '@/storages/types.ts';
|
||||||
|
|
||||||
const debug = Debug('ditto:pipeline');
|
const debug = Debug('ditto:pipeline');
|
||||||
|
|
||||||
|
@ -24,24 +23,24 @@ const debug = Debug('ditto:pipeline');
|
||||||
* Common pipeline function to process (and maybe store) events.
|
* Common pipeline function to process (and maybe store) events.
|
||||||
* It is idempotent, so it can be called multiple times for the same event.
|
* It is idempotent, so it can be called multiple times for the same event.
|
||||||
*/
|
*/
|
||||||
async function handleEvent(event: Event): Promise<void> {
|
async function handleEvent(event: DittoEvent): Promise<void> {
|
||||||
const signal = AbortSignal.timeout(5000);
|
const signal = AbortSignal.timeout(5000);
|
||||||
if (!(await verifySignatureWorker(event))) return;
|
if (!(await verifySignatureWorker(event))) return;
|
||||||
const wanted = reqmeister.isWanted(event);
|
const wanted = reqmeister.isWanted(event);
|
||||||
if (await encounterEvent(event)) return;
|
if (await encounterEvent(event)) return;
|
||||||
debug(`Event<${event.kind}> ${event.id}`);
|
debug(`Event<${event.kind}> ${event.id}`);
|
||||||
const data = await getEventData(event);
|
await hydrateEvent(event);
|
||||||
|
|
||||||
await Promise.all([
|
await Promise.all([
|
||||||
storeEvent(event, data, { force: wanted }),
|
storeEvent(event, { force: wanted }),
|
||||||
processDeletions(event),
|
processDeletions(event),
|
||||||
trackRelays(event),
|
trackRelays(event),
|
||||||
trackHashtags(event),
|
trackHashtags(event),
|
||||||
fetchRelatedEvents(event, data, signal),
|
fetchRelatedEvents(event, signal),
|
||||||
processMedia(event, data),
|
processMedia(event),
|
||||||
payZap(event, data, signal),
|
payZap(event, signal),
|
||||||
streamOut(event, data),
|
streamOut(event),
|
||||||
broadcast(event, data),
|
broadcast(event),
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -53,10 +52,10 @@ async function encounterEvent(event: Event): Promise<boolean> {
|
||||||
return preexisting;
|
return preexisting;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Preload data that will be useful to several tasks. */
|
/** Hydrate the event with the user, if applicable. */
|
||||||
async function getEventData({ pubkey }: Event): Promise<EventData> {
|
async function hydrateEvent(event: DittoEvent): Promise<void> {
|
||||||
const user = await findUser({ pubkey });
|
const [user] = await eventsDB.filter([{ kinds: [30361], authors: [Conf.pubkey], limit: 1 }]);
|
||||||
return { user };
|
event.user = user;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Check if the pubkey is the `DITTO_NSEC` pubkey. */
|
/** Check if the pubkey is the `DITTO_NSEC` pubkey. */
|
||||||
|
@ -67,11 +66,11 @@ interface StoreEventOpts {
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Maybe store the event, if eligible. */
|
/** Maybe store the event, if eligible. */
|
||||||
async function storeEvent(event: Event, data: EventData, opts: StoreEventOpts = {}): Promise<void> {
|
async function storeEvent(event: DittoEvent, opts: StoreEventOpts = {}): Promise<void> {
|
||||||
if (isEphemeralKind(event.kind)) return;
|
if (isEphemeralKind(event.kind)) return;
|
||||||
const { force = false } = opts;
|
const { force = false } = opts;
|
||||||
|
|
||||||
if (force || data.user || isAdminEvent(event) || await isLocallyFollowed(event.pubkey)) {
|
if (force || event.user || isAdminEvent(event) || await isLocallyFollowed(event.pubkey)) {
|
||||||
const isDeleted = await eventsDB.count(
|
const isDeleted = await eventsDB.count(
|
||||||
[{ kinds: [5], authors: [Conf.pubkey, event.pubkey], '#e': [event.id], limit: 1 }],
|
[{ kinds: [5], authors: [Conf.pubkey, event.pubkey], '#e': [event.id], limit: 1 }],
|
||||||
) > 0;
|
) > 0;
|
||||||
|
@ -80,7 +79,7 @@ async function storeEvent(event: Event, data: EventData, opts: StoreEventOpts =
|
||||||
return Promise.reject(new RelayError('blocked', 'event was deleted'));
|
return Promise.reject(new RelayError('blocked', 'event was deleted'));
|
||||||
} else {
|
} else {
|
||||||
await Promise.all([
|
await Promise.all([
|
||||||
eventsDB.add(event, { data }).catch(debug),
|
eventsDB.add(event).catch(debug),
|
||||||
updateStats(event).catch(debug),
|
updateStats(event).catch(debug),
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
@ -144,8 +143,8 @@ function trackRelays(event: Event) {
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Queue related events to fetch. */
|
/** Queue related events to fetch. */
|
||||||
function fetchRelatedEvents(event: Event, data: EventData, signal: AbortSignal) {
|
function fetchRelatedEvents(event: DittoEvent, signal: AbortSignal) {
|
||||||
if (!data.user) {
|
if (!event.user) {
|
||||||
reqmeister.req({ kinds: [0], authors: [event.pubkey] }, { signal }).catch(() => {});
|
reqmeister.req({ kinds: [0], authors: [event.pubkey] }, { signal }).catch(() => {});
|
||||||
}
|
}
|
||||||
for (const [name, id, relay] of event.tags) {
|
for (const [name, id, relay] of event.tags) {
|
||||||
|
@ -156,7 +155,7 @@ function fetchRelatedEvents(event: Event, data: EventData, signal: AbortSignal)
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Delete unattached media entries that are attached to the event. */
|
/** Delete unattached media entries that are attached to the event. */
|
||||||
function processMedia({ tags, pubkey }: Event, { user }: EventData) {
|
function processMedia({ tags, pubkey, user }: DittoEvent) {
|
||||||
if (user) {
|
if (user) {
|
||||||
const urls = getTagSet(tags, 'media');
|
const urls = getTagSet(tags, 'media');
|
||||||
return deleteAttachedMedia(pubkey, [...urls]);
|
return deleteAttachedMedia(pubkey, [...urls]);
|
||||||
|
@ -164,8 +163,8 @@ function processMedia({ tags, pubkey }: Event, { user }: EventData) {
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Emit Nostr Wallet Connect event from zaps so users may pay. */
|
/** Emit Nostr Wallet Connect event from zaps so users may pay. */
|
||||||
async function payZap(event: Event, data: EventData, signal: AbortSignal) {
|
async function payZap(event: DittoEvent, signal: AbortSignal) {
|
||||||
if (event.kind !== 9734 || !data.user) return;
|
if (event.kind !== 9734 || !event.user) return;
|
||||||
|
|
||||||
const lnurl = event.tags.find(([name]) => name === 'lnurl')?.[1];
|
const lnurl = event.tags.find(([name]) => name === 'lnurl')?.[1];
|
||||||
const amount = Number(event.tags.find(([name]) => name === 'amount')?.[1]);
|
const amount = Number(event.tags.find(([name]) => name === 'amount')?.[1]);
|
||||||
|
@ -212,10 +211,10 @@ async function payZap(event: Event, data: EventData, signal: AbortSignal) {
|
||||||
const isFresh = (event: Event): boolean => eventAge(event) < Time.seconds(10);
|
const isFresh = (event: Event): boolean => eventAge(event) < Time.seconds(10);
|
||||||
|
|
||||||
/** Distribute the event through active subscriptions. */
|
/** Distribute the event through active subscriptions. */
|
||||||
function streamOut(event: Event, data: EventData) {
|
function streamOut(event: Event) {
|
||||||
if (!isFresh(event)) return;
|
if (!isFresh(event)) return;
|
||||||
|
|
||||||
for (const sub of Sub.matches(event, data)) {
|
for (const sub of Sub.matches(event)) {
|
||||||
sub.stream(event);
|
sub.stream(event);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -224,8 +223,8 @@ function streamOut(event: Event, data: EventData) {
|
||||||
* Publish the event to other relays.
|
* Publish the event to other relays.
|
||||||
* This should only be done in certain circumstances, like mentioning a user or publishing deletions.
|
* This should only be done in certain circumstances, like mentioning a user or publishing deletions.
|
||||||
*/
|
*/
|
||||||
function broadcast(event: Event, data: EventData) {
|
function broadcast(event: DittoEvent) {
|
||||||
if (!data.user || !isFresh(event)) return;
|
if (!event.user || !isFresh(event)) return;
|
||||||
|
|
||||||
if (event.kind === 5) {
|
if (event.kind === 5) {
|
||||||
client.add(event);
|
client.add(event);
|
||||||
|
|
|
@ -6,12 +6,11 @@ import { isDittoInternalKind, isParameterizedReplaceableKind, isReplaceableKind
|
||||||
import { jsonMetaContentSchema } from '@/schemas/nostr.ts';
|
import { jsonMetaContentSchema } from '@/schemas/nostr.ts';
|
||||||
import { isNostrId, isURL } from '@/utils.ts';
|
import { isNostrId, isURL } from '@/utils.ts';
|
||||||
|
|
||||||
import { type DittoEvent, EventStore, type GetEventsOpts, type StoreEventOpts } from './types.ts';
|
import { type DittoEvent, EventStore, type GetEventsOpts } from './types.ts';
|
||||||
|
|
||||||
/** Function to decide whether or not to index a tag. */
|
/** Function to decide whether or not to index a tag. */
|
||||||
type TagCondition = ({ event, count, value }: {
|
type TagCondition = ({ event, count, value }: {
|
||||||
event: Event;
|
event: DittoEvent;
|
||||||
opts: StoreEventOpts;
|
|
||||||
count: number;
|
count: number;
|
||||||
value: string;
|
value: string;
|
||||||
}) => boolean;
|
}) => boolean;
|
||||||
|
@ -19,8 +18,8 @@ type TagCondition = ({ event, count, value }: {
|
||||||
/** Conditions for when to index certain tags. */
|
/** Conditions for when to index certain tags. */
|
||||||
const tagConditions: Record<string, TagCondition> = {
|
const tagConditions: Record<string, TagCondition> = {
|
||||||
'd': ({ event, count }) => count === 0 && isParameterizedReplaceableKind(event.kind),
|
'd': ({ event, count }) => count === 0 && isParameterizedReplaceableKind(event.kind),
|
||||||
'e': ({ event, count, value, opts }) => ((opts.data?.user && event.kind === 10003) || count < 15) && isNostrId(value),
|
'e': ({ event, count, value }) => ((event.user && event.kind === 10003) || count < 15) && isNostrId(value),
|
||||||
'media': ({ count, value, opts }) => (opts.data?.user || count < 4) && isURL(value),
|
'media': ({ event, count, value }) => (event.user || count < 4) && isURL(value),
|
||||||
'P': ({ event, count, value }) => event.kind === 9735 && count === 0 && isNostrId(value),
|
'P': ({ event, count, value }) => event.kind === 9735 && count === 0 && isNostrId(value),
|
||||||
'p': ({ event, count, value }) => (count < 15 || event.kind === 3) && isNostrId(value),
|
'p': ({ event, count, value }) => (count < 15 || event.kind === 3) && isNostrId(value),
|
||||||
'proxy': ({ count, value }) => count === 0 && isURL(value),
|
'proxy': ({ count, value }) => count === 0 && isURL(value),
|
||||||
|
@ -66,7 +65,7 @@ class EventsDB implements EventStore {
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Insert an event (and its tags) into the database. */
|
/** Insert an event (and its tags) into the database. */
|
||||||
async add(event: Event, opts: StoreEventOpts = {}): Promise<void> {
|
async add(event: DittoEvent): Promise<void> {
|
||||||
this.#debug('EVENT', JSON.stringify(event));
|
this.#debug('EVENT', JSON.stringify(event));
|
||||||
|
|
||||||
if (isDittoInternalKind(event.kind) && event.pubkey !== Conf.pubkey) {
|
if (isDittoInternalKind(event.kind) && event.pubkey !== Conf.pubkey) {
|
||||||
|
@ -92,7 +91,7 @@ class EventsDB implements EventStore {
|
||||||
|
|
||||||
/** Index event tags depending on the conditions defined above. */
|
/** Index event tags depending on the conditions defined above. */
|
||||||
async function indexTags() {
|
async function indexTags() {
|
||||||
const tags = filterIndexableTags(event, opts);
|
const tags = filterIndexableTags(event);
|
||||||
const rows = tags.map(([tag, value]) => ({ event_id: event.id, tag, value }));
|
const rows = tags.map(([tag, value]) => ({ event_id: event.id, tag, value }));
|
||||||
|
|
||||||
if (!tags.length) return;
|
if (!tags.length) return;
|
||||||
|
@ -361,7 +360,7 @@ class EventsDB implements EventStore {
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Return only the tags that should be indexed. */
|
/** Return only the tags that should be indexed. */
|
||||||
function filterIndexableTags(event: Event, opts: StoreEventOpts): string[][] {
|
function filterIndexableTags(event: DittoEvent): string[][] {
|
||||||
const tagCounts: Record<string, number> = {};
|
const tagCounts: Record<string, number> = {};
|
||||||
|
|
||||||
function getCount(name: string) {
|
function getCount(name: string) {
|
||||||
|
@ -375,7 +374,6 @@ function filterIndexableTags(event: Event, opts: StoreEventOpts): string[][] {
|
||||||
function checkCondition(name: string, value: string, condition: TagCondition) {
|
function checkCondition(name: string, value: string, condition: TagCondition) {
|
||||||
return condition({
|
return condition({
|
||||||
event,
|
event,
|
||||||
opts,
|
|
||||||
count: getCount(name),
|
count: getCount(name),
|
||||||
value,
|
value,
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
import { type DittoDB } from '@/db.ts';
|
import { type DittoDB } from '@/db.ts';
|
||||||
import { type Event } from '@/deps.ts';
|
import { type Event } from '@/deps.ts';
|
||||||
import { type DittoFilter } from '@/filter.ts';
|
import { type DittoFilter } from '@/filter.ts';
|
||||||
import { type EventData } from '@/types.ts';
|
|
||||||
|
|
||||||
/** Additional options to apply to the whole subscription. */
|
/** Additional options to apply to the whole subscription. */
|
||||||
interface GetEventsOpts {
|
interface GetEventsOpts {
|
||||||
|
@ -15,8 +14,6 @@ interface GetEventsOpts {
|
||||||
|
|
||||||
/** Options when storing an event. */
|
/** Options when storing an event. */
|
||||||
interface StoreEventOpts {
|
interface StoreEventOpts {
|
||||||
/** Event data to store. */
|
|
||||||
data?: EventData;
|
|
||||||
/** Relays to use, if applicable. */
|
/** Relays to use, if applicable. */
|
||||||
relays?: WebSocket['url'][];
|
relays?: WebSocket['url'][];
|
||||||
}
|
}
|
||||||
|
|
11
src/subs.ts
11
src/subs.ts
|
@ -1,9 +1,8 @@
|
||||||
import { Debug, type Event } from '@/deps.ts';
|
import { Debug } from '@/deps.ts';
|
||||||
|
import { type DittoFilter } from '@/filter.ts';
|
||||||
|
import { type DittoEvent } from '@/storages/types.ts';
|
||||||
import { Subscription } from '@/subscription.ts';
|
import { Subscription } from '@/subscription.ts';
|
||||||
|
|
||||||
import type { DittoFilter } from '@/filter.ts';
|
|
||||||
import type { EventData } from '@/types.ts';
|
|
||||||
|
|
||||||
const debug = Debug('ditto:subs');
|
const debug = Debug('ditto:subs');
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -69,10 +68,10 @@ class SubscriptionStore {
|
||||||
* }
|
* }
|
||||||
* ```
|
* ```
|
||||||
*/
|
*/
|
||||||
*matches(event: Event, data: EventData): Iterable<Subscription> {
|
*matches(event: DittoEvent): Iterable<Subscription> {
|
||||||
for (const subs of this.#store.values()) {
|
for (const subs of this.#store.values()) {
|
||||||
for (const sub of subs.values()) {
|
for (const sub of subs.values()) {
|
||||||
if (sub.matches(event, data)) {
|
if (sub.matches(event)) {
|
||||||
yield sub;
|
yield sub;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
import { type Event, Machina } from '@/deps.ts';
|
import { type Event, Machina } from '@/deps.ts';
|
||||||
import { type DittoFilter, matchDittoFilters } from '@/filter.ts';
|
import { type DittoFilter, matchDittoFilters } from '@/filter.ts';
|
||||||
|
import { type DittoEvent } from '@/storages/types.ts';
|
||||||
import type { EventData } from '@/types.ts';
|
|
||||||
|
|
||||||
class Subscription<K extends number = number> implements AsyncIterable<Event<K>> {
|
class Subscription<K extends number = number> implements AsyncIterable<Event<K>> {
|
||||||
filters: DittoFilter<K>[];
|
filters: DittoFilter<K>[];
|
||||||
|
@ -16,8 +15,8 @@ class Subscription<K extends number = number> implements AsyncIterable<Event<K>>
|
||||||
this.#machina.push(event);
|
this.#machina.push(event);
|
||||||
}
|
}
|
||||||
|
|
||||||
matches(event: Event, data: EventData): boolean {
|
matches(event: DittoEvent): boolean {
|
||||||
return matchDittoFilters(this.filters, event, data);
|
return matchDittoFilters(this.filters, event);
|
||||||
}
|
}
|
||||||
|
|
||||||
close() {
|
close() {
|
||||||
|
|
|
@ -1,6 +0,0 @@
|
||||||
import { User } from '@/db/users.ts';
|
|
||||||
interface EventData {
|
|
||||||
user: User | undefined;
|
|
||||||
}
|
|
||||||
|
|
||||||
export type { EventData };
|
|
Loading…
Reference in New Issue