@peerbit/any-store-opfs 2.0.2-0691c73

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,199 @@
1
+ import { type AnyStore, type MaybePromise } from "@peerbit/any-store-interface";
2
+ import * as memory from "@peerbit/any-store-interface/messages";
3
+ import { createWorker } from "./create.js"
4
+ import { v4 as uuid } from "uuid";
5
+ import { serialize, deserialize } from "@dao-xyz/borsh";
6
+ function memoryIterator(
7
+ client: {
8
+ request<T extends memory.MemoryRequest>(
9
+ request: memory.MemoryRequest
10
+ ): Promise<T>;
11
+ },
12
+ level: string[]
13
+ ): {
14
+ [Symbol.asyncIterator]: () => AsyncIterator<[string, Uint8Array], void, void>;
15
+ } {
16
+ return {
17
+ [Symbol.asyncIterator]() {
18
+ const iteratorId = uuid();
19
+ return {
20
+ next: async () => {
21
+ const resp = await client.request<memory.RESP_Iterator_Next>(
22
+ new memory.REQ_Iterator_Next({ id: iteratorId, level })
23
+ );
24
+ if (resp.keys.length > 1) {
25
+ throw new Error("Unsupported iteration response");
26
+ }
27
+ // Will only have 0 or 1 element for now
28
+ for (let i = 0; i < resp.keys.length; i++) {
29
+ return {
30
+ done: false,
31
+ value: [resp.keys[i], resp.values[i]] as [string, Uint8Array]
32
+ } as { done: false; value: [string, Uint8Array] };
33
+ }
34
+ return { done: true, value: undefined } as {
35
+ done: true;
36
+ value: undefined;
37
+ };
38
+ },
39
+ async return() {
40
+ await client.request<memory.RESP_Iterator_Next>(
41
+ new memory.REQ_Iterator_Stop({ id: iteratorId, level })
42
+ );
43
+ return { done: true, value: undefined } as {
44
+ done: true;
45
+ value: undefined;
46
+ };
47
+ }
48
+ };
49
+ }
50
+ };
51
+ }
52
+
53
+ export class OPFSStore implements AnyStore {
54
+ worker: Worker;
55
+ levelMap: Map<string, AnyStore>;
56
+ root: AnyStore;
57
+
58
+ private _responseCallbacks: Map<
59
+ string,
60
+ { fn: (message: memory.MemoryRequest) => any; once: boolean }
61
+ > = new Map();
62
+
63
+ private _createStorage: (level: string[]) => AnyStore;
64
+ constructor(readonly directory?: string) {
65
+ this.levelMap = new Map();
66
+ this._createStorage = (level: string[] = []): AnyStore => {
67
+ return {
68
+ clear: async () => {
69
+ await this.request<memory.RESP_Clear>(
70
+ new memory.REQ_Clear({ level })
71
+ );
72
+ },
73
+ del: async (key) => {
74
+ await this.request<memory.RESP_Del>(
75
+ new memory.REQ_Del({ level, key })
76
+ );
77
+ },
78
+ get: async (key) => {
79
+ return (
80
+ await this.request<memory.RESP_Get>(
81
+ new memory.REQ_Get({ level, key })
82
+ )
83
+ ).bytes;
84
+ },
85
+ put: async (key, value) => {
86
+ await this.request<memory.RESP_Put>(
87
+ new memory.REQ_Put({ level, key, bytes: value })
88
+ );
89
+ },
90
+ status: async () =>
91
+ (
92
+ await this.request<memory.RESP_Status>(
93
+ new memory.REQ_Status({ level })
94
+ )
95
+ ).status,
96
+ sublevel: async (name) => {
97
+ await this.request(new memory.REQ_Sublevel({ level, name }));
98
+ const newLevels = [...level, name];
99
+ const sublevel = this._createStorage(newLevels);
100
+ this.levelMap.set(memory.levelKey(newLevels), sublevel);
101
+ return sublevel;
102
+ },
103
+
104
+ iterator: () => memoryIterator(this, level),
105
+ close: async () => {
106
+ await this.request<memory.RESP_Close>(
107
+ new memory.REQ_Close({ level })
108
+ );
109
+ /* this.levelMap.delete(memory.levelKey(level)); */
110
+ },
111
+ open: async () => {
112
+ await this.request<memory.RESP_Open>(new memory.REQ_Open({ level }));
113
+ },
114
+
115
+ size: async () => {
116
+ const size = await this.request<memory.RESP_Size>(
117
+ new memory.REQ_Size({ level })
118
+ );
119
+ return size.size;
120
+ }
121
+ };
122
+ };
123
+ }
124
+ status() {
125
+ return this.worker ? this.root.status() : "closed";
126
+ }
127
+ async close(): Promise<void> {
128
+ this.worker.terminate();
129
+ this.worker = undefined!;
130
+ this._responseCallbacks.clear();
131
+ this.levelMap.clear();
132
+ }
133
+ async open(): Promise<void> {
134
+ if (!this.worker) {
135
+ /* if (
136
+ !(globalThis as any)["__playwright_test__"] &&
137
+ (await navigator.storage.persist()) === false
138
+ ) {
139
+ throw new Error("OPFS not allowed to persist data");
140
+ } */
141
+ this.worker = createWorker(this.directory);
142
+ this.root = this._createStorage([]);
143
+ this.worker.addEventListener("message", async (ev) => {
144
+ const message = deserialize(ev.data, memory.MemoryMessage);
145
+ this._responseCallbacks.get(message.messageId)!.fn(message);
146
+ });
147
+ await this.root.open();
148
+ }
149
+ }
150
+ get(key: string): MaybePromise<Uint8Array | undefined> {
151
+ return this.root.get(key);
152
+ }
153
+ put(key: string, value: Uint8Array) {
154
+ return this.root.put(key, value);
155
+ }
156
+ del(key: any): MaybePromise<void> {
157
+ return this.root.del(key);
158
+ }
159
+ sublevel(name: string): AnyStore | MaybePromise<AnyStore> {
160
+ return this.root.sublevel(name);
161
+ }
162
+ iterator(): {
163
+ [Symbol.asyncIterator]: () => AsyncIterator<
164
+ [string, Uint8Array],
165
+ void,
166
+ void
167
+ >;
168
+ } {
169
+ return this.root.iterator();
170
+ }
171
+ clear(): MaybePromise<void> {
172
+ return this.root.clear();
173
+ }
174
+
175
+ size(): MaybePromise<number> {
176
+ return this.root.size();
177
+ }
178
+
179
+ async request<T extends memory.MemoryRequest>(
180
+ request: memory.MemoryRequest
181
+ ): Promise<T> {
182
+ return new Promise<T>((resolve, reject) => {
183
+ const onResponse = (message: memory.MemoryRequest) => {
184
+ this._responseCallbacks.delete(request.messageId);
185
+ if (message instanceof memory.RESP_Error) {
186
+ reject(new Error(message.error));
187
+ } else {
188
+ resolve(message as T);
189
+ }
190
+ };
191
+ this._responseCallbacks.set(request.messageId, {
192
+ fn: onResponse,
193
+ once: true
194
+ });
195
+ const bytes = serialize(request);
196
+ this.worker.postMessage(bytes, [bytes.buffer]);
197
+ });
198
+ }
199
+ }
package/src/worker.ts ADDED
@@ -0,0 +1,350 @@
1
+ // TODO make bundle by removing unecessary dependencies
2
+ import { deserialize, serialize } from "@dao-xyz/borsh";
3
+ import { type AnyStore } from "@peerbit/any-store-interface"
4
+ import * as memory from "@peerbit/any-store-interface/messages";
5
+ import { fromBase64URL, toBase64URL, ready } from "@peerbit/crypto";
6
+ import { BinaryReader, BinaryWriter } from "@dao-xyz/borsh";
7
+ import { waitForResolved } from "@peerbit/time";
8
+
9
+ const directory = location.hash.split("#")?.[1];
10
+
11
+ const encodeName = (name: string): string => {
12
+ // since "/" and perhaps other characters might not be allowed we do encode
13
+ const writer = new BinaryWriter();
14
+ writer.string(name);
15
+ return toBase64URL(writer.finalize());
16
+ };
17
+
18
+ const decodeName = (name: string): string => {
19
+ // since "/" and perhaps other characters might not be allowed we do encode
20
+ const writer = new BinaryReader(fromBase64URL(name));
21
+ return writer.string();
22
+ };
23
+
24
+ const waitForSyncAcccess = async (
25
+ fileHandle: FileSystemFileHandle
26
+ ): Promise<FileSystemSyncAccessHandle> => {
27
+ try {
28
+ const handle = await fileHandle.createSyncAccessHandle();
29
+ return handle;
30
+ } catch (error) {
31
+ const handle = await waitForResolved(() =>
32
+ fileHandle.createSyncAccessHandle()
33
+ );
34
+ if (!handle) {
35
+ throw error;
36
+ }
37
+ return handle;
38
+ }
39
+ };
40
+
41
+ const createWriteHandle = async (fileHandle: FileSystemFileHandle) => {
42
+ // In Chrome on GET DOMException: Failed to execute 'createWritable' on 'FileSystemFileHandle': Failed to create swap file
43
+ // hence below is not used for now
44
+ /*
45
+ if (fileHandle.createWritable != null) {
46
+ return fileHandle.createWritable({ keepExistingData: false });
47
+ } */
48
+ return waitForSyncAcccess(fileHandle);
49
+ };
50
+
51
+ export class OPFSStoreWorker {
52
+ private _levels: Map<string, AnyStore>;
53
+ private _rootStore: AnyStore;
54
+
55
+ private _memoryIterator: Map<
56
+ string,
57
+ AsyncIterator<[string, ArrayBuffer], void, void>
58
+ >;
59
+
60
+ constructor() {
61
+ const postMessageFn = postMessage;
62
+ this._memoryIterator = new Map();
63
+ this._levels = new Map();
64
+
65
+ const createMemory = (
66
+ root?: FileSystemDirectoryHandle,
67
+ levels: string[] = []
68
+ ): AnyStore => {
69
+ let isOpen = false;
70
+
71
+ let m: FileSystemDirectoryHandle = root!;
72
+ let sizeCache: number = 0;
73
+ const sizeMap: Map<string, number> = new Map(); // files size per key
74
+
75
+ const calculateSize = async () => {
76
+ sizeCache = 0;
77
+ for await (const value of m.values()) {
78
+ if (value.kind === "file") {
79
+ try {
80
+ const handle = await waitForSyncAcccess(value);
81
+ const handleSize = handle.getSize();
82
+ sizeMap.set(value.name, handleSize);
83
+ sizeCache += handleSize;
84
+ handle.close();
85
+ } catch (error) {
86
+ // TODO better error handling.
87
+ // most commonly error can be thrown when we are invoking getSize and files does not exist (anymore)
88
+ }
89
+ }
90
+ }
91
+ };
92
+ // 'open' | 'closed' is just a virtual thing since OPFS is always open as soone as we get the FileSystemDirectoryHandle
93
+ // TODO remove status? or assume not storage adapters can be closed?
94
+ const open = async () => {
95
+ await ready;
96
+
97
+ if (!m) {
98
+ m = await navigator.storage.getDirectory();
99
+ if (directory) {
100
+ m = await m.getDirectoryHandle(encodeName(directory), {
101
+ create: true
102
+ });
103
+ }
104
+ await calculateSize();
105
+ };
106
+ isOpen = true;
107
+ };
108
+ return {
109
+ clear: async () => {
110
+ for await (const key of m.keys()) {
111
+ m.removeEntry(key, { recursive: true });
112
+ }
113
+ sizeCache = 0;
114
+ sizeMap.clear();
115
+ },
116
+
117
+ del: async (key: string) => {
118
+ const encodedKey = encodeName(key);
119
+ try {
120
+ await m.removeEntry(encodedKey, { recursive: true });
121
+ } catch (error) {
122
+ if (error instanceof DOMException) {
123
+ return;
124
+ } else {
125
+ throw error;
126
+ }
127
+ }
128
+ const prevSize = sizeMap.get(encodedKey);
129
+ if (prevSize != null) {
130
+ sizeCache -= prevSize;
131
+ sizeMap.delete(encodedKey);
132
+ }
133
+ },
134
+
135
+ get: async (key: string) => {
136
+ try {
137
+ const fileHandle = await m.getFileHandle(encodeName(key));
138
+ const file = await fileHandle.getFile();
139
+ const buffer = await file.arrayBuffer();
140
+ return new Uint8Array(buffer);
141
+ } catch (error) {
142
+ if (
143
+ error instanceof DOMException &&
144
+ error.name === "NotFoundError"
145
+ ) {
146
+ return undefined;
147
+ } else if (error) {
148
+ return undefined;
149
+ } else {
150
+ throw error;
151
+ }
152
+ }
153
+ },
154
+ put: async (key: string, value: Uint8Array) => {
155
+ const encodedKey = encodeName(key);
156
+
157
+ const fileHandle = await m.getFileHandle(encodedKey, {
158
+ create: true
159
+ });
160
+ const writeFileHandle = await createWriteHandle(fileHandle);
161
+ writeFileHandle.write(value);
162
+ writeFileHandle.flush();
163
+ writeFileHandle.close();
164
+
165
+ const prevSize = sizeMap.get(encodedKey);
166
+ if (prevSize) {
167
+ sizeCache -= prevSize;
168
+ }
169
+ sizeCache += value.byteLength;
170
+ sizeMap.set(encodedKey, value.byteLength);
171
+ },
172
+
173
+ size: async () => {
174
+ return sizeCache;
175
+ },
176
+ status: () => (isOpen ? "open" : "closed"),
177
+
178
+ sublevel: async (name) => {
179
+ const encodedName = encodeName(name);
180
+ const fileHandle = await m.getDirectoryHandle(encodedName, {
181
+ create: true
182
+ });
183
+ const sublevel = [...levels, name];
184
+ const subMemory = createMemory(fileHandle, sublevel);
185
+ this._levels.set(memory.levelKey(sublevel), subMemory);
186
+ await subMemory.open();
187
+ return subMemory;
188
+ },
189
+
190
+ async *iterator(): AsyncGenerator<[string, Uint8Array], void, void> {
191
+ for await (const v of m.values()) {
192
+ if (v.kind == "file") {
193
+ yield [
194
+ decodeName(v.name),
195
+ new Uint8Array(await (await v.getFile()).arrayBuffer())
196
+ ];
197
+ }
198
+ }
199
+ },
200
+ close: async () => {
201
+ sizeCache = undefined as any; // TODO types
202
+ isOpen = false;
203
+ this._memoryIterator.clear();
204
+ },
205
+ open
206
+ };
207
+ };
208
+
209
+ this._rootStore = createMemory();
210
+
211
+ self.addEventListener("message", async (ev) => {
212
+ const message = deserialize(ev["data"], memory.MemoryRequest);
213
+ try {
214
+ if (message instanceof memory.MemoryMessage) {
215
+ if (message instanceof memory.REQ_Open) {
216
+ if (await this._rootStore.status() === "closed") {
217
+ await this._rootStore.open();
218
+ }
219
+ await this.respond(
220
+ message,
221
+ new memory.RESP_Open({ level: message.level }),
222
+ postMessageFn
223
+ );
224
+ return;
225
+ }
226
+
227
+ const m =
228
+ message.level.length === 0
229
+ ? this._rootStore
230
+ : this._levels.get(memory.levelKey(message.level));
231
+ if (!m) {
232
+ throw new Error("Recieved memory message for an undefined level");
233
+ } else if (message instanceof memory.REQ_Clear) {
234
+ await m.clear();
235
+ await this.respond(
236
+ message,
237
+ new memory.RESP_Clear({ level: message.level }),
238
+ postMessageFn
239
+ );
240
+ } else if (message instanceof memory.REQ_Close) {
241
+ console.log("CLOSE > ", message.level)
242
+ await m.close();
243
+ await this.respond(
244
+ message,
245
+ new memory.RESP_Close({ level: message.level }),
246
+ postMessageFn
247
+ );
248
+ } else if (message instanceof memory.REQ_Del) {
249
+ await m.del(message.key);
250
+ await this.respond(
251
+ message,
252
+ new memory.RESP_Del({ level: message.level }),
253
+ postMessageFn
254
+ );
255
+ } else if (message instanceof memory.REQ_Iterator_Next) {
256
+ let iterator = this._memoryIterator.get(message.id);
257
+ if (!iterator) {
258
+ iterator = m.iterator()[Symbol.asyncIterator]();
259
+ this._memoryIterator.set(message.id, iterator);
260
+ }
261
+ const next = await iterator.next();
262
+ await this.respond(
263
+ message,
264
+ new memory.RESP_Iterator_Next({
265
+ keys: next.done ? [] : [(next.value as any)[0]],
266
+ values: next.done ? [] : [new Uint8Array((next.value as any)[1])],
267
+ level: message.level
268
+ }),
269
+ postMessageFn
270
+ );
271
+ if (next.done) {
272
+ this._memoryIterator.delete(message.id);
273
+ }
274
+ } else if (message instanceof memory.REQ_Iterator_Stop) {
275
+ this._memoryIterator.delete(message.id);
276
+ await this.respond(
277
+ message,
278
+ new memory.RESP_Iterator_Stop({ level: message.level }),
279
+ postMessageFn
280
+ );
281
+ } else if (message instanceof memory.REQ_Get) {
282
+ const value = await m.get(message.key);
283
+ await this.respond(
284
+ message,
285
+ new memory.RESP_Get({
286
+ bytes: value ? new Uint8Array(value) : undefined,
287
+ level: message.level
288
+ }),
289
+ postMessageFn
290
+ );
291
+ } else if (message instanceof memory.REQ_Put) {
292
+ await m.put(message.key, message.bytes);
293
+ await this.respond(
294
+ message,
295
+ new memory.RESP_Put({ level: message.level }),
296
+ postMessageFn
297
+ );
298
+ } else if (message instanceof memory.REQ_Size) {
299
+ await this.respond(
300
+ message,
301
+ new memory.RESP_Size({
302
+ size: await m.size(),
303
+ level: message.level
304
+ }),
305
+ postMessageFn
306
+ );
307
+ } else if (message instanceof memory.REQ_Status) {
308
+ await this.respond(
309
+ message,
310
+ new memory.RESP_Status({
311
+ status: await m.status(),
312
+ level: message.level
313
+ }),
314
+ postMessageFn
315
+ );
316
+ } else if (message instanceof memory.REQ_Sublevel) {
317
+ await m.sublevel(message.name);
318
+
319
+ await this.respond(
320
+ message,
321
+ new memory.RESP_Sublevel({ level: message.level }),
322
+ postMessageFn
323
+ );
324
+ }
325
+ }
326
+ } catch (error) {
327
+ await this.respond(
328
+ message,
329
+ new memory.RESP_Error({
330
+ error: (error as any)?.["message"] || error?.constructor.name,
331
+ level: (message as any)["level"] || []
332
+ }),
333
+ postMessageFn
334
+ );
335
+ }
336
+ });
337
+ }
338
+
339
+ async respond(
340
+ request: memory.MemoryRequest,
341
+ response: memory.MemoryRequest,
342
+ postMessageFn = postMessage
343
+ ) {
344
+ response.messageId = request.messageId;
345
+ const bytes = serialize(response);
346
+ postMessageFn(bytes, { transfer: [bytes.buffer] });
347
+ }
348
+ }
349
+
350
+ new OPFSStoreWorker();