Memoize some stores, improve room support

This commit is contained in:
Jon Staab
2025-11-03 13:20:53 -08:00
parent 3132b8c59a
commit 8426febcdf
6 changed files with 91 additions and 30 deletions
+8 -6
View File
@@ -1,5 +1,5 @@
import {derived, Readable} from "svelte/store"
import {withGetter} from "@welshman/store"
import {withGetter, memoized} from "@welshman/store"
import {pubkey} from "./session.js"
import {profilesByPubkey, loadProfile} from "./profiles.js"
import {followsByPubkey, loadFollows} from "./follows.js"
@@ -19,13 +19,15 @@ export type MakeUserDataOptions<T> = {
export const makeUserData = <T>({mapStore, loadItem}: MakeUserDataOptions<T>) =>
withGetter(
derived([mapStore, pubkey], ([$mapStore, $pubkey]) => {
if (!$pubkey) return undefined
memoized(
derived([mapStore, pubkey], ([$mapStore, $pubkey]) => {
if (!$pubkey) return undefined
loadItem($pubkey)
loadItem($pubkey)
return $mapStore.get($pubkey)
}),
return $mapStore.get($pubkey)
}),
),
)
export const makeUserLoader =
+3 -3
View File
@@ -1030,11 +1030,11 @@ export const call = <T>(f: () => T, ...args: unknown[]) => f()
* @param f - Function to memoize
* @returns Memoized function
*/
export const memoize = <T>(f: (...args: any[]) => T) => {
let prevArgs: any[]
export const memoize = <T, Args extends any[]>(f: (...args: Args) => T) => {
let prevArgs: Args
let result: T
return (...args: any[]) => {
return (...args: Args): T => {
if (!equals(prevArgs, args)) {
prevArgs = args
result = f(...args)
+2 -1
View File
@@ -1,6 +1,7 @@
import {readable, derived, writable, Readable, Subscriber} from "svelte/store"
import {batch, indexBy, remove, assoc, now} from "@welshman/lib"
import {withGetter, ReadableWithGetter} from "./getter.js"
import {memoized} from "./memoize.js"
// Collection utility
@@ -126,7 +127,7 @@ export const collection = <T>({name, store, getKey, load}: CollectionOptions<T>)
// store will update when it arrives
loadItem(key, relays)
return derived(indexStore, $index => $index.get(key))
return memoized<T | undefined>(derived(indexStore, $index => $index.get(key)))
}
const onItem = (cb: Subscriber<T>) => {
+1
View File
@@ -1,6 +1,7 @@
export * from "./synced.js"
export * from "./getter.js"
export * from "./throttle.js"
export * from "./memoize.js"
export * from "./custom.js"
export * from "./repository.js"
export * from "./collection.js"
+8
View File
@@ -0,0 +1,8 @@
import {Readable, Subscriber} from "svelte/store"
import {memoize} from "@welshman/lib"
export const memoized = <T>(store: Readable<T>) => {
const {subscribe} = store
return {...store, subscribe: (f: Subscriber<T>) => subscribe(memoize(f))}
}
+69 -20
View File
@@ -1,4 +1,4 @@
import {randomId, nthNe} from "@welshman/lib"
import {randomId, spec} from "@welshman/lib"
import {
ROOM_META,
ROOM_DELETE,
@@ -8,10 +8,18 @@ import {
ROOM_LEAVE,
} from "./Kinds.js"
import {makeEvent, TrustedEvent, getIdentifier} from "./Events.js"
import {getTag, getTagValue} from "./Tags.js"
export type RoomMeta = {
id: string
tags: string[][]
h: string
name?: string
about?: string
picture?: string
pictureMeta?: string[]
isClosed?: boolean
isHidden?: boolean
isPrivate?: boolean
isRestricted?: boolean
event?: TrustedEvent
}
@@ -19,38 +27,79 @@ export type PublishedRoomMeta = Omit<RoomMeta, "event"> & {
event: TrustedEvent
}
export const makeRoomMeta = (room: Partial<RoomMeta> = {}): RoomMeta => ({
id: randomId(),
tags: [],
...room,
})
export const makeRoomMeta = (room: Partial<RoomMeta> = {}): RoomMeta => {
return {
h: randomId(),
...room,
}
}
export const readRoomMeta = (event: TrustedEvent): PublishedRoomMeta => {
if (event.kind !== ROOM_META) {
throw new Error("Invalid group meta event")
}
const id = getIdentifier(event)
const h = getIdentifier(event)
if (!id) {
if (!h) {
throw new Error("Group meta event had no d tag")
}
const tags = event.tags.filter(nthNe(0, "d"))
return {id, tags, event}
return {
h,
event,
name: getTagValue("name", event.tags),
about: getTagValue("about", event.tags),
picture: getTagValue("picture", event.tags),
pictureMeta: getTag("picture", event.tags)?.slice(2),
isClosed: event.tags.some(spec(["closed"])),
isHidden: event.tags.some(spec(["hidden"])),
isPrivate: event.tags.some(spec(["private"])),
isRestricted: event.tags.some(spec(["restricted"])),
}
}
export const makeRoomCreateEvent = (room: RoomMeta) =>
makeEvent(ROOM_CREATE, {tags: [["h", room.id]]})
makeEvent(ROOM_CREATE, {tags: [["h", room.h]]})
export const makeRoomDeleteEvent = (room: RoomMeta) =>
makeEvent(ROOM_DELETE, {tags: [["h", room.id]]})
makeEvent(ROOM_DELETE, {tags: [["h", room.h]]})
export const makeRoomEditEvent = (room: RoomMeta) =>
makeEvent(ROOM_EDIT_META, {tags: [["h", room.id], ...room.tags]})
export const makeRoomEditEvent = (room: RoomMeta) => {
const tags = [["h", room.h]]
export const makeRoomJoinEvent = (room: RoomMeta) => makeEvent(ROOM_JOIN, {tags: [["h", room.id]]})
if (room.name) tags.push(["name", room.name])
if (room.about) tags.push(["about", room.about])
export const makeRoomLeaveEvent = (room: RoomMeta) =>
makeEvent(ROOM_LEAVE, {tags: [["h", room.id]]})
if (room.picture) {
const tag = ["picture", room.picture]
if (room.pictureMeta) {
tag.push(...room.pictureMeta)
}
tags.push(tag)
}
if (room.isClosed) tags.push(["closed"])
if (room.isHidden) tags.push(["hidden"])
if (room.isPrivate) tags.push(["private"])
if (room.isRestricted) tags.push(["restricted"])
if (room.event) {
for (const t of room.event.tags) {
if (tags.some(spec(t.slice(0, 1)))) continue
if (["closed", "hidden", "private", "restricted"].includes(t[0])) continue
tags.push(t)
}
}
console.log(room, tags)
return makeEvent(ROOM_EDIT_META, {tags})
}
export const makeRoomJoinEvent = (room: RoomMeta) => makeEvent(ROOM_JOIN, {tags: [["h", room.h]]})
export const makeRoomLeaveEvent = (room: RoomMeta) => makeEvent(ROOM_LEAVE, {tags: [["h", room.h]]})