createPersistenceAdapter
ts
import { createPersistenceAdapter } from '@signaldb/core'
While SignalDB comes with a few built-in Persistence Adapters, there may be scenarios where you need to create a custom one to cater to specific requirements.
You can create a custom persistene adapter by calling createPersistenceAdapter
and supplying a PersistenceAdapter
compatible object as follows:
ts
interface Changeset<T> {
added: T[],
modified: T[],
removed: T[],
}
// contains either items or changes (but not both)
type LoadResponse<T> =
{ items: T[], changes?: never }
| { items?: never, changes: Changeset<T> }
interface PersistenceAdapter<T> {
register(onChange: (data?: LoadResponse<T>) => void | Promise<void>): Promise<void>,
load(): Promise<LoadResponse<T>>,
save(items: T[], changes: Changeset<T>): Promise<void>,
unregister?(): Promise<void>,
}
- register is called when initializing the collection. The
onChange
function should be called when data in the adapter was updated externally so the collection can update its internal memory. You can optionally directly pass aLoadResponse<T>
object returned from theload
function to make the implementation of your adapter more straightforward. - load is called to load data from the adapter and should return a
LoadResponse<T>
which includes either anitems
property containing all of the items, or achangeset
property containing only the changes. The collection will update its internal memory by either replacing all of its items, or applying the changeset to make differential changes, respectively. - save is called when data was updated, and should save the data. Both
items
andchanges
are provided so you can chose which one you'd like to use. - unregister? (optional) is called when the
dispose
method of the collection is called. Allows you to clean up things.
Here is a short example how the File system persistence adapter is implemented:
js
import fs from 'fs'
import { createPersistenceAdapter } from '@signaldb/core'
export default function createFilesystemAdapter(filename: string) {
return createPersistenceAdapter({
async register(onChange) {
const exists = await fs.promises.access(filename).then(() => true).catch(() => false)
if (!exists) await fs.promises.writeFile(filename, '[]')
fs.watch(filename, { encoding: 'utf8' }, () => {
void onChange()
})
},
async load() {
const exists = await fs.promises.access(filename).then(() => true).catch(() => false)
if (!exists) return { items: [] }
const contents = await fs.promises.readFile(filename, 'utf8')
const items = JSON.parse(contents)
return { items }
},
async save(items) {
await fs.promises.writeFile(filename, JSON.stringify(items))
},
})
}