From c21b9baef408d4c1b93ce4d43acb4fafc07867e8 Mon Sep 17 00:00:00 2001 From: sliterok <12751644+sliterok@users.noreply.github.com> Date: Wed, 13 Mar 2024 13:13:51 +0100 Subject: [PATCH] write cache --- src/db/index.ts | 104 +++++++++++++++++++++++++++++----------- src/entry-client.ts | 22 --------- src/routes/UserGrid.tsx | 4 +- 3 files changed, 77 insertions(+), 53 deletions(-) delete mode 100644 src/entry-client.ts diff --git a/src/db/index.ts b/src/db/index.ts index 0709492..4822af3 100644 --- a/src/db/index.ts +++ b/src/db/index.ts @@ -19,11 +19,28 @@ import { IQueryOptions, IImportInput, } from './types' -import { batchReduce, mergeUint8Arrays } from './helpers' import deepmerge from 'deepmerge' +import { batchReduce, mergeUint8Arrays } from './helpers' + +const pendingWrites = new Set() +const promiseMap = new Map>() +const timeoutMap = new Map>() +const dataMap = new Map() + +const getPendingWritePromises = (): Promise[] => { + const promises = Array(pendingWrites.size) + let i = 0 + for (const fileName of pendingWrites.keys()) { + promises[i++] = promiseMap.get(fileName) + } + return promises +} const readFile = async (dir: FileSystemDirectoryHandle, fileName: string, encoder?: IEncoder | false) => { try { + if (dataMap.has(fileName)) { + return dataMap.get(fileName) + } const fileHandle = await dir.getFileHandle(fileName) const accessHandle = await fileHandle.createSyncAccessHandle() const uintArray = new Uint8Array(new ArrayBuffer(accessHandle.getSize())) @@ -46,18 +63,37 @@ const readFile = async (dir: FileSystemDirectoryHandle, fileName: string, encode } } -const writeFile = async (dir: FileSystemDirectoryHandle, fileName: string, data: Record | Uint8Array, encoder?: IEncoder | false) => { - const fileHandle = await dir.getFileHandle(fileName, { create: true }) - const writable = await fileHandle.createWritable() +const writeFile = async (dir: FileSystemDirectoryHandle, fileName: string, data: Record, encoder?: IEncoder) => { + if (timeoutMap.has(fileName)) { + const timeout = timeoutMap.get(fileName)! + clearTimeout(timeout) + } - let encoded: Uint8Array + const promise = new Promise(res => { + const timeout = setTimeout(async () => { + const fileHandle = await dir.getFileHandle(fileName, { create: true }) + const writable = await fileHandle.createWritable() - if (encoder === false) encoded = new Uint8Array(data as Uint8Array) - else if (encoder) encoded = encoder.encode(data) - else encoded = encode(data) + let encoded: Uint8Array - await writable.write(encoded) - await writable.close() + if (encoder) encoded = encoder.encode(data) + else encoded = encode(data) + + await writable.write(encoded) + await writable.close() + + promiseMap.delete(fileName) + timeoutMap.delete(fileName) + pendingWrites.delete(fileName) + dataMap.delete(fileName) + res() + }, 10) + timeoutMap.set(fileName, timeout) + }) + + pendingWrites.add(fileName) + promiseMap.set(fileName, promise) + dataMap.set(fileName, data) } export class FileStoreStrategy extends SerializeStrategy { @@ -166,7 +202,7 @@ export class OPFSDB { const records = await this.readMany(indexArray) const responsesLoaded = performance.now() // eslint-disable-next-line no-console - console.log('indexes:', indexesFinish - start, 'records:', responsesLoaded - indexesFinish) + console.log(`indexes: ${Math.floor(indexesFinish - start)}ms, records: ${Math.floor(responsesLoaded - indexesFinish)}ms`) return records } @@ -191,22 +227,30 @@ export class OPFSDB { // } async readMany(ids: string[]): Promise { - const result = batchReduce(ids, 20).map(async ids => { + const promises = batchReduce(ids, 200).map(async ids => { let size = 0 - const records = await Promise.all( - ids.map(async id => { - const file: Uint8Array = await readFile(this.recordsRoot, id, false) - size += file.length - return file - }) - ) - const merged = mergeUint8Arrays(size, ...records) + const unEncodedData: T[] = [] + const binaryData = ( + await Promise.all( + ids.map(async id => { + const file = await readFile(this.recordsRoot, id, false) + if (file instanceof Uint8Array) { + size += file.length + return file + } else { + unEncodedData.push(file) + } + }) + ) + ).filter(file => file) as Uint8Array[] + const merged = mergeUint8Arrays(size, ...binaryData) const decoded = this.encoder.decodeMultiple(merged) as T[] - return decoded + return [...unEncodedData, ...decoded] }) - const response = await Promise.all(result) - // const rawRecords = await Promise.all() - return response.flat() + // const promises = ids.map(id => readFile(this.recordsRoot, id, this.encoder)) + const result = await Promise.all(promises) + + return result.flat() } async read(id: string): Promise { @@ -235,6 +279,7 @@ export class OPFSDB { // } // })(), ]) + await Promise.all(getPendingWritePromises()) } async insert(id: string, value: T, fullRecord?: boolean) { @@ -266,6 +311,8 @@ export class OPFSDB { if ((updated || added) && !deleted) await tree.insert(id, newValue) } } + + await Promise.all(getPendingWritePromises()) } async delete(id: string, oldRecord?: T) { @@ -321,6 +368,7 @@ export const dropCommand = ({ tableName }: ICommandInput): Promise(command: ICommandInputs) => { // try { let response: T[] | string[] + const start = performance.now() switch (command.name) { case 'createTable': await createTableCommand(command as ICreateTableInput) @@ -347,9 +395,7 @@ export const command = async (command: ICommandInputs throw new Error('unknown command') } - return response! //new Response(JSON.stringify(response! || {}), { status: 200 }) - // } catch (error) { - // console.error(command.name, error) - // return new Response(null, { status: 500, statusText: (error as Error).message }) - // } + // eslint-disable-next-line no-console + console.log(`${command.name} cmd took: ${Math.round(performance.now() - start)}ms`) + return response! } diff --git a/src/entry-client.ts b/src/entry-client.ts deleted file mode 100644 index a10a3b7..0000000 --- a/src/entry-client.ts +++ /dev/null @@ -1,22 +0,0 @@ -import { IFetchDb } from './db/types' -import { startClient } from 'rakkasjs/client' - -declare module 'rakkasjs' { - interface PageContext { - dbFetch: IFetchDb - } -} - -startClient({ - hooks: { - beforeStart() { - // Do something before starting the client - }, - extendPageContext(ctx) { - ctx.dbFetch = async (url, body) => { - const response = await ctx.fetch(url, { body: JSON.stringify(body), method: 'POST' }) - return response.json() - } - }, - }, -}) diff --git a/src/routes/UserGrid.tsx b/src/routes/UserGrid.tsx index 1e0cd1f..187c19c 100644 --- a/src/routes/UserGrid.tsx +++ b/src/routes/UserGrid.tsx @@ -87,14 +87,14 @@ export default function MainLayout() { const records = Array(1000) .fill(true) .map(() => generateUser()) - const start = performance.now() + await sendCommand, IUser>({ name: 'import', tableName: 'users', records, }) // eslint-disable-next-line no-console - console.log('uploading users', (i + 1) * 1000, 'of 10000, time took:', performance.now() - start) + console.log('uploading users', (i + 1) * 1000, 'of 10000') } })