@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.
- package/dist/peerbit/anystore-opfs-worker.min.js +1 -0
- package/dist/src/create.d.ts +3 -0
- package/dist/src/create.d.ts.map +1 -0
- package/dist/src/create.js +3 -0
- package/dist/src/create.js.map +1 -0
- package/dist/src/index.d.ts +2 -0
- package/dist/src/index.d.ts.map +1 -0
- package/dist/src/index.js +2 -0
- package/dist/src/index.js.map +1 -0
- package/dist/src/opfs-store.d.ts +28 -0
- package/dist/src/opfs-store.d.ts.map +1 -0
- package/dist/src/opfs-store.js +147 -0
- package/dist/src/opfs-store.js.map +1 -0
- package/dist/src/worker.d.ts +9 -0
- package/dist/src/worker.d.ts.map +1 -0
- package/dist/src/worker.js +280 -0
- package/dist/src/worker.js.map +1 -0
- package/package.json +80 -0
- package/src/create.ts +2 -0
- package/src/index.ts +1 -0
- package/src/opfs-store.ts +199 -0
- package/src/worker.ts +350 -0
|
@@ -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();
|