Merge branch 'nstore' into 'main'
Implement NStore interface from NLib See merge request soapbox-pub/ditto!106
This commit is contained in:
commit
152c52427a
|
@ -39,6 +39,6 @@ async function usersToEvents() {
|
||||||
created_at: Math.floor(new Date(row.inserted_at).getTime() / 1000),
|
created_at: Math.floor(new Date(row.inserted_at).getTime() / 1000),
|
||||||
});
|
});
|
||||||
|
|
||||||
await eventsDB.add(event);
|
await eventsDB.event(event);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -134,7 +134,7 @@ const accountStatusesController: AppController = async (c) => {
|
||||||
const { pinned, limit, exclude_replies, tagged } = accountStatusesQuerySchema.parse(c.req.query());
|
const { pinned, limit, exclude_replies, tagged } = accountStatusesQuerySchema.parse(c.req.query());
|
||||||
|
|
||||||
if (pinned) {
|
if (pinned) {
|
||||||
const [pinEvent] = await eventsDB.filter([{ kinds: [10001], authors: [pubkey], limit: 1 }]);
|
const [pinEvent] = await eventsDB.query([{ kinds: [10001], authors: [pubkey], limit: 1 }]);
|
||||||
if (pinEvent) {
|
if (pinEvent) {
|
||||||
const pinnedEventIds = getTagSet(pinEvent.tags, 'e');
|
const pinnedEventIds = getTagSet(pinEvent.tags, 'e');
|
||||||
return renderStatuses(c, [...pinnedEventIds].reverse());
|
return renderStatuses(c, [...pinnedEventIds].reverse());
|
||||||
|
@ -156,7 +156,7 @@ const accountStatusesController: AppController = async (c) => {
|
||||||
filter['#t'] = [tagged];
|
filter['#t'] = [tagged];
|
||||||
}
|
}
|
||||||
|
|
||||||
let events = await eventsDB.filter([filter]);
|
let events = await eventsDB.query([filter]);
|
||||||
|
|
||||||
if (exclude_replies) {
|
if (exclude_replies) {
|
||||||
events = events.filter((event) => !findReplyTag(event.tags));
|
events = events.filter((event) => !findReplyTag(event.tags));
|
||||||
|
@ -293,7 +293,7 @@ const favouritesController: AppController = async (c) => {
|
||||||
const pubkey = c.get('pubkey')!;
|
const pubkey = c.get('pubkey')!;
|
||||||
const params = paginationSchema.parse(c.req.query());
|
const params = paginationSchema.parse(c.req.query());
|
||||||
|
|
||||||
const events7 = await eventsDB.filter(
|
const events7 = await eventsDB.query(
|
||||||
[{ kinds: [7], authors: [pubkey], ...params }],
|
[{ kinds: [7], authors: [pubkey], ...params }],
|
||||||
{ signal: AbortSignal.timeout(1000) },
|
{ signal: AbortSignal.timeout(1000) },
|
||||||
);
|
);
|
||||||
|
@ -302,7 +302,7 @@ const favouritesController: AppController = async (c) => {
|
||||||
.map((event) => event.tags.find((tag) => tag[0] === 'e')?.[1])
|
.map((event) => event.tags.find((tag) => tag[0] === 'e')?.[1])
|
||||||
.filter((id): id is string => !!id);
|
.filter((id): id is string => !!id);
|
||||||
|
|
||||||
const events1 = await eventsDB.filter(
|
const events1 = await eventsDB.query(
|
||||||
[{ kinds: [1], ids, relations: ['author', 'event_stats', 'author_stats'] }],
|
[{ kinds: [1], ids, relations: ['author', 'event_stats', 'author_stats'] }],
|
||||||
{
|
{
|
||||||
signal: AbortSignal.timeout(1000),
|
signal: AbortSignal.timeout(1000),
|
||||||
|
|
|
@ -39,9 +39,9 @@ const adminAccountsController: AppController = async (c) => {
|
||||||
|
|
||||||
const { since, until, limit } = paginationSchema.parse(c.req.query());
|
const { since, until, limit } = paginationSchema.parse(c.req.query());
|
||||||
|
|
||||||
const events = await eventsDB.filter([{ kinds: [30361], authors: [Conf.pubkey], since, until, limit }]);
|
const events = await eventsDB.query([{ kinds: [30361], authors: [Conf.pubkey], since, until, limit }]);
|
||||||
const pubkeys = events.map((event) => event.tags.find(([name]) => name === 'd')?.[1]!);
|
const pubkeys = events.map((event) => event.tags.find(([name]) => name === 'd')?.[1]!);
|
||||||
const authors = await eventsDB.filter([{ kinds: [0], authors: pubkeys }]);
|
const authors = await eventsDB.query([{ kinds: [0], authors: pubkeys }]);
|
||||||
|
|
||||||
for (const event of events) {
|
for (const event of events) {
|
||||||
const d = event.tags.find(([name]) => name === 'd')?.[1];
|
const d = event.tags.find(([name]) => name === 'd')?.[1];
|
||||||
|
|
|
@ -7,7 +7,7 @@ import { renderAccounts } from '@/views.ts';
|
||||||
const blocksController: AppController = async (c) => {
|
const blocksController: AppController = async (c) => {
|
||||||
const pubkey = c.get('pubkey')!;
|
const pubkey = c.get('pubkey')!;
|
||||||
|
|
||||||
const [event10000] = await eventsDB.filter([
|
const [event10000] = await eventsDB.query([
|
||||||
{ kinds: [10000], authors: [pubkey], limit: 1 },
|
{ kinds: [10000], authors: [pubkey], limit: 1 },
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
|
|
@ -7,7 +7,7 @@ import { renderStatuses } from '@/views.ts';
|
||||||
const bookmarksController: AppController = async (c) => {
|
const bookmarksController: AppController = async (c) => {
|
||||||
const pubkey = c.get('pubkey')!;
|
const pubkey = c.get('pubkey')!;
|
||||||
|
|
||||||
const [event10003] = await eventsDB.filter([
|
const [event10003] = await eventsDB.query([
|
||||||
{ kinds: [10003], authors: [pubkey], limit: 1 },
|
{ kinds: [10003], authors: [pubkey], limit: 1 },
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
|
|
@ -6,7 +6,7 @@ import { eventsDB } from '@/storages.ts';
|
||||||
const instanceController: AppController = async (c) => {
|
const instanceController: AppController = async (c) => {
|
||||||
const { host, protocol } = Conf.url;
|
const { host, protocol } = Conf.url;
|
||||||
|
|
||||||
const [event] = await eventsDB.filter([{ kinds: [0], authors: [Conf.pubkey], limit: 1 }]);
|
const [event] = await eventsDB.query([{ kinds: [0], authors: [Conf.pubkey], limit: 1 }]);
|
||||||
const meta = jsonServerMetaSchema.parse(event?.content);
|
const meta = jsonServerMetaSchema.parse(event?.content);
|
||||||
|
|
||||||
/** Protocol to use for WebSocket URLs, depending on the protocol of the `LOCAL_DOMAIN`. */
|
/** Protocol to use for WebSocket URLs, depending on the protocol of the `LOCAL_DOMAIN`. */
|
||||||
|
|
|
@ -7,7 +7,7 @@ const notificationsController: AppController = async (c) => {
|
||||||
const pubkey = c.get('pubkey')!;
|
const pubkey = c.get('pubkey')!;
|
||||||
const { since, until } = paginationSchema.parse(c.req.query());
|
const { since, until } = paginationSchema.parse(c.req.query());
|
||||||
|
|
||||||
const events = await eventsDB.filter(
|
const events = await eventsDB.query(
|
||||||
[{ kinds: [1], '#p': [pubkey], since, until }],
|
[{ kinds: [1], '#p': [pubkey], since, until }],
|
||||||
{ signal: AbortSignal.timeout(3000) },
|
{ signal: AbortSignal.timeout(3000) },
|
||||||
);
|
);
|
||||||
|
|
|
@ -8,7 +8,7 @@ import { createAdminEvent } from '@/utils/api.ts';
|
||||||
import { jsonSchema } from '@/schema.ts';
|
import { jsonSchema } from '@/schema.ts';
|
||||||
|
|
||||||
const frontendConfigController: AppController = async (c) => {
|
const frontendConfigController: AppController = async (c) => {
|
||||||
const [event] = await eventsDB.filter([{
|
const [event] = await eventsDB.query([{
|
||||||
kinds: [30078],
|
kinds: [30078],
|
||||||
authors: [Conf.pubkey],
|
authors: [Conf.pubkey],
|
||||||
'#d': ['pub.ditto.pleroma.config'],
|
'#d': ['pub.ditto.pleroma.config'],
|
||||||
|
@ -36,7 +36,7 @@ const frontendConfigController: AppController = async (c) => {
|
||||||
const configController: AppController = async (c) => {
|
const configController: AppController = async (c) => {
|
||||||
const { pubkey } = Conf;
|
const { pubkey } = Conf;
|
||||||
|
|
||||||
const [event] = await eventsDB.filter([{
|
const [event] = await eventsDB.query([{
|
||||||
kinds: [30078],
|
kinds: [30078],
|
||||||
authors: [pubkey],
|
authors: [pubkey],
|
||||||
'#d': ['pub.ditto.pleroma.config'],
|
'#d': ['pub.ditto.pleroma.config'],
|
||||||
|
@ -54,7 +54,7 @@ const configController: AppController = async (c) => {
|
||||||
const updateConfigController: AppController = async (c) => {
|
const updateConfigController: AppController = async (c) => {
|
||||||
const { pubkey } = Conf;
|
const { pubkey } = Conf;
|
||||||
|
|
||||||
const [event] = await eventsDB.filter([{
|
const [event] = await eventsDB.query([{
|
||||||
kinds: [30078],
|
kinds: [30078],
|
||||||
authors: [pubkey],
|
authors: [pubkey],
|
||||||
'#d': ['pub.ditto.pleroma.config'],
|
'#d': ['pub.ditto.pleroma.config'],
|
||||||
|
|
|
@ -78,7 +78,7 @@ function searchEvents({ q, type, limit, account_id }: SearchQuery, signal: Abort
|
||||||
filter.authors = [account_id];
|
filter.authors = [account_id];
|
||||||
}
|
}
|
||||||
|
|
||||||
return searchStore.filter([filter], { signal });
|
return searchStore.query([filter], { signal });
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Get event kinds to search from `type` query param. */
|
/** Get event kinds to search from `type` query param. */
|
||||||
|
@ -96,7 +96,7 @@ function typeToKinds(type: SearchQuery['type']): number[] {
|
||||||
/** Resolve a searched value into an event, if applicable. */
|
/** Resolve a searched value into an event, if applicable. */
|
||||||
async function lookupEvent(query: SearchQuery, signal: AbortSignal): Promise<NostrEvent | undefined> {
|
async function lookupEvent(query: SearchQuery, signal: AbortSignal): Promise<NostrEvent | undefined> {
|
||||||
const filters = await getLookupFilters(query, signal);
|
const filters = await getLookupFilters(query, signal);
|
||||||
const [event] = await searchStore.filter(filters, { limit: 1, signal });
|
const [event] = await searchStore.query(filters, { limit: 1, signal });
|
||||||
return event;
|
return event;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -32,7 +32,7 @@ const hashtagTimelineController: AppController = (c) => {
|
||||||
|
|
||||||
/** Render statuses for timelines. */
|
/** Render statuses for timelines. */
|
||||||
async function renderStatuses(c: AppContext, filters: DittoFilter[], signal = AbortSignal.timeout(1000)) {
|
async function renderStatuses(c: AppContext, filters: DittoFilter[], signal = AbortSignal.timeout(1000)) {
|
||||||
const events = await eventsDB.filter(
|
const events = await eventsDB.query(
|
||||||
filters.map((filter) => ({ ...filter, relations: ['author', 'event_stats', 'author_stats'] })),
|
filters.map((filter) => ({ ...filter, relations: ['author', 'event_stats', 'author_stats'] })),
|
||||||
{ signal },
|
{ signal },
|
||||||
);
|
);
|
||||||
|
|
|
@ -4,7 +4,7 @@ import { jsonServerMetaSchema } from '@/schemas/nostr.ts';
|
||||||
import { eventsDB } from '@/storages.ts';
|
import { eventsDB } from '@/storages.ts';
|
||||||
|
|
||||||
const relayInfoController: AppController = async (c) => {
|
const relayInfoController: AppController = async (c) => {
|
||||||
const [event] = await eventsDB.filter([{ kinds: [0], authors: [Conf.pubkey], limit: 1 }]);
|
const [event] = await eventsDB.query([{ kinds: [0], authors: [Conf.pubkey], limit: 1 }]);
|
||||||
const meta = jsonServerMetaSchema.parse(event?.content);
|
const meta = jsonServerMetaSchema.parse(event?.content);
|
||||||
|
|
||||||
return c.json({
|
return c.json({
|
||||||
|
|
|
@ -63,7 +63,7 @@ function connectStream(socket: WebSocket) {
|
||||||
async function handleReq([_, subId, ...rest]: ClientREQ): Promise<void> {
|
async function handleReq([_, subId, ...rest]: ClientREQ): Promise<void> {
|
||||||
const filters = prepareFilters(rest);
|
const filters = prepareFilters(rest);
|
||||||
|
|
||||||
for (const event of await eventsDB.filter(filters, { limit: FILTER_LIMIT })) {
|
for (const event of await eventsDB.query(filters, { limit: FILTER_LIMIT })) {
|
||||||
send(['EVENT', subId, event]);
|
send(['EVENT', subId, event]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -65,7 +65,7 @@ async function findUser(user: Partial<User>): Promise<User | undefined> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const [event] = await eventsDB.filter([filter]);
|
const [event] = await eventsDB.query([filter]);
|
||||||
|
|
||||||
if (event) {
|
if (event) {
|
||||||
return {
|
return {
|
||||||
|
|
|
@ -91,6 +91,8 @@ export {
|
||||||
NIP05,
|
NIP05,
|
||||||
type NostrEvent,
|
type NostrEvent,
|
||||||
type NostrFilter,
|
type NostrFilter,
|
||||||
} from 'https://gitlab.com/soapbox-pub/nlib/-/raw/5d711597f3b2a163817cc1fb0f1f3ce8cede7cf7/mod.ts';
|
type NStore,
|
||||||
|
type NStoreOpts,
|
||||||
|
} from 'https://gitlab.com/soapbox-pub/nlib/-/raw/057ecc6e2ce813db6e2279288fbfd08c5b53cc0c/mod.ts';
|
||||||
|
|
||||||
export type * as TypeFest from 'npm:type-fest@^4.3.0';
|
export type * as TypeFest from 'npm:type-fest@^4.3.0';
|
||||||
|
|
|
@ -27,34 +27,34 @@ 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, signal)) return;
|
||||||
debug(`NostrEvent<${event.kind}> ${event.id}`);
|
debug(`NostrEvent<${event.kind}> ${event.id}`);
|
||||||
await hydrateEvent(event);
|
await hydrateEvent(event);
|
||||||
|
|
||||||
await Promise.all([
|
await Promise.all([
|
||||||
storeEvent(event, { force: wanted }),
|
storeEvent(event, { force: wanted, signal }),
|
||||||
processDeletions(event),
|
processDeletions(event, signal),
|
||||||
trackRelays(event),
|
trackRelays(event),
|
||||||
trackHashtags(event),
|
trackHashtags(event),
|
||||||
fetchRelatedEvents(event, signal),
|
fetchRelatedEvents(event, signal),
|
||||||
processMedia(event),
|
processMedia(event),
|
||||||
payZap(event, signal),
|
payZap(event, signal),
|
||||||
streamOut(event),
|
streamOut(event),
|
||||||
broadcast(event),
|
broadcast(event, signal),
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Encounter the event, and return whether it has already been encountered. */
|
/** Encounter the event, and return whether it has already been encountered. */
|
||||||
async function encounterEvent(event: NostrEvent): Promise<boolean> {
|
async function encounterEvent(event: NostrEvent, signal: AbortSignal): Promise<boolean> {
|
||||||
const preexisting = (await memorelay.count([{ ids: [event.id] }])) > 0;
|
const preexisting = (await memorelay.count([{ ids: [event.id] }])) > 0;
|
||||||
memorelay.add(event);
|
memorelay.event(event, { signal });
|
||||||
reqmeister.add(event);
|
reqmeister.event(event, { signal });
|
||||||
return preexisting;
|
return preexisting;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Hydrate the event with the user, if applicable. */
|
/** Hydrate the event with the user, if applicable. */
|
||||||
async function hydrateEvent(event: DittoEvent): Promise<void> {
|
async function hydrateEvent(event: DittoEvent): Promise<void> {
|
||||||
const [user] = await eventsDB.filter([{ kinds: [30361], authors: [Conf.pubkey], limit: 1 }]);
|
const [user] = await eventsDB.query([{ kinds: [30361], authors: [Conf.pubkey], limit: 1 }]);
|
||||||
event.user = user;
|
event.user = user;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -62,24 +62,26 @@ async function hydrateEvent(event: DittoEvent): Promise<void> {
|
||||||
const isAdminEvent = ({ pubkey }: NostrEvent): boolean => pubkey === Conf.pubkey;
|
const isAdminEvent = ({ pubkey }: NostrEvent): boolean => pubkey === Conf.pubkey;
|
||||||
|
|
||||||
interface StoreEventOpts {
|
interface StoreEventOpts {
|
||||||
force?: boolean;
|
force: boolean;
|
||||||
|
signal: AbortSignal;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Maybe store the event, if eligible. */
|
/** Maybe store the event, if eligible. */
|
||||||
async function storeEvent(event: DittoEvent, 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, signal } = opts;
|
||||||
|
|
||||||
if (force || event.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 }],
|
||||||
|
opts,
|
||||||
) > 0;
|
) > 0;
|
||||||
|
|
||||||
if (isDeleted) {
|
if (isDeleted) {
|
||||||
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).catch(debug),
|
eventsDB.event(event, { signal }).catch(debug),
|
||||||
updateStats(event).catch(debug),
|
updateStats(event).catch(debug),
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
@ -89,20 +91,20 @@ async function storeEvent(event: DittoEvent, opts: StoreEventOpts = {}): Promise
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Query to-be-deleted events, ensure their pubkey matches, then delete them from the database. */
|
/** Query to-be-deleted events, ensure their pubkey matches, then delete them from the database. */
|
||||||
async function processDeletions(event: NostrEvent): Promise<void> {
|
async function processDeletions(event: NostrEvent, signal: AbortSignal): Promise<void> {
|
||||||
if (event.kind === 5) {
|
if (event.kind === 5) {
|
||||||
const ids = getTagSet(event.tags, 'e');
|
const ids = getTagSet(event.tags, 'e');
|
||||||
|
|
||||||
if (event.pubkey === Conf.pubkey) {
|
if (event.pubkey === Conf.pubkey) {
|
||||||
await eventsDB.deleteFilters([{ ids: [...ids] }]);
|
await eventsDB.remove([{ ids: [...ids] }], { signal });
|
||||||
} else {
|
} else {
|
||||||
const events = await eventsDB.filter([{
|
const events = await eventsDB.query(
|
||||||
ids: [...ids],
|
[{ ids: [...ids], authors: [event.pubkey] }],
|
||||||
authors: [event.pubkey],
|
{ signal },
|
||||||
}]);
|
);
|
||||||
|
|
||||||
const deleteIds = events.map(({ id }) => id);
|
const deleteIds = events.map(({ id }) => id);
|
||||||
await eventsDB.deleteFilters([{ ids: deleteIds }]);
|
await eventsDB.remove([{ ids: deleteIds }], { signal });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -148,7 +150,7 @@ function fetchRelatedEvents(event: DittoEvent, signal: AbortSignal) {
|
||||||
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) {
|
||||||
if (name === 'e' && !memorelay.count([{ ids: [id] }])) {
|
if (name === 'e' && !memorelay.count([{ ids: [id] }], { signal })) {
|
||||||
reqmeister.req({ ids: [id] }, { relays: [relay] }).catch(() => {});
|
reqmeister.req({ ids: [id] }, { relays: [relay] }).catch(() => {});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -223,11 +225,11 @@ function streamOut(event: NostrEvent) {
|
||||||
* 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: DittoEvent) {
|
function broadcast(event: DittoEvent, signal: AbortSignal) {
|
||||||
if (!event.user || !isFresh(event)) return;
|
if (!event.user || !isFresh(event)) return;
|
||||||
|
|
||||||
if (event.kind === 5) {
|
if (event.kind === 5) {
|
||||||
client.add(event);
|
client.event(event, { signal });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -25,7 +25,7 @@ const getEvent = async (
|
||||||
const { kind, relations, signal = AbortSignal.timeout(1000) } = opts;
|
const { kind, relations, signal = AbortSignal.timeout(1000) } = opts;
|
||||||
const microfilter: IdMicrofilter = { ids: [id] };
|
const microfilter: IdMicrofilter = { ids: [id] };
|
||||||
|
|
||||||
const [memoryEvent] = await memorelay.filter([microfilter], opts) as DittoEvent[];
|
const [memoryEvent] = await memorelay.query([microfilter], opts) as DittoEvent[];
|
||||||
|
|
||||||
if (memoryEvent && !relations) {
|
if (memoryEvent && !relations) {
|
||||||
debug(`getEvent: ${id.slice(0, 8)} found in memory`);
|
debug(`getEvent: ${id.slice(0, 8)} found in memory`);
|
||||||
|
@ -37,13 +37,13 @@ const getEvent = async (
|
||||||
filter.kinds = [kind];
|
filter.kinds = [kind];
|
||||||
}
|
}
|
||||||
|
|
||||||
const dbEvent = await eventsDB.filter([filter], { limit: 1, signal })
|
const dbEvent = await eventsDB.query([filter], { limit: 1, signal })
|
||||||
.then(([event]) => event);
|
.then(([event]) => event);
|
||||||
|
|
||||||
// TODO: make this DRY-er.
|
// TODO: make this DRY-er.
|
||||||
|
|
||||||
if (dbEvent && !dbEvent.author) {
|
if (dbEvent && !dbEvent.author) {
|
||||||
const [author] = await memorelay.filter([{ kinds: [0], authors: [dbEvent.pubkey] }], opts);
|
const [author] = await memorelay.query([{ kinds: [0], authors: [dbEvent.pubkey] }], opts);
|
||||||
dbEvent.author = author;
|
dbEvent.author = author;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -53,7 +53,7 @@ const getEvent = async (
|
||||||
}
|
}
|
||||||
|
|
||||||
if (memoryEvent && !memoryEvent.author) {
|
if (memoryEvent && !memoryEvent.author) {
|
||||||
const [author] = await memorelay.filter([{ kinds: [0], authors: [memoryEvent.pubkey] }], opts);
|
const [author] = await memorelay.query([{ kinds: [0], authors: [memoryEvent.pubkey] }], opts);
|
||||||
memoryEvent.author = author;
|
memoryEvent.author = author;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -77,13 +77,13 @@ const getAuthor = async (pubkey: string, opts: GetEventOpts = {}): Promise<Nostr
|
||||||
const { relations, signal = AbortSignal.timeout(1000) } = opts;
|
const { relations, signal = AbortSignal.timeout(1000) } = opts;
|
||||||
const microfilter: AuthorMicrofilter = { kinds: [0], authors: [pubkey] };
|
const microfilter: AuthorMicrofilter = { kinds: [0], authors: [pubkey] };
|
||||||
|
|
||||||
const [memoryEvent] = await memorelay.filter([microfilter], opts);
|
const [memoryEvent] = await memorelay.query([microfilter], opts);
|
||||||
|
|
||||||
if (memoryEvent && !relations) {
|
if (memoryEvent && !relations) {
|
||||||
return memoryEvent;
|
return memoryEvent;
|
||||||
}
|
}
|
||||||
|
|
||||||
const dbEvent = await eventsDB.filter(
|
const dbEvent = await eventsDB.query(
|
||||||
[{ authors: [pubkey], relations, kinds: [0], limit: 1 }],
|
[{ authors: [pubkey], relations, kinds: [0], limit: 1 }],
|
||||||
{ limit: 1, signal },
|
{ limit: 1, signal },
|
||||||
).then(([event]) => event);
|
).then(([event]) => event);
|
||||||
|
@ -96,7 +96,7 @@ const getAuthor = async (pubkey: string, opts: GetEventOpts = {}): Promise<Nostr
|
||||||
|
|
||||||
/** Get users the given pubkey follows. */
|
/** Get users the given pubkey follows. */
|
||||||
const getFollows = async (pubkey: string, signal?: AbortSignal): Promise<NostrEvent | undefined> => {
|
const getFollows = async (pubkey: string, signal?: AbortSignal): Promise<NostrEvent | undefined> => {
|
||||||
const [event] = await eventsDB.filter([{ authors: [pubkey], kinds: [3], limit: 1 }], { limit: 1, signal });
|
const [event] = await eventsDB.query([{ authors: [pubkey], kinds: [3], limit: 1 }], { limit: 1, signal });
|
||||||
return event;
|
return event;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -132,7 +132,7 @@ async function getAncestors(event: NostrEvent, result: NostrEvent[] = []): Promi
|
||||||
}
|
}
|
||||||
|
|
||||||
function getDescendants(eventId: string, signal = AbortSignal.timeout(2000)): Promise<NostrEvent[]> {
|
function getDescendants(eventId: string, signal = AbortSignal.timeout(2000)): Promise<NostrEvent[]> {
|
||||||
return eventsDB.filter(
|
return eventsDB.query(
|
||||||
[{ kinds: [1], '#e': [eventId], relations: ['author', 'event_stats', 'author_stats'] }],
|
[{ kinds: [1], '#e': [eventId], relations: ['author', 'event_stats', 'author_stats'] }],
|
||||||
{ limit: 200, signal },
|
{ limit: 200, signal },
|
||||||
);
|
);
|
||||||
|
@ -140,7 +140,7 @@ function getDescendants(eventId: string, signal = AbortSignal.timeout(2000)): Pr
|
||||||
|
|
||||||
/** Returns whether the pubkey is followed by a local user. */
|
/** Returns whether the pubkey is followed by a local user. */
|
||||||
async function isLocallyFollowed(pubkey: string): Promise<boolean> {
|
async function isLocallyFollowed(pubkey: string): Promise<boolean> {
|
||||||
const [event] = await eventsDB.filter([{ kinds: [3], '#p': [pubkey], local: true, limit: 1 }], { limit: 1 });
|
const [event] = await eventsDB.query([{ kinds: [3], '#p': [pubkey], local: true, limit: 1 }], { limit: 1 });
|
||||||
return Boolean(event);
|
return Boolean(event);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -126,7 +126,7 @@ function eventStatsQuery(diffs: EventStatDiff[]) {
|
||||||
|
|
||||||
/** Get the last version of the event, if any. */
|
/** Get the last version of the event, if any. */
|
||||||
async function maybeGetPrev(event: NostrEvent): Promise<NostrEvent> {
|
async function maybeGetPrev(event: NostrEvent): Promise<NostrEvent> {
|
||||||
const [prev] = await eventsDB.filter([
|
const [prev] = await eventsDB.query([
|
||||||
{ kinds: [event.kind], authors: [event.pubkey], limit: 1 },
|
{ kinds: [event.kind], authors: [event.pubkey], limit: 1 },
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
|
|
@ -11,36 +11,36 @@ const eventsDB = new EventsDB(db);
|
||||||
|
|
||||||
Deno.test('count filters', async () => {
|
Deno.test('count filters', async () => {
|
||||||
assertEquals(await eventsDB.count([{ kinds: [1] }]), 0);
|
assertEquals(await eventsDB.count([{ kinds: [1] }]), 0);
|
||||||
await eventsDB.add(event1);
|
await eventsDB.event(event1);
|
||||||
assertEquals(await eventsDB.count([{ kinds: [1] }]), 1);
|
assertEquals(await eventsDB.count([{ kinds: [1] }]), 1);
|
||||||
});
|
});
|
||||||
|
|
||||||
Deno.test('insert and filter events', async () => {
|
Deno.test('insert and filter events', async () => {
|
||||||
await eventsDB.add(event1);
|
await eventsDB.event(event1);
|
||||||
|
|
||||||
assertEquals(await eventsDB.filter([{ kinds: [1] }]), [event1]);
|
assertEquals(await eventsDB.query([{ kinds: [1] }]), [event1]);
|
||||||
assertEquals(await eventsDB.filter([{ kinds: [3] }]), []);
|
assertEquals(await eventsDB.query([{ kinds: [3] }]), []);
|
||||||
assertEquals(await eventsDB.filter([{ since: 1691091000 }]), [event1]);
|
assertEquals(await eventsDB.query([{ since: 1691091000 }]), [event1]);
|
||||||
assertEquals(await eventsDB.filter([{ until: 1691091000 }]), []);
|
assertEquals(await eventsDB.query([{ until: 1691091000 }]), []);
|
||||||
assertEquals(
|
assertEquals(
|
||||||
await eventsDB.filter([{ '#proxy': ['https://gleasonator.com/objects/8f6fac53-4f66-4c6e-ac7d-92e5e78c3e79'] }]),
|
await eventsDB.query([{ '#proxy': ['https://gleasonator.com/objects/8f6fac53-4f66-4c6e-ac7d-92e5e78c3e79'] }]),
|
||||||
[event1],
|
[event1],
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
Deno.test('delete events', async () => {
|
Deno.test('delete events', async () => {
|
||||||
await eventsDB.add(event1);
|
await eventsDB.event(event1);
|
||||||
assertEquals(await eventsDB.filter([{ kinds: [1] }]), [event1]);
|
assertEquals(await eventsDB.query([{ kinds: [1] }]), [event1]);
|
||||||
await eventsDB.deleteFilters([{ kinds: [1] }]);
|
await eventsDB.remove([{ kinds: [1] }]);
|
||||||
assertEquals(await eventsDB.filter([{ kinds: [1] }]), []);
|
assertEquals(await eventsDB.query([{ kinds: [1] }]), []);
|
||||||
});
|
});
|
||||||
|
|
||||||
Deno.test('query events with local filter', async () => {
|
Deno.test('query events with local filter', async () => {
|
||||||
await eventsDB.add(event1);
|
await eventsDB.event(event1);
|
||||||
|
|
||||||
assertEquals(await eventsDB.filter([{}]), [event1]);
|
assertEquals(await eventsDB.query([{}]), [event1]);
|
||||||
assertEquals(await eventsDB.filter([{ local: true }]), []);
|
assertEquals(await eventsDB.query([{ local: true }]), []);
|
||||||
assertEquals(await eventsDB.filter([{ local: false }]), [event1]);
|
assertEquals(await eventsDB.query([{ local: false }]), [event1]);
|
||||||
|
|
||||||
const userEvent = await buildUserEvent({
|
const userEvent = await buildUserEvent({
|
||||||
username: 'alex',
|
username: 'alex',
|
||||||
|
@ -48,20 +48,20 @@ Deno.test('query events with local filter', async () => {
|
||||||
inserted_at: new Date(),
|
inserted_at: new Date(),
|
||||||
admin: false,
|
admin: false,
|
||||||
});
|
});
|
||||||
await eventsDB.add(userEvent);
|
await eventsDB.event(userEvent);
|
||||||
|
|
||||||
assertEquals(await eventsDB.filter([{ kinds: [1], local: true }]), [event1]);
|
assertEquals(await eventsDB.query([{ kinds: [1], local: true }]), [event1]);
|
||||||
assertEquals(await eventsDB.filter([{ kinds: [1], local: false }]), []);
|
assertEquals(await eventsDB.query([{ kinds: [1], local: false }]), []);
|
||||||
});
|
});
|
||||||
|
|
||||||
Deno.test('inserting replaceable events', async () => {
|
Deno.test('inserting replaceable events', async () => {
|
||||||
assertEquals(await eventsDB.count([{ kinds: [0], authors: [event0.pubkey] }]), 0);
|
assertEquals(await eventsDB.count([{ kinds: [0], authors: [event0.pubkey] }]), 0);
|
||||||
|
|
||||||
await eventsDB.add(event0);
|
await eventsDB.event(event0);
|
||||||
await assertRejects(() => eventsDB.add(event0));
|
await assertRejects(() => eventsDB.event(event0));
|
||||||
assertEquals(await eventsDB.count([{ kinds: [0], authors: [event0.pubkey] }]), 1);
|
assertEquals(await eventsDB.count([{ kinds: [0], authors: [event0.pubkey] }]), 1);
|
||||||
|
|
||||||
const changeEvent = { ...event0, id: '123', created_at: event0.created_at + 1 };
|
const changeEvent = { ...event0, id: '123', created_at: event0.created_at + 1 };
|
||||||
await eventsDB.add(changeEvent);
|
await eventsDB.event(changeEvent);
|
||||||
assertEquals(await eventsDB.filter([{ kinds: [0] }]), [changeEvent]);
|
assertEquals(await eventsDB.query([{ kinds: [0] }]), [changeEvent]);
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import { Conf } from '@/config.ts';
|
import { Conf } from '@/config.ts';
|
||||||
import { type DittoDB } from '@/db.ts';
|
import { type DittoDB } from '@/db.ts';
|
||||||
import { Debug, Kysely, type NostrEvent, type SelectQueryBuilder } from '@/deps.ts';
|
import { Debug, Kysely, type NostrEvent, type NStore, type NStoreOpts, type SelectQueryBuilder } from '@/deps.ts';
|
||||||
import { cleanEvent } from '@/events.ts';
|
import { cleanEvent } from '@/events.ts';
|
||||||
import { normalizeFilters } from '@/filter.ts';
|
import { normalizeFilters } from '@/filter.ts';
|
||||||
import { DittoEvent } from '@/interfaces/DittoEvent.ts';
|
import { DittoEvent } from '@/interfaces/DittoEvent.ts';
|
||||||
|
@ -8,8 +8,7 @@ import { type DittoFilter } from '@/interfaces/DittoFilter.ts';
|
||||||
import { isDittoInternalKind, isParameterizedReplaceableKind, isReplaceableKind } from '@/kinds.ts';
|
import { isDittoInternalKind, isParameterizedReplaceableKind, isReplaceableKind } from '@/kinds.ts';
|
||||||
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 { abortError } from '@/utils/abort.ts';
|
||||||
import { type 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 }: {
|
||||||
|
@ -56,19 +55,16 @@ type EventQuery = SelectQueryBuilder<DittoDB, 'events', {
|
||||||
}>;
|
}>;
|
||||||
|
|
||||||
/** SQLite database storage adapter for Nostr events. */
|
/** SQLite database storage adapter for Nostr events. */
|
||||||
class EventsDB implements EventStore {
|
class EventsDB implements NStore {
|
||||||
#db: Kysely<DittoDB>;
|
#db: Kysely<DittoDB>;
|
||||||
#debug = Debug('ditto:db:events');
|
#debug = Debug('ditto:db:events');
|
||||||
|
|
||||||
/** NIPs supported by this storage method. */
|
|
||||||
supportedNips = [1, 45, 50];
|
|
||||||
|
|
||||||
constructor(db: Kysely<DittoDB>) {
|
constructor(db: Kysely<DittoDB>) {
|
||||||
this.#db = db;
|
this.#db = db;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Insert an event (and its tags) into the database. */
|
/** Insert an event (and its tags) into the database. */
|
||||||
async add(event: NostrEvent): Promise<void> {
|
async event(event: NostrEvent, _opts?: NStoreOpts): Promise<void> {
|
||||||
event = cleanEvent(event);
|
event = cleanEvent(event);
|
||||||
this.#debug('EVENT', JSON.stringify(event));
|
this.#debug('EVENT', JSON.stringify(event));
|
||||||
|
|
||||||
|
@ -268,7 +264,7 @@ class EventsDB implements EventStore {
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Get events for filters from the database. */
|
/** Get events for filters from the database. */
|
||||||
async filter(filters: DittoFilter[], opts: GetEventsOpts = {}): Promise<DittoEvent[]> {
|
async query(filters: DittoFilter[], opts: NStoreOpts = {}): Promise<DittoEvent[]> {
|
||||||
filters = normalizeFilters(filters); // Improves performance of `{ kinds: [0], authors: ['...'] }` queries.
|
filters = normalizeFilters(filters); // Improves performance of `{ kinds: [0], authors: ['...'] }` queries.
|
||||||
|
|
||||||
if (opts.signal?.aborted) return Promise.resolve([]);
|
if (opts.signal?.aborted) return Promise.resolve([]);
|
||||||
|
@ -341,7 +337,7 @@ class EventsDB implements EventStore {
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Delete events based on filters from the database. */
|
/** Delete events based on filters from the database. */
|
||||||
async deleteFilters(filters: DittoFilter[]): Promise<void> {
|
async remove(filters: DittoFilter[], _opts?: NStoreOpts): Promise<void> {
|
||||||
if (!filters.length) return Promise.resolve();
|
if (!filters.length) return Promise.resolve();
|
||||||
this.#debug('DELETE', JSON.stringify(filters));
|
this.#debug('DELETE', JSON.stringify(filters));
|
||||||
|
|
||||||
|
@ -349,8 +345,10 @@ class EventsDB implements EventStore {
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Get number of events that would be returned by filters. */
|
/** Get number of events that would be returned by filters. */
|
||||||
async count(filters: DittoFilter[]): Promise<number> {
|
async count(filters: DittoFilter[], opts: NStoreOpts = {}): Promise<number> {
|
||||||
|
if (opts.signal?.aborted) return Promise.reject(abortError());
|
||||||
if (!filters.length) return Promise.resolve(0);
|
if (!filters.length) return Promise.resolve(0);
|
||||||
|
|
||||||
this.#debug('COUNT', JSON.stringify(filters));
|
this.#debug('COUNT', JSON.stringify(filters));
|
||||||
const query = this.getEventsQuery(filters);
|
const query = this.getEventsQuery(filters);
|
||||||
|
|
||||||
|
|
|
@ -1,11 +1,11 @@
|
||||||
|
import { type NStore } from '@/deps.ts';
|
||||||
import { type DittoEvent } from '@/interfaces/DittoEvent.ts';
|
import { type DittoEvent } from '@/interfaces/DittoEvent.ts';
|
||||||
import { type DittoFilter } from '@/interfaces/DittoFilter.ts';
|
import { type DittoFilter } from '@/interfaces/DittoFilter.ts';
|
||||||
import { type EventStore } from '@/storages/types.ts';
|
|
||||||
|
|
||||||
interface HydrateEventOpts {
|
interface HydrateEventOpts {
|
||||||
events: DittoEvent[];
|
events: DittoEvent[];
|
||||||
filters: DittoFilter[];
|
filters: DittoFilter[];
|
||||||
storage: EventStore;
|
storage: NStore;
|
||||||
signal?: AbortSignal;
|
signal?: AbortSignal;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -15,7 +15,7 @@ async function hydrateEvents(opts: HydrateEventOpts): Promise<DittoEvent[]> {
|
||||||
|
|
||||||
if (filters.some((filter) => filter.relations?.includes('author'))) {
|
if (filters.some((filter) => filter.relations?.includes('author'))) {
|
||||||
const pubkeys = new Set([...events].map((event) => event.pubkey));
|
const pubkeys = new Set([...events].map((event) => event.pubkey));
|
||||||
const authors = await storage.filter([{ kinds: [0], authors: [...pubkeys] }], { signal });
|
const authors = await storage.query([{ kinds: [0], authors: [...pubkeys] }], { signal });
|
||||||
|
|
||||||
for (const event of events) {
|
for (const event of events) {
|
||||||
event.author = authors.find((author) => author.pubkey === event.pubkey);
|
event.author = authors.find((author) => author.pubkey === event.pubkey);
|
||||||
|
|
|
@ -13,10 +13,10 @@ const memorelay = new Memorelay({
|
||||||
Deno.test('memorelay', async () => {
|
Deno.test('memorelay', async () => {
|
||||||
assertEquals(await memorelay.count([{ ids: [event1.id] }]), 0);
|
assertEquals(await memorelay.count([{ ids: [event1.id] }]), 0);
|
||||||
|
|
||||||
await memorelay.add(event1);
|
await memorelay.event(event1);
|
||||||
|
|
||||||
assertEquals(await memorelay.count([{ ids: [event1.id] }]), 1);
|
assertEquals(await memorelay.count([{ ids: [event1.id] }]), 1);
|
||||||
|
|
||||||
const result = await memorelay.filter([{ ids: [event1.id] }]);
|
const result = await memorelay.query([{ ids: [event1.id] }]);
|
||||||
assertEquals(result[0], event1);
|
assertEquals(result[0], event1);
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,16 +1,21 @@
|
||||||
import { Debug, LRUCache, matchFilter, type NostrEvent, type NostrFilter, NSet } from '@/deps.ts';
|
import {
|
||||||
|
Debug,
|
||||||
|
LRUCache,
|
||||||
|
matchFilter,
|
||||||
|
type NostrEvent,
|
||||||
|
type NostrFilter,
|
||||||
|
NSet,
|
||||||
|
type NStore,
|
||||||
|
type NStoreOpts,
|
||||||
|
} from '@/deps.ts';
|
||||||
import { normalizeFilters } from '@/filter.ts';
|
import { normalizeFilters } from '@/filter.ts';
|
||||||
|
import { abortError } from '@/utils/abort.ts';
|
||||||
import { type EventStore, type GetEventsOpts } from './types.ts';
|
|
||||||
|
|
||||||
/** In-memory data store for events. */
|
/** In-memory data store for events. */
|
||||||
class Memorelay implements EventStore {
|
class Memorelay implements NStore {
|
||||||
#debug = Debug('ditto:memorelay');
|
#debug = Debug('ditto:memorelay');
|
||||||
#cache: LRUCache<string, NostrEvent>;
|
#cache: LRUCache<string, NostrEvent>;
|
||||||
|
|
||||||
/** NIPs supported by this storage method. */
|
|
||||||
supportedNips = [1, 45];
|
|
||||||
|
|
||||||
constructor(...args: ConstructorParameters<typeof LRUCache<string, NostrEvent>>) {
|
constructor(...args: ConstructorParameters<typeof LRUCache<string, NostrEvent>>) {
|
||||||
this.#cache = new LRUCache<string, NostrEvent>(...args);
|
this.#cache = new LRUCache<string, NostrEvent>(...args);
|
||||||
}
|
}
|
||||||
|
@ -25,13 +30,12 @@ class Memorelay implements EventStore {
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Get events from memory. */
|
/** Get events from memory. */
|
||||||
filter(filters: NostrFilter[], opts: GetEventsOpts = {}): Promise<NostrEvent[]> {
|
query(filters: NostrFilter[], opts: NStoreOpts = {}): Promise<NostrEvent[]> {
|
||||||
|
if (opts.signal?.aborted) return Promise.reject(abortError());
|
||||||
|
|
||||||
filters = normalizeFilters(filters);
|
filters = normalizeFilters(filters);
|
||||||
|
|
||||||
if (opts.signal?.aborted) return Promise.resolve([]);
|
|
||||||
if (!filters.length) return Promise.resolve([]);
|
|
||||||
|
|
||||||
this.#debug('REQ', JSON.stringify(filters));
|
this.#debug('REQ', JSON.stringify(filters));
|
||||||
|
if (!filters.length) return Promise.resolve([]);
|
||||||
|
|
||||||
/** Event results to return. */
|
/** Event results to return. */
|
||||||
const results = new NSet<NostrEvent>();
|
const results = new NSet<NostrEvent>();
|
||||||
|
@ -90,20 +94,21 @@ class Memorelay implements EventStore {
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Insert an event into memory. */
|
/** Insert an event into memory. */
|
||||||
add(event: NostrEvent): Promise<void> {
|
event(event: NostrEvent, opts: NStoreOpts = {}): Promise<void> {
|
||||||
|
if (opts.signal?.aborted) return Promise.reject(abortError());
|
||||||
this.#cache.set(event.id, event);
|
this.#cache.set(event.id, event);
|
||||||
return Promise.resolve();
|
return Promise.resolve();
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Count events in memory for the filters. */
|
/** Count events in memory for the filters. */
|
||||||
async count(filters: NostrFilter[]): Promise<number> {
|
async count(filters: NostrFilter[], opts?: NStoreOpts): Promise<number> {
|
||||||
const events = await this.filter(filters);
|
const events = await this.query(filters, opts);
|
||||||
return events.length;
|
return events.length;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Delete events from memory. */
|
/** Delete events from memory. */
|
||||||
async deleteFilters(filters: NostrFilter[]): Promise<void> {
|
async remove(filters: NostrFilter[], opts: NStoreOpts): Promise<void> {
|
||||||
for (const event of await this.filter(filters)) {
|
for (const event of await this.query(filters, opts)) {
|
||||||
this.#cache.delete(event.id);
|
this.#cache.delete(event.id);
|
||||||
}
|
}
|
||||||
return Promise.resolve();
|
return Promise.resolve();
|
||||||
|
|
|
@ -1,24 +1,21 @@
|
||||||
import { Debug, NSet } from '@/deps.ts';
|
import { Debug, NSet, type NStore, type NStoreOpts } from '@/deps.ts';
|
||||||
import { normalizeFilters } from '@/filter.ts';
|
import { normalizeFilters } from '@/filter.ts';
|
||||||
import { type DittoEvent } from '@/interfaces/DittoEvent.ts';
|
import { type DittoEvent } from '@/interfaces/DittoEvent.ts';
|
||||||
import { type DittoFilter } from '@/interfaces/DittoFilter.ts';
|
import { type DittoFilter } from '@/interfaces/DittoFilter.ts';
|
||||||
|
import { abortError } from '@/utils/abort.ts';
|
||||||
import { type EventStore, type GetEventsOpts, type StoreEventOpts } from './types.ts';
|
|
||||||
|
|
||||||
interface OptimizerOpts {
|
interface OptimizerOpts {
|
||||||
db: EventStore;
|
db: NStore;
|
||||||
cache: EventStore;
|
cache: NStore;
|
||||||
client: EventStore;
|
client: NStore;
|
||||||
}
|
}
|
||||||
|
|
||||||
class Optimizer implements EventStore {
|
class Optimizer implements NStore {
|
||||||
#debug = Debug('ditto:optimizer');
|
#debug = Debug('ditto:optimizer');
|
||||||
|
|
||||||
#db: EventStore;
|
#db: NStore;
|
||||||
#cache: EventStore;
|
#cache: NStore;
|
||||||
#client: EventStore;
|
#client: NStore;
|
||||||
|
|
||||||
supportedNips = [1];
|
|
||||||
|
|
||||||
constructor(opts: OptimizerOpts) {
|
constructor(opts: OptimizerOpts) {
|
||||||
this.#db = opts.db;
|
this.#db = opts.db;
|
||||||
|
@ -26,25 +23,23 @@ class Optimizer implements EventStore {
|
||||||
this.#client = opts.client;
|
this.#client = opts.client;
|
||||||
}
|
}
|
||||||
|
|
||||||
async add(event: DittoEvent, opts?: StoreEventOpts | undefined): Promise<void> {
|
async event(event: DittoEvent, opts?: NStoreOpts | undefined): Promise<void> {
|
||||||
|
if (opts?.signal?.aborted) return Promise.reject(abortError());
|
||||||
|
|
||||||
await Promise.all([
|
await Promise.all([
|
||||||
this.#db.add(event, opts),
|
this.#db.event(event, opts),
|
||||||
this.#cache.add(event, opts),
|
this.#cache.event(event, opts),
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
async filter(
|
async query(filters: DittoFilter[], opts: NStoreOpts = {}): Promise<DittoEvent[]> {
|
||||||
filters: DittoFilter[],
|
if (opts?.signal?.aborted) return Promise.reject(abortError());
|
||||||
opts: GetEventsOpts | undefined = {},
|
|
||||||
): Promise<DittoEvent[]> {
|
|
||||||
this.#debug('REQ', JSON.stringify(filters));
|
|
||||||
|
|
||||||
const { limit = Infinity } = opts;
|
|
||||||
filters = normalizeFilters(filters);
|
filters = normalizeFilters(filters);
|
||||||
|
this.#debug('REQ', JSON.stringify(filters));
|
||||||
if (opts?.signal?.aborted) return Promise.resolve([]);
|
|
||||||
if (!filters.length) return Promise.resolve([]);
|
if (!filters.length) return Promise.resolve([]);
|
||||||
|
|
||||||
|
const { limit = Infinity } = opts;
|
||||||
const results = new NSet<DittoEvent>();
|
const results = new NSet<DittoEvent>();
|
||||||
|
|
||||||
// Filters with IDs are immutable, so we can take them straight from the cache if we have them.
|
// Filters with IDs are immutable, so we can take them straight from the cache if we have them.
|
||||||
|
@ -53,7 +48,7 @@ class Optimizer implements EventStore {
|
||||||
if (filter.ids) {
|
if (filter.ids) {
|
||||||
this.#debug(`Filter[${i}] is an IDs filter; querying cache...`);
|
this.#debug(`Filter[${i}] is an IDs filter; querying cache...`);
|
||||||
const ids = new Set<string>(filter.ids);
|
const ids = new Set<string>(filter.ids);
|
||||||
for (const event of await this.#cache.filter([filter], opts)) {
|
for (const event of await this.#cache.query([filter], opts)) {
|
||||||
ids.delete(event.id);
|
ids.delete(event.id);
|
||||||
results.add(event);
|
results.add(event);
|
||||||
if (results.size >= limit) return getResults();
|
if (results.size >= limit) return getResults();
|
||||||
|
@ -67,7 +62,7 @@ class Optimizer implements EventStore {
|
||||||
|
|
||||||
// Query the database for events.
|
// Query the database for events.
|
||||||
this.#debug('Querying database...');
|
this.#debug('Querying database...');
|
||||||
for (const dbEvent of await this.#db.filter(filters, opts)) {
|
for (const dbEvent of await this.#db.query(filters, opts)) {
|
||||||
results.add(dbEvent);
|
results.add(dbEvent);
|
||||||
if (results.size >= limit) return getResults();
|
if (results.size >= limit) return getResults();
|
||||||
}
|
}
|
||||||
|
@ -80,14 +75,14 @@ class Optimizer implements EventStore {
|
||||||
|
|
||||||
// Query the cache again.
|
// Query the cache again.
|
||||||
this.#debug('Querying cache...');
|
this.#debug('Querying cache...');
|
||||||
for (const cacheEvent of await this.#cache.filter(filters, opts)) {
|
for (const cacheEvent of await this.#cache.query(filters, opts)) {
|
||||||
results.add(cacheEvent);
|
results.add(cacheEvent);
|
||||||
if (results.size >= limit) return getResults();
|
if (results.size >= limit) return getResults();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Finally, query the client.
|
// Finally, query the client.
|
||||||
this.#debug('Querying client...');
|
this.#debug('Querying client...');
|
||||||
for (const clientEvent of await this.#client.filter(filters, opts)) {
|
for (const clientEvent of await this.#client.query(filters, opts)) {
|
||||||
results.add(clientEvent);
|
results.add(clientEvent);
|
||||||
if (results.size >= limit) return getResults();
|
if (results.size >= limit) return getResults();
|
||||||
}
|
}
|
||||||
|
@ -100,12 +95,12 @@ class Optimizer implements EventStore {
|
||||||
return getResults();
|
return getResults();
|
||||||
}
|
}
|
||||||
|
|
||||||
countEvents(_filters: DittoFilter[]): Promise<number> {
|
count(_filters: DittoFilter[]): Promise<number> {
|
||||||
throw new Error('COUNT not implemented.');
|
return Promise.reject(new Error('COUNT not implemented.'));
|
||||||
}
|
}
|
||||||
|
|
||||||
deleteEvents(_filters: DittoFilter[]): Promise<void> {
|
remove(_filters: DittoFilter[]): Promise<void> {
|
||||||
throw new Error('DELETE not implemented.');
|
return Promise.reject(new Error('DELETE not implemented.'));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,16 @@
|
||||||
import { Debug, matchFilters, type NostrEvent, type NostrFilter, NSet, type RelayPoolWorker } from '@/deps.ts';
|
import {
|
||||||
|
Debug,
|
||||||
|
matchFilters,
|
||||||
|
type NostrEvent,
|
||||||
|
type NostrFilter,
|
||||||
|
NSet,
|
||||||
|
type NStore,
|
||||||
|
type NStoreOpts,
|
||||||
|
type RelayPoolWorker,
|
||||||
|
} from '@/deps.ts';
|
||||||
import { cleanEvent } from '@/events.ts';
|
import { cleanEvent } from '@/events.ts';
|
||||||
import { normalizeFilters } from '@/filter.ts';
|
import { normalizeFilters } from '@/filter.ts';
|
||||||
import { type EventStore, type GetEventsOpts, type StoreEventOpts } from '@/storages/types.ts';
|
import { abortError } from '@/utils/abort.ts';
|
||||||
|
|
||||||
interface PoolStoreOpts {
|
interface PoolStoreOpts {
|
||||||
pool: InstanceType<typeof RelayPoolWorker>;
|
pool: InstanceType<typeof RelayPoolWorker>;
|
||||||
|
@ -11,7 +20,7 @@ interface PoolStoreOpts {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
class PoolStore implements EventStore {
|
class PoolStore implements NStore {
|
||||||
#debug = Debug('ditto:client');
|
#debug = Debug('ditto:client');
|
||||||
#pool: InstanceType<typeof RelayPoolWorker>;
|
#pool: InstanceType<typeof RelayPoolWorker>;
|
||||||
#relays: WebSocket['url'][];
|
#relays: WebSocket['url'][];
|
||||||
|
@ -19,31 +28,31 @@ class PoolStore implements EventStore {
|
||||||
handleEvent(event: NostrEvent): Promise<void>;
|
handleEvent(event: NostrEvent): Promise<void>;
|
||||||
};
|
};
|
||||||
|
|
||||||
supportedNips = [1];
|
|
||||||
|
|
||||||
constructor(opts: PoolStoreOpts) {
|
constructor(opts: PoolStoreOpts) {
|
||||||
this.#pool = opts.pool;
|
this.#pool = opts.pool;
|
||||||
this.#relays = opts.relays;
|
this.#relays = opts.relays;
|
||||||
this.#publisher = opts.publisher;
|
this.#publisher = opts.publisher;
|
||||||
}
|
}
|
||||||
|
|
||||||
add(event: NostrEvent, opts: StoreEventOpts = {}): Promise<void> {
|
event(event: NostrEvent, opts: NStoreOpts = {}): Promise<void> {
|
||||||
|
if (opts.signal?.aborted) return Promise.reject(abortError());
|
||||||
const { relays = this.#relays } = opts;
|
const { relays = this.#relays } = opts;
|
||||||
|
|
||||||
event = cleanEvent(event);
|
event = cleanEvent(event);
|
||||||
this.#debug('EVENT', event);
|
this.#debug('EVENT', event);
|
||||||
|
|
||||||
this.#pool.publish(event, relays);
|
this.#pool.publish(event, relays);
|
||||||
return Promise.resolve();
|
return Promise.resolve();
|
||||||
}
|
}
|
||||||
|
|
||||||
filter(filters: NostrFilter[], opts: GetEventsOpts = {}): Promise<NostrEvent[]> {
|
query(filters: NostrFilter[], opts: NStoreOpts = {}): Promise<NostrEvent[]> {
|
||||||
filters = normalizeFilters(filters);
|
if (opts.signal?.aborted) return Promise.reject(abortError());
|
||||||
|
|
||||||
if (opts.signal?.aborted) return Promise.resolve([]);
|
filters = normalizeFilters(filters);
|
||||||
|
this.#debug('REQ', JSON.stringify(filters));
|
||||||
if (!filters.length) return Promise.resolve([]);
|
if (!filters.length) return Promise.resolve([]);
|
||||||
|
|
||||||
this.#debug('REQ', JSON.stringify(filters));
|
return new Promise((resolve, reject) => {
|
||||||
|
|
||||||
return new Promise((resolve) => {
|
|
||||||
const results = new NSet<NostrEvent>();
|
const results = new NSet<NostrEvent>();
|
||||||
|
|
||||||
const unsub = this.#pool.subscribe(
|
const unsub = this.#pool.subscribe(
|
||||||
|
@ -74,10 +83,13 @@ class PoolStore implements EventStore {
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
opts.signal?.addEventListener('abort', () => {
|
const onAbort = () => {
|
||||||
unsub();
|
unsub();
|
||||||
resolve([...results]);
|
reject(abortError());
|
||||||
});
|
opts.signal?.removeEventListener('abort', onAbort);
|
||||||
|
};
|
||||||
|
|
||||||
|
opts.signal?.addEventListener('abort', onAbort);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -85,7 +97,7 @@ class PoolStore implements EventStore {
|
||||||
return Promise.reject(new Error('COUNT not implemented'));
|
return Promise.reject(new Error('COUNT not implemented'));
|
||||||
}
|
}
|
||||||
|
|
||||||
deleteFilters() {
|
remove() {
|
||||||
return Promise.reject(new Error('Cannot delete events from relays. Create a kind 5 event instead.'));
|
return Promise.reject(new Error('Cannot delete events from relays. Create a kind 5 event instead.'));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,10 +1,10 @@
|
||||||
import { Debug, EventEmitter, type NostrEvent, type NostrFilter } from '@/deps.ts';
|
import { Debug, EventEmitter, type NostrEvent, type NostrFilter, type NStore, type NStoreOpts } from '@/deps.ts';
|
||||||
import { eventToMicroFilter, getFilterId, isMicrofilter, type MicroFilter } from '@/filter.ts';
|
import { eventToMicroFilter, getFilterId, isMicrofilter, type MicroFilter } from '@/filter.ts';
|
||||||
import { type EventStore, GetEventsOpts } from '@/storages/types.ts';
|
|
||||||
import { Time } from '@/utils/time.ts';
|
import { Time } from '@/utils/time.ts';
|
||||||
|
import { abortError } from '@/utils/abort.ts';
|
||||||
|
|
||||||
interface ReqmeisterOpts {
|
interface ReqmeisterOpts {
|
||||||
client: EventStore;
|
client: NStore;
|
||||||
delay?: number;
|
delay?: number;
|
||||||
timeout?: number;
|
timeout?: number;
|
||||||
}
|
}
|
||||||
|
@ -17,7 +17,7 @@ interface ReqmeisterReqOpts {
|
||||||
type ReqmeisterQueueItem = [string, MicroFilter, WebSocket['url'][]];
|
type ReqmeisterQueueItem = [string, MicroFilter, WebSocket['url'][]];
|
||||||
|
|
||||||
/** Batches requests to Nostr relays using microfilters. */
|
/** Batches requests to Nostr relays using microfilters. */
|
||||||
class Reqmeister extends EventEmitter<{ [filterId: string]: (event: NostrEvent) => any }> implements EventStore {
|
class Reqmeister extends EventEmitter<{ [filterId: string]: (event: NostrEvent) => any }> implements NStore {
|
||||||
#debug = Debug('ditto:reqmeister');
|
#debug = Debug('ditto:reqmeister');
|
||||||
|
|
||||||
#opts: ReqmeisterOpts;
|
#opts: ReqmeisterOpts;
|
||||||
|
@ -25,8 +25,6 @@ class Reqmeister extends EventEmitter<{ [filterId: string]: (event: NostrEvent)
|
||||||
#promise!: Promise<void>;
|
#promise!: Promise<void>;
|
||||||
#resolve!: () => void;
|
#resolve!: () => void;
|
||||||
|
|
||||||
supportedNips = [];
|
|
||||||
|
|
||||||
constructor(opts: ReqmeisterOpts) {
|
constructor(opts: ReqmeisterOpts) {
|
||||||
super();
|
super();
|
||||||
this.#opts = opts;
|
this.#opts = opts;
|
||||||
|
@ -66,11 +64,10 @@ class Reqmeister extends EventEmitter<{ [filterId: string]: (event: NostrEvent)
|
||||||
if (wantedAuthors.size) filters.push({ kinds: [0], authors: [...wantedAuthors] });
|
if (wantedAuthors.size) filters.push({ kinds: [0], authors: [...wantedAuthors] });
|
||||||
|
|
||||||
if (filters.length) {
|
if (filters.length) {
|
||||||
this.#debug('REQ', JSON.stringify(filters));
|
const events = await client.query(filters, { signal: AbortSignal.timeout(timeout) });
|
||||||
const events = await client.filter(filters, { signal: AbortSignal.timeout(timeout) });
|
|
||||||
|
|
||||||
for (const event of events) {
|
for (const event of events) {
|
||||||
this.add(event);
|
this.event(event);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -85,7 +82,7 @@ class Reqmeister extends EventEmitter<{ [filterId: string]: (event: NostrEvent)
|
||||||
} = opts;
|
} = opts;
|
||||||
|
|
||||||
if (signal.aborted) {
|
if (signal.aborted) {
|
||||||
return Promise.reject(new DOMException('Aborted', 'AbortError'));
|
return Promise.reject(abortError());
|
||||||
}
|
}
|
||||||
|
|
||||||
const filterId = getFilterId(filter);
|
const filterId = getFilterId(filter);
|
||||||
|
@ -109,7 +106,7 @@ class Reqmeister extends EventEmitter<{ [filterId: string]: (event: NostrEvent)
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
add(event: NostrEvent): Promise<void> {
|
event(event: NostrEvent, _opts?: NStoreOpts): Promise<void> {
|
||||||
const filterId = getFilterId(eventToMicroFilter(event));
|
const filterId = getFilterId(eventToMicroFilter(event));
|
||||||
this.#queue = this.#queue.filter(([id]) => id !== filterId);
|
this.#queue = this.#queue.filter(([id]) => id !== filterId);
|
||||||
this.emit(filterId, event);
|
this.emit(filterId, event);
|
||||||
|
@ -121,13 +118,15 @@ class Reqmeister extends EventEmitter<{ [filterId: string]: (event: NostrEvent)
|
||||||
return this.#queue.some(([id]) => id === filterId);
|
return this.#queue.some(([id]) => id === filterId);
|
||||||
}
|
}
|
||||||
|
|
||||||
filter(filters: NostrFilter[], opts?: GetEventsOpts | undefined): Promise<NostrEvent[]> {
|
query(filters: NostrFilter[], opts?: NStoreOpts): Promise<NostrEvent[]> {
|
||||||
if (opts?.signal?.aborted) return Promise.resolve([]);
|
if (opts?.signal?.aborted) return Promise.reject(abortError());
|
||||||
|
|
||||||
|
this.#debug('REQ', JSON.stringify(filters));
|
||||||
if (!filters.length) return Promise.resolve([]);
|
if (!filters.length) return Promise.resolve([]);
|
||||||
|
|
||||||
const promises = filters.reduce<Promise<NostrEvent>[]>((result, filter) => {
|
const promises = filters.reduce<Promise<NostrEvent>[]>((result, filter) => {
|
||||||
if (isMicrofilter(filter)) {
|
if (isMicrofilter(filter)) {
|
||||||
result.push(this.req(filter) as Promise<NostrEvent>);
|
result.push(this.req(filter, opts));
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
}, []);
|
}, []);
|
||||||
|
@ -136,11 +135,11 @@ class Reqmeister extends EventEmitter<{ [filterId: string]: (event: NostrEvent)
|
||||||
}
|
}
|
||||||
|
|
||||||
count(_filters: NostrFilter[]): Promise<number> {
|
count(_filters: NostrFilter[]): Promise<number> {
|
||||||
throw new Error('COUNT not implemented.');
|
return Promise.reject(new Error('COUNT not implemented.'));
|
||||||
}
|
}
|
||||||
|
|
||||||
deleteFilters(_filters: NostrFilter[]): Promise<void> {
|
remove(_filters: NostrFilter[]): Promise<void> {
|
||||||
throw new Error('DELETE not implemented.');
|
return Promise.reject(new Error('DELETE not implemented.'));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,27 +1,25 @@
|
||||||
import { NiceRelay } from 'https://gitlab.com/soapbox-pub/nostr-machina/-/raw/5f4fb59c90c092e5aa59c01e6556a4bec264c167/mod.ts';
|
import { NiceRelay } from 'https://gitlab.com/soapbox-pub/nostr-machina/-/raw/5f4fb59c90c092e5aa59c01e6556a4bec264c167/mod.ts';
|
||||||
|
|
||||||
import { Debug, type NostrEvent, type NostrFilter, NSet } from '@/deps.ts';
|
import { Debug, type NostrEvent, type NostrFilter, NSet, type NStore, type NStoreOpts } from '@/deps.ts';
|
||||||
import { normalizeFilters } from '@/filter.ts';
|
import { normalizeFilters } from '@/filter.ts';
|
||||||
import { type DittoEvent } from '@/interfaces/DittoEvent.ts';
|
import { type DittoEvent } from '@/interfaces/DittoEvent.ts';
|
||||||
import { type DittoFilter } from '@/interfaces/DittoFilter.ts';
|
import { type DittoFilter } from '@/interfaces/DittoFilter.ts';
|
||||||
import { hydrateEvents } from '@/storages/hydrate.ts';
|
import { hydrateEvents } from '@/storages/hydrate.ts';
|
||||||
import { type EventStore, type GetEventsOpts, type StoreEventOpts } from '@/storages/types.ts';
|
import { abortError } from '@/utils/abort.ts';
|
||||||
|
|
||||||
interface SearchStoreOpts {
|
interface SearchStoreOpts {
|
||||||
relay: string | undefined;
|
relay: string | undefined;
|
||||||
fallback: EventStore;
|
fallback: NStore;
|
||||||
hydrator?: EventStore;
|
hydrator?: NStore;
|
||||||
}
|
}
|
||||||
|
|
||||||
class SearchStore implements EventStore {
|
class SearchStore implements NStore {
|
||||||
#debug = Debug('ditto:storages:search');
|
#debug = Debug('ditto:storages:search');
|
||||||
|
|
||||||
#fallback: EventStore;
|
#fallback: NStore;
|
||||||
#hydrator: EventStore;
|
#hydrator: NStore;
|
||||||
#relay: NiceRelay | undefined;
|
#relay: NiceRelay | undefined;
|
||||||
|
|
||||||
supportedNips = [50];
|
|
||||||
|
|
||||||
constructor(opts: SearchStoreOpts) {
|
constructor(opts: SearchStoreOpts) {
|
||||||
this.#fallback = opts.fallback;
|
this.#fallback = opts.fallback;
|
||||||
this.#hydrator = opts.hydrator ?? this;
|
this.#hydrator = opts.hydrator ?? this;
|
||||||
|
@ -31,17 +29,14 @@ class SearchStore implements EventStore {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
add(_event: NostrEvent, _opts?: StoreEventOpts | undefined): Promise<void> {
|
event(_event: NostrEvent, _opts?: NStoreOpts): Promise<void> {
|
||||||
throw new Error('EVENT not implemented.');
|
return Promise.reject(new Error('EVENT not implemented.'));
|
||||||
}
|
}
|
||||||
|
|
||||||
async filter(
|
async query(filters: DittoFilter[], opts?: NStoreOpts): Promise<DittoEvent[]> {
|
||||||
filters: DittoFilter[],
|
|
||||||
opts?: GetEventsOpts | undefined,
|
|
||||||
): Promise<DittoEvent[]> {
|
|
||||||
filters = normalizeFilters(filters);
|
filters = normalizeFilters(filters);
|
||||||
|
|
||||||
if (opts?.signal?.aborted) return Promise.resolve([]);
|
if (opts?.signal?.aborted) return Promise.reject(abortError());
|
||||||
if (!filters.length) return Promise.resolve([]);
|
if (!filters.length) return Promise.resolve([]);
|
||||||
|
|
||||||
this.#debug('REQ', JSON.stringify(filters));
|
this.#debug('REQ', JSON.stringify(filters));
|
||||||
|
@ -70,16 +65,16 @@ class SearchStore implements EventStore {
|
||||||
return hydrateEvents({ events: [...events], filters, storage: this.#hydrator, signal: opts?.signal });
|
return hydrateEvents({ events: [...events], filters, storage: this.#hydrator, signal: opts?.signal });
|
||||||
} else {
|
} else {
|
||||||
this.#debug(`Searching for "${query}" locally...`);
|
this.#debug(`Searching for "${query}" locally...`);
|
||||||
return this.#fallback.filter(filters, opts);
|
return this.#fallback.query(filters, opts);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
count(_filters: NostrFilter[]): Promise<number> {
|
count(_filters: NostrFilter[]): Promise<number> {
|
||||||
throw new Error('COUNT not implemented.');
|
return Promise.reject(new Error('COUNT not implemented.'));
|
||||||
}
|
}
|
||||||
|
|
||||||
deleteFilters(_filters: NostrFilter[]): Promise<void> {
|
remove(_filters: NostrFilter[]): Promise<void> {
|
||||||
throw new Error('DELETE not implemented.');
|
return Promise.reject(new Error('DELETE not implemented.'));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,34 +0,0 @@
|
||||||
import { type DittoEvent } from '@/interfaces/DittoEvent.ts';
|
|
||||||
import { type DittoFilter } from '@/interfaces/DittoFilter.ts';
|
|
||||||
|
|
||||||
/** Additional options to apply to the whole subscription. */
|
|
||||||
interface GetEventsOpts {
|
|
||||||
/** Signal to abort the request. */
|
|
||||||
signal?: AbortSignal;
|
|
||||||
/** Event limit for the whole subscription. */
|
|
||||||
limit?: number;
|
|
||||||
/** Relays to use, if applicable. */
|
|
||||||
relays?: WebSocket['url'][];
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Options when storing an event. */
|
|
||||||
interface StoreEventOpts {
|
|
||||||
/** Relays to use, if applicable. */
|
|
||||||
relays?: WebSocket['url'][];
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Storage interface for Nostr events. */
|
|
||||||
interface EventStore {
|
|
||||||
/** Indicates NIPs supported by this data store, similar to NIP-11. For example, `50` would indicate support for `search` filters. */
|
|
||||||
supportedNips: readonly number[];
|
|
||||||
/** Add an event to the store. */
|
|
||||||
add(event: DittoEvent, opts?: StoreEventOpts): Promise<void>;
|
|
||||||
/** Get events from filters. */
|
|
||||||
filter(filters: DittoFilter[], opts?: GetEventsOpts): Promise<DittoEvent[]>;
|
|
||||||
/** Get the number of events from filters. */
|
|
||||||
count?(filters: DittoFilter[]): Promise<number>;
|
|
||||||
/** Delete events from filters. */
|
|
||||||
deleteFilters?(filters: DittoFilter[]): Promise<void>;
|
|
||||||
}
|
|
||||||
|
|
||||||
export type { EventStore, GetEventsOpts, StoreEventOpts };
|
|
|
@ -0,0 +1,6 @@
|
||||||
|
/** Creates an `AbortError` object matching the Fetch API. */
|
||||||
|
function abortError() {
|
||||||
|
return new DOMException('The signal has been aborted', 'AbortError');
|
||||||
|
}
|
||||||
|
|
||||||
|
export { abortError };
|
|
@ -51,7 +51,7 @@ async function updateEvent<K extends number, E extends EventStub>(
|
||||||
fn: (prev: NostrEvent | undefined) => E,
|
fn: (prev: NostrEvent | undefined) => E,
|
||||||
c: AppContext,
|
c: AppContext,
|
||||||
): Promise<NostrEvent> {
|
): Promise<NostrEvent> {
|
||||||
const [prev] = await eventsDB.filter([filter], { limit: 1 });
|
const [prev] = await eventsDB.query([filter], { limit: 1 });
|
||||||
return createEvent(fn(prev), c);
|
return createEvent(fn(prev), c);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -11,14 +11,14 @@ async function renderEventAccounts(c: AppContext, filters: NostrFilter[], signal
|
||||||
return c.json([]);
|
return c.json([]);
|
||||||
}
|
}
|
||||||
|
|
||||||
const events = await eventsDB.filter(filters, { signal });
|
const events = await eventsDB.query(filters, { signal });
|
||||||
const pubkeys = new Set(events.map(({ pubkey }) => pubkey));
|
const pubkeys = new Set(events.map(({ pubkey }) => pubkey));
|
||||||
|
|
||||||
if (!pubkeys.size) {
|
if (!pubkeys.size) {
|
||||||
return c.json([]);
|
return c.json([]);
|
||||||
}
|
}
|
||||||
|
|
||||||
const authors = await eventsDB.filter(
|
const authors = await eventsDB.query(
|
||||||
[{ kinds: [0], authors: [...pubkeys], relations: ['author_stats'] }],
|
[{ kinds: [0], authors: [...pubkeys], relations: ['author_stats'] }],
|
||||||
{ signal },
|
{ signal },
|
||||||
);
|
);
|
||||||
|
@ -33,7 +33,7 @@ async function renderEventAccounts(c: AppContext, filters: NostrFilter[], signal
|
||||||
async function renderAccounts(c: AppContext, authors: string[], signal = AbortSignal.timeout(1000)) {
|
async function renderAccounts(c: AppContext, authors: string[], signal = AbortSignal.timeout(1000)) {
|
||||||
const { since, until, limit } = paginationSchema.parse(c.req.query());
|
const { since, until, limit } = paginationSchema.parse(c.req.query());
|
||||||
|
|
||||||
const events = await eventsDB.filter(
|
const events = await eventsDB.query(
|
||||||
[{ kinds: [0], authors, relations: ['author_stats'], since, until, limit }],
|
[{ kinds: [0], authors, relations: ['author_stats'], since, until, limit }],
|
||||||
{ signal },
|
{ signal },
|
||||||
);
|
);
|
||||||
|
@ -53,7 +53,7 @@ async function renderStatuses(c: AppContext, ids: string[], signal = AbortSignal
|
||||||
|
|
||||||
const { limit } = paginationSchema.parse(c.req.query());
|
const { limit } = paginationSchema.parse(c.req.query());
|
||||||
|
|
||||||
const events = await eventsDB.filter(
|
const events = await eventsDB.query(
|
||||||
[{ kinds: [1], ids, relations: ['author', 'event_stats', 'author_stats'], limit }],
|
[{ kinds: [1], ids, relations: ['author', 'event_stats', 'author_stats'], limit }],
|
||||||
{ signal },
|
{ signal },
|
||||||
);
|
);
|
||||||
|
|
|
@ -2,7 +2,7 @@ import { eventsDB } from '@/storages.ts';
|
||||||
import { hasTag } from '@/tags.ts';
|
import { hasTag } from '@/tags.ts';
|
||||||
|
|
||||||
async function renderRelationship(sourcePubkey: string, targetPubkey: string) {
|
async function renderRelationship(sourcePubkey: string, targetPubkey: string) {
|
||||||
const events = await eventsDB.filter([
|
const events = await eventsDB.query([
|
||||||
{ kinds: [3], authors: [sourcePubkey], limit: 1 },
|
{ kinds: [3], authors: [sourcePubkey], limit: 1 },
|
||||||
{ kinds: [3], authors: [targetPubkey], limit: 1 },
|
{ kinds: [3], authors: [targetPubkey], limit: 1 },
|
||||||
{ kinds: [10000], authors: [sourcePubkey], limit: 1 },
|
{ kinds: [10000], authors: [sourcePubkey], limit: 1 },
|
||||||
|
|
|
@ -36,7 +36,7 @@ async function renderStatus(event: DittoEvent, viewerPubkey?: string) {
|
||||||
Promise.all(mentionedPubkeys.map(toMention)),
|
Promise.all(mentionedPubkeys.map(toMention)),
|
||||||
firstUrl ? unfurlCardCached(firstUrl) : null,
|
firstUrl ? unfurlCardCached(firstUrl) : null,
|
||||||
viewerPubkey
|
viewerPubkey
|
||||||
? await eventsDB.filter([
|
? await eventsDB.query([
|
||||||
{ kinds: [6], '#e': [event.id], authors: [viewerPubkey], limit: 1 },
|
{ kinds: [6], '#e': [event.id], authors: [viewerPubkey], limit: 1 },
|
||||||
{ kinds: [7], '#e': [event.id], authors: [viewerPubkey], limit: 1 },
|
{ kinds: [7], '#e': [event.id], authors: [viewerPubkey], limit: 1 },
|
||||||
{ kinds: [9734], '#e': [event.id], authors: [viewerPubkey], limit: 1 },
|
{ kinds: [9734], '#e': [event.id], authors: [viewerPubkey], limit: 1 },
|
||||||
|
|
Loading…
Reference in New Issue