Simplify storage adapters
This commit is contained in:
+8
-188
@@ -2,7 +2,7 @@ import {openDB, deleteDB} from "idb"
|
||||
import {IDBPDatabase} from "idb"
|
||||
import {writable} from "svelte/store"
|
||||
import {Unsubscriber, Writable} from "svelte/store"
|
||||
import {indexBy, equals, throttle, fromPairs} from "@welshman/lib"
|
||||
import {indexBy, call, equals, throttle, fromPairs} from "@welshman/lib"
|
||||
import {TrustedEvent} from "@welshman/util"
|
||||
import {Repository} from "@welshman/relay"
|
||||
import {Tracker} from "@welshman/net"
|
||||
@@ -15,8 +15,8 @@ export type StorageAdapterOptions = {
|
||||
|
||||
export type StorageAdapter = {
|
||||
keyPath: string
|
||||
store: Writable<any[]>
|
||||
options: StorageAdapterOptions
|
||||
init: () => Promise<void>
|
||||
sync: () => Unsubscriber
|
||||
}
|
||||
|
||||
export let db: IDBPDatabase | undefined
|
||||
@@ -60,41 +60,6 @@ export const bulkDelete = async (name: string, ids: string[]) => {
|
||||
await tx.done
|
||||
}
|
||||
|
||||
export const initIndexedDbAdapter = async (name: string, adapter: StorageAdapter) => {
|
||||
let prevRecords = await getAll(name)
|
||||
|
||||
adapter.store.set(prevRecords)
|
||||
|
||||
setTimeout(() => {
|
||||
adapter.store.subscribe(async (currentRecords: any[]) => {
|
||||
if (dead.get()) {
|
||||
return
|
||||
}
|
||||
|
||||
const currentIds = new Set(currentRecords.map(item => item[adapter.keyPath]))
|
||||
const removedRecords = prevRecords.filter(r => !currentIds.has(r[adapter.keyPath]))
|
||||
|
||||
const prevRecordsById = indexBy(item => item[adapter.keyPath], prevRecords)
|
||||
const updatedRecords = currentRecords.filter(
|
||||
r => !equals(r, prevRecordsById.get(r[adapter.keyPath])),
|
||||
)
|
||||
|
||||
prevRecords = currentRecords
|
||||
|
||||
if (updatedRecords.length > 0) {
|
||||
await bulkPut(name, updatedRecords)
|
||||
}
|
||||
|
||||
if (removedRecords.length > 0) {
|
||||
await bulkDelete(
|
||||
name,
|
||||
removedRecords.map(item => item[adapter.keyPath]),
|
||||
)
|
||||
}
|
||||
})
|
||||
}, adapter.options.throttle || 0)
|
||||
}
|
||||
|
||||
export const initStorage = async (
|
||||
name: string,
|
||||
version: number,
|
||||
@@ -128,9 +93,11 @@ export const initStorage = async (
|
||||
},
|
||||
})
|
||||
|
||||
await Promise.all(
|
||||
Object.entries(adapters).map(([name, config]) => initIndexedDbAdapter(name, config)),
|
||||
)
|
||||
await Promise.all(Object.values(adapters).map(adapter => adapter.init()))
|
||||
|
||||
const unsubscribers = Object.values(adapters).map(adapter => adapter.sync())
|
||||
|
||||
return () => unsubscribers.forEach(call)
|
||||
}
|
||||
|
||||
export const closeStorage = async () => {
|
||||
@@ -146,150 +113,3 @@ export const clearStorage = async () => {
|
||||
db = undefined // force initStorage to run again in tests
|
||||
}
|
||||
}
|
||||
|
||||
const migrate = (data: any[], options: StorageAdapterOptions) =>
|
||||
options.migrate ? options.migrate(data) : data
|
||||
|
||||
export const storageAdapters = {
|
||||
fromCollectionStore: <T>(
|
||||
keyPath: string,
|
||||
store: Writable<T[]>,
|
||||
options: StorageAdapterOptions = {},
|
||||
) => ({
|
||||
options,
|
||||
keyPath,
|
||||
store: throttled(options.throttle || 0, store),
|
||||
}),
|
||||
fromObjectStore: <T>(
|
||||
store: Writable<Record<string, T>>,
|
||||
options: StorageAdapterOptions = {},
|
||||
) => ({
|
||||
options,
|
||||
keyPath: "key",
|
||||
store: adapter({
|
||||
store: throttled(options.throttle || 0, store),
|
||||
forward: (data: Record<string, T>) =>
|
||||
migrate(
|
||||
Object.entries(data).map(([key, value]) => ({key, value})),
|
||||
options,
|
||||
),
|
||||
backward: (data: {key: string; value: T}[]) =>
|
||||
fromPairs(data.map(({key, value}) => [key, value])),
|
||||
}),
|
||||
}),
|
||||
fromMapStore: <T>(store: Writable<Map<string, T>>, options: StorageAdapterOptions = {}) => ({
|
||||
options,
|
||||
keyPath: "key",
|
||||
store: adapter({
|
||||
store: throttled(options.throttle || 0, store),
|
||||
forward: (data: Map<string, T>) =>
|
||||
migrate(
|
||||
Array.from(data.entries()).map(([key, value]) => ({key, value})),
|
||||
options,
|
||||
),
|
||||
backward: (data: {key: string; value: T}[]) =>
|
||||
new Map(data.map(({key, value}) => [key, value])),
|
||||
}),
|
||||
}),
|
||||
fromTracker: (tracker: Tracker, options: StorageAdapterOptions = {}) => ({
|
||||
options,
|
||||
keyPath: "key",
|
||||
store: custom(
|
||||
setter => {
|
||||
let onUpdate = () =>
|
||||
setter(
|
||||
migrate(
|
||||
Array.from(tracker.relaysById.entries()).map(([key, urls]) => ({
|
||||
key,
|
||||
value: Array.from(urls),
|
||||
})),
|
||||
options,
|
||||
),
|
||||
)
|
||||
|
||||
if (options.throttle) {
|
||||
onUpdate = throttle(options.throttle, onUpdate)
|
||||
}
|
||||
|
||||
onUpdate()
|
||||
tracker.on("update", onUpdate)
|
||||
|
||||
return () => tracker.off("update", onUpdate)
|
||||
},
|
||||
{
|
||||
set: (data: {key: string; value: string[]}[]) =>
|
||||
tracker.load(new Map(data.map(({key, value}) => [key, new Set(value)]))),
|
||||
},
|
||||
),
|
||||
}),
|
||||
fromRepository: (repository: Repository, options: StorageAdapterOptions = {}) => ({
|
||||
options,
|
||||
keyPath: "id",
|
||||
store: custom(
|
||||
setter => {
|
||||
let onUpdate = () => setter(migrate(repository.dump(), options))
|
||||
|
||||
if (options.throttle) {
|
||||
onUpdate = throttle(options.throttle, onUpdate)
|
||||
}
|
||||
|
||||
onUpdate()
|
||||
repository.on("update", onUpdate)
|
||||
|
||||
return () => repository.off("update", onUpdate)
|
||||
},
|
||||
{
|
||||
set: (events: TrustedEvent[]) => repository.load(events),
|
||||
},
|
||||
),
|
||||
}),
|
||||
fromRepositoryAndTracker: (
|
||||
repository: Repository,
|
||||
tracker: Tracker,
|
||||
options: StorageAdapterOptions = {},
|
||||
) => ({
|
||||
options,
|
||||
keyPath: "id",
|
||||
store: custom(
|
||||
setter => {
|
||||
let onUpdate = () => {
|
||||
const events = migrate(repository.dump(), options)
|
||||
|
||||
setter(
|
||||
events.map(event => {
|
||||
const relays = Array.from(tracker.getRelays(event.id))
|
||||
|
||||
return {id: event.id, event, relays}
|
||||
}),
|
||||
)
|
||||
}
|
||||
|
||||
if (options.throttle) {
|
||||
onUpdate = throttle(options.throttle, onUpdate)
|
||||
}
|
||||
|
||||
onUpdate()
|
||||
tracker.on("update", onUpdate)
|
||||
repository.on("update", onUpdate)
|
||||
|
||||
return () => {
|
||||
tracker.off("update", onUpdate)
|
||||
}
|
||||
},
|
||||
{
|
||||
set: (items: {event: TrustedEvent; relays: string[]}[]) => {
|
||||
const events: TrustedEvent[] = []
|
||||
const relaysById = new Map<string, Set<string>>()
|
||||
|
||||
for (const {event, relays} of items) {
|
||||
events.push(event)
|
||||
relaysById.set(event.id, new Set(relays))
|
||||
}
|
||||
|
||||
repository.load(events)
|
||||
tracker.load(relaysById)
|
||||
},
|
||||
},
|
||||
),
|
||||
}),
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user