Add since to reqs that get re-opened to avoid re-downloading and gaps in active subscriptions. Also add some getters for event stores and event sort utils

This commit is contained in:
Jon Staab
2025-12-04 15:16:56 -08:00
parent 7b96e52349
commit d81ec62d2b
5 changed files with 126 additions and 75 deletions
-1
View File
@@ -207,7 +207,6 @@ export const round = (precision: number, x: number) =>
// ----------------------------------------------------------------------------
/** One minute in seconds */
export const MINUTE = 60
/** One hour in seconds */
+1 -1
View File
@@ -368,7 +368,7 @@ describe("policy", () => {
socket.emit(SocketEvent.Status, SocketStatus.Closed)
// Advance past the reopen delay
await vi.advanceTimersByTimeAsync(30000)
await vi.advanceTimersByTimeAsync(31000)
// Should resend the pending event
expect(sendSpy).toHaveBeenCalledWith(event)
+22 -4
View File
@@ -1,5 +1,5 @@
import {on, ms, nthNe, always, call, sleep, ago, now} from "@welshman/lib"
import {RELAY_JOIN, StampedEvent, SignedEvent} from "@welshman/util"
import {on, ms, omit, nthNe, always, call, sleep, ago, now} from "@welshman/lib"
import {RELAY_JOIN, StampedEvent, SignedEvent, Filter} from "@welshman/util"
import {
ClientMessage,
isClientAuth,
@@ -142,11 +142,29 @@ export const socketPolicyCloseInactive = (socket: Socket) => {
// If the socket closed and we have no error, reopen it but don't flap
if (isClosed && pending.size) {
sleep(Math.max(0, ms(5 - (now() - lastOpen)))).then(() => {
const since = now()
const delay = Math.max(0, ms(5 - (now() - lastOpen)))
sleep(delay).then(() => {
socket.attemptToOpen()
for (const message of pending.values()) {
socket.send(message)
// Add since to avoid re-downloading stuff on reconnect. If limit=0, remove it to catch up
if (isClientReq(message) && delay > 0) {
const filters: Filter[] = []
for (let filter of message.slice(2) as Filter[]) {
if (filter.limit === 0) {
filter = omit(["limit"], filter)
}
filters.push({...filter, since})
}
socket.send([...message.slice(0, 2), ...filters])
} else {
socket.send(message)
}
}
})
}
+98 -68
View File
@@ -1,6 +1,13 @@
import {readable, Readable} from "svelte/store"
import {on, assoc, now, mapPop, Maybe, MaybeAsync, call, sortBy, first} from "@welshman/lib"
import {matchFilters, getIdFilters, Filter, TrustedEvent} from "@welshman/util"
import {
matchFilters,
getIdFilters,
Filter,
TrustedEvent,
sortEventsAsc,
sortEventsDesc,
} from "@welshman/util"
import {Repository, RepositoryUpdate, Tracker} from "@welshman/net"
import {deriveDeduplicated} from "./misc.js"
@@ -8,23 +15,25 @@ import {deriveDeduplicated} from "./misc.js"
export type EventsById = Map<string, TrustedEvent>
export type DeriveEventsByIdOptions = {
export type EventsByIdOptions = {
filters: Filter[]
repository: Repository
includeDeleted?: boolean
}
export const deriveEventsById = ({
filters,
repository,
includeDeleted,
}: DeriveEventsByIdOptions) => {
export const getEventsById = ({filters, repository, includeDeleted}: EventsByIdOptions) => {
const eventsById = new Map<string, TrustedEvent>()
return readable(eventsById, set => {
for (const event of repository.query(filters, {includeDeleted})) {
eventsById.set(event.id, event)
}
for (const event of repository.query(filters, {includeDeleted})) {
eventsById.set(event.id, event)
}
return eventsById
}
export const deriveEventsById = ({filters, repository, includeDeleted}: EventsByIdOptions) =>
readable<EventsById>(new Map(), set => {
const eventsById = getEventsById({filters, repository, includeDeleted})
set(eventsById)
@@ -49,28 +58,23 @@ export const deriveEventsById = ({
}
})
})
}
export const deriveArray = <T>(itemsByIdStore: Readable<Map<string, T>>) =>
deriveDeduplicated(itemsByIdStore, itemsById => Array.from(itemsById.values()))
export const deriveEventsAsc = (eventsByIdStore: Readable<EventsById>) =>
deriveDeduplicated(eventsByIdStore, eventsById => sortBy(e => e.created_at, eventsById.values()))
deriveDeduplicated(eventsByIdStore, eventsById => sortEventsAsc(eventsById.values()))
export const deriveEventsDesc = (eventsByIdStore: Readable<EventsById>) =>
deriveDeduplicated(eventsByIdStore, eventsById => sortBy(e => -e.created_at, eventsById.values()))
deriveDeduplicated(eventsByIdStore, eventsById => sortEventsDesc(eventsById.values()))
export type DeriveEventOptions = {
export type EventOptions = {
repository: Repository
includeDeleted?: boolean
onDerive?: (filters: Filter[], ...args: any[]) => void
}
export const makeDeriveEvent = ({
repository,
includeDeleted = false,
onDerive,
}: DeriveEventOptions) => {
export const makeDeriveEvent = ({repository, includeDeleted = false, onDerive}: EventOptions) => {
return (idOrAddress: string, ...args: any[]) => {
const filters = getIdFilters([idOrAddress])
@@ -103,58 +107,75 @@ export const makeDeriveEvent = ({
export type EventsByIdByUrl = Map<string, EventsById>
export type DeriveEventsByIdByUrlOptions = DeriveEventsByIdOptions & {
export type EventsByIdByUrlOptions = EventsByIdOptions & {
tracker: Tracker
}
export const getEventsByIdByUrl = ({
filters,
tracker,
repository,
includeDeleted,
}: EventsByIdByUrlOptions) => {
const eventsByIdByUrl: EventsByIdByUrl = new Map()
for (const event of repository.query(filters, {includeDeleted})) {
for (const url of tracker.getRelays(event.id)) {
let eventsById = eventsByIdByUrl.get(url)
if (!eventsById) {
eventsById = new Map()
eventsByIdByUrl.set(url, eventsById)
}
eventsById.set(event.id, event)
}
}
return eventsByIdByUrl
}
export const deriveEventsByIdByUrl = ({
filters,
tracker,
repository,
includeDeleted,
}: DeriveEventsByIdByUrlOptions) => {
const eventsByIdByUrl: EventsByIdByUrl = new Map()
}: EventsByIdByUrlOptions) =>
readable<EventsByIdByUrl>(new Map(), set => {
const eventsByIdByUrl = getEventsByIdByUrl({filters, tracker, repository, includeDeleted})
const addEvent = (url: string, event: TrustedEvent) => {
if (!matchFilters(filters, event)) return false
const addEvent = (url: string, event: TrustedEvent) => {
if (!matchFilters(filters, event)) return false
const eventsById = eventsByIdByUrl.get(url)
const eventsById = eventsByIdByUrl.get(url)
if (eventsById?.has(event.id)) return false
if (eventsById?.has(event.id)) return false
// Create a new map so we can detect which key changed
const newEventsById = new Map(eventsById)
// Create a new map so we can detect which key changed
const newEventsById = new Map(eventsById)
newEventsById.set(event.id, event)
eventsByIdByUrl.set(url, newEventsById)
return true
}
const removeEvent = (url: string, id: string) => {
const eventsById = eventsByIdByUrl.get(url)
if (eventsById?.has(id)) {
eventsById.delete(id)
if (eventsById.size === 0) {
eventsByIdByUrl.delete(url)
} else {
// Create a new map so we can detect which key changed
eventsByIdByUrl.set(url, new Map(eventsById))
}
newEventsById.set(event.id, event)
eventsByIdByUrl.set(url, newEventsById)
return true
}
return false
}
const removeEvent = (url: string, id: string) => {
const eventsById = eventsByIdByUrl.get(url)
return readable(eventsByIdByUrl, set => {
for (const event of repository.query(filters, {includeDeleted})) {
for (const url of tracker.getRelays(event.id)) {
addEvent(url, event)
if (eventsById?.has(id)) {
eventsById.delete(id)
if (eventsById.size === 0) {
eventsByIdByUrl.delete(url)
} else {
// Create a new map so we can detect which key changed
eventsByIdByUrl.set(url, new Map(eventsById))
}
return true
}
return false
}
set(eventsByIdByUrl)
@@ -211,33 +232,42 @@ export const deriveEventsByIdByUrl = ({
return () => unsubscribers.forEach(call)
})
}
export type DeriveEventsByIdForUrlOptions = DeriveEventsByIdOptions & {
export type EventsByIdForUrlOptions = EventsByIdOptions & {
url: string
tracker: Tracker
}
export const getEventsByIdForUrl = ({
url,
filters,
tracker,
repository,
includeDeleted,
}: EventsByIdForUrlOptions) => {
const initialIds = Array.from(tracker.getIds(url))
const initialFilters = filters.map(assoc("ids", initialIds))
const eventsById: EventsById = new Map()
for (const event of repository.query(initialFilters, {includeDeleted})) {
eventsById.set(event.id, event)
}
return eventsById
}
export const deriveEventsByIdForUrl = ({
url,
filters,
tracker,
repository,
includeDeleted,
}: DeriveEventsByIdForUrlOptions) => {
const eventsById: EventsById = new Map()
}: EventsByIdForUrlOptions) => {
let eventsById = getEventsByIdForUrl({url, filters, tracker, repository, includeDeleted})
return readable(eventsById, set => {
const reset = () => {
const initialIds = Array.from(tracker.getIds(url))
const initialFilters = filters.map(assoc("ids", initialIds))
eventsById.clear()
for (const event of repository.query(initialFilters, {includeDeleted})) {
eventsById.set(event.id, event)
}
eventsById = getEventsByIdForUrl({url, filters, tracker, repository, includeDeleted})
set(eventsById)
}
@@ -295,7 +325,7 @@ export type EventToItem<T> = (event: TrustedEvent) => MaybeAsync<Maybe<T>>
export type GetItem<T> = (key: string, ...args: any[]) => Maybe<T>
export type DeriveItemsByKeyOptions<T> = {
export type ItemsByKeyOptions<T> = {
getKey: (item: T) => string
filters: Filter[]
repository: Repository
@@ -309,7 +339,7 @@ export const deriveItemsByKey = <T>({
repository,
eventToItem,
includeDeleted,
}: DeriveItemsByKeyOptions<T>) => {
}: ItemsByKeyOptions<T>) => {
const deferred = new Map<string, Promise<Maybe<T>>>()
const itemsByKey = new Map<string, T>()
const idsByKey = new Map<string, string>()
+5 -1
View File
@@ -1,7 +1,7 @@
import {verifiedSymbol, verifyEvent as verifyEventPure} from "nostr-tools/pure"
import {setNostrWasm, verifyEvent as verifyEventWasm} from "nostr-tools/wasm"
import {initNostrWasm} from "nostr-wasm"
import {mapVals, lte, first, pick, now} from "@welshman/lib"
import {mapVals, sortBy, lte, first, pick, now} from "@welshman/lib"
import {getReplyTags, getCommentTags, getReplyTagValues, getCommentTagValues} from "./Tags.js"
import {getAddress, Address} from "./Address.js"
import {
@@ -195,3 +195,7 @@ export const isChildOf = (child: EventTemplate, parent: HashedEvent) => {
return getIdAndAddress(parent).some(x => idsAndAddrs.includes(x))
}
export const sortEventsAsc = (events: Iterable<TrustedEvent>) => sortBy(e => e.created_at, events)
export const sortEventsDesc = (events: Iterable<TrustedEvent>) => sortBy(e => -e.created_at, events)