@verdant-web/store 4.1.1 → 4.1.3
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/bundle/index.js +5 -5
- package/dist/bundle/index.js.map +4 -4
- package/dist/esm/__tests__/queries.test.js +51 -2
- package/dist/esm/__tests__/queries.test.js.map +1 -1
- package/dist/esm/client/Client.d.ts +3 -1
- package/dist/esm/client/Client.js +8 -1
- package/dist/esm/client/Client.js.map +1 -1
- package/dist/esm/client/ClientDescriptor.d.ts +6 -2
- package/dist/esm/client/ClientDescriptor.js +7 -6
- package/dist/esm/client/ClientDescriptor.js.map +1 -1
- package/dist/esm/context/context.d.ts +17 -3
- package/dist/esm/entities/Entity.d.ts +2 -2
- package/dist/esm/entities/Entity.js +5 -1
- package/dist/esm/entities/Entity.js.map +1 -1
- package/dist/esm/entities/types.d.ts +41 -0
- package/dist/esm/files/FileManager.d.ts +3 -1
- package/dist/esm/files/FileManager.js +9 -2
- package/dist/esm/files/FileManager.js.map +1 -1
- package/dist/esm/persistence/PersistenceFiles.js +3 -2
- package/dist/esm/persistence/PersistenceFiles.js.map +1 -1
- package/dist/esm/persistence/idb/files/IdbPersistenceFileDb.d.ts +2 -2
- package/dist/esm/persistence/idb/files/IdbPersistenceFileDb.js +12 -4
- package/dist/esm/persistence/idb/files/IdbPersistenceFileDb.js.map +1 -1
- package/dist/esm/persistence/idb/queries/IdbDocumentDb.js +35 -18
- package/dist/esm/persistence/idb/queries/IdbDocumentDb.js.map +1 -1
- package/dist/esm/persistence/persistence.js +3 -3
- package/dist/esm/persistence/persistence.js.map +1 -1
- package/dist/esm/queries/BaseQuery.js +2 -1
- package/dist/esm/queries/BaseQuery.js.map +1 -1
- package/dist/esm/queries/QueryCache.d.ts +8 -1
- package/dist/esm/queries/QueryCache.js +33 -2
- package/dist/esm/queries/QueryCache.js.map +1 -1
- package/dist/esm/sync/FileSync.d.ts +1 -1
- package/dist/esm/sync/FileSync.js +2 -1
- package/dist/esm/sync/FileSync.js.map +1 -1
- package/package.json +1 -3
- package/src/__tests__/queries.test.ts +70 -3
- package/src/client/Client.ts +13 -7
- package/src/client/ClientDescriptor.ts +14 -8
- package/src/context/context.ts +18 -3
- package/src/entities/Entity.ts +8 -4
- package/src/entities/types.ts +57 -24
- package/src/files/FileManager.ts +15 -2
- package/src/persistence/PersistenceFiles.ts +11 -6
- package/src/persistence/idb/files/IdbPersistenceFileDb.ts +22 -9
- package/src/persistence/idb/queries/IdbDocumentDb.ts +43 -18
- package/src/persistence/persistence.ts +4 -2
- package/src/queries/BaseQuery.ts +3 -1
- package/src/queries/QueryCache.ts +56 -2
- package/src/sync/FileSync.ts +3 -2
|
@@ -8,8 +8,8 @@ import {
|
|
|
8
8
|
import { Context } from '../../../context/context.js';
|
|
9
9
|
import { PersistenceDocumentDb } from '../../interfaces.js';
|
|
10
10
|
import { IdbService } from '../IdbService.js';
|
|
11
|
-
import { getRange } from './ranges.js';
|
|
12
11
|
import { closeDatabase, getSizeOfObjectStore, isAbortError } from '../util.js';
|
|
12
|
+
import { getRange } from './ranges.js';
|
|
13
13
|
|
|
14
14
|
export class IdbDocumentDb extends IdbService implements PersistenceDocumentDb {
|
|
15
15
|
private ctx;
|
|
@@ -149,7 +149,7 @@ export class IdbDocumentDb extends IdbService implements PersistenceDocumentDb {
|
|
|
149
149
|
}),
|
|
150
150
|
};
|
|
151
151
|
|
|
152
|
-
await Promise.
|
|
152
|
+
const results = await Promise.allSettled(
|
|
153
153
|
entities.map(async (e) => {
|
|
154
154
|
const snapshot = e.getSnapshot();
|
|
155
155
|
try {
|
|
@@ -168,6 +168,26 @@ export class IdbDocumentDb extends IdbService implements PersistenceDocumentDb {
|
|
|
168
168
|
}
|
|
169
169
|
}),
|
|
170
170
|
);
|
|
171
|
+
|
|
172
|
+
const failures = results.filter((r) => r.status === 'rejected');
|
|
173
|
+
if (failures.length) {
|
|
174
|
+
// in the case of a failure to save a document, it doesn't quite make sense to cancel or rollback whatever is
|
|
175
|
+
// currently happening. when restoring imports, etc, this makes the app stuck. if only a few docs failed, maybe
|
|
176
|
+
// there's some data corruption somewhere, but we can just lose those without affecting the rest of the data.
|
|
177
|
+
if (failures.length === results.length) {
|
|
178
|
+
// but if ALL of them failed, that's trouble...
|
|
179
|
+
throw new Error(
|
|
180
|
+
'Failed to save any documents. Something must be quite wrong.',
|
|
181
|
+
);
|
|
182
|
+
}
|
|
183
|
+
this.ctx.log(
|
|
184
|
+
'error',
|
|
185
|
+
'Failed to save documents:',
|
|
186
|
+
failures,
|
|
187
|
+
". See logs above. This only affects querying these documents. Let's hope a future attempt will correct them...",
|
|
188
|
+
);
|
|
189
|
+
}
|
|
190
|
+
|
|
171
191
|
options.transaction.commit();
|
|
172
192
|
};
|
|
173
193
|
|
|
@@ -190,22 +210,27 @@ export class IdbDocumentDb extends IdbService implements PersistenceDocumentDb {
|
|
|
190
210
|
) => {
|
|
191
211
|
this.ctx.log('debug', `Saving document indexes for querying ${oid}`);
|
|
192
212
|
const { collection, id } = decomposeOid(oid);
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
213
|
+
try {
|
|
214
|
+
if (!doc) {
|
|
215
|
+
await this.run(collection, (store) => store.delete(id), {
|
|
216
|
+
mode: 'readwrite',
|
|
217
|
+
transaction,
|
|
218
|
+
});
|
|
219
|
+
this.ctx.log('debug', `Deleted document indexes for querying ${oid}`);
|
|
220
|
+
} else {
|
|
221
|
+
const schema = this.ctx.schema.collections[collection];
|
|
222
|
+
// no need to validate before storing; the entity's snapshot is already validated.
|
|
223
|
+
const indexes = getIndexValues(schema, doc);
|
|
224
|
+
indexes['@@@snapshot'] = JSON.stringify(doc);
|
|
225
|
+
await this.run(collection, (store) => store.put(indexes), {
|
|
226
|
+
mode: 'readwrite',
|
|
227
|
+
transaction,
|
|
228
|
+
});
|
|
229
|
+
this.ctx.log('debug', `Save complete for ${oid}`, indexes);
|
|
230
|
+
}
|
|
231
|
+
} catch (err) {
|
|
232
|
+
this.ctx.log('error', `Error saving document ${oid}`, err);
|
|
233
|
+
throw err;
|
|
209
234
|
}
|
|
210
235
|
};
|
|
211
236
|
}
|
|
@@ -1,12 +1,12 @@
|
|
|
1
1
|
import { EventSubscriber, getOidRoot, VerdantError } from '@verdant-web/common';
|
|
2
2
|
import { Context, InitialContext } from '../context/context.js';
|
|
3
|
+
import { ShutdownHandler } from '../context/ShutdownHandler.js';
|
|
3
4
|
import { getWipNamespace } from '../utils/wip.js';
|
|
4
5
|
import { ExportedData } from './interfaces.js';
|
|
6
|
+
import { migrate } from './migration/migrate.js';
|
|
5
7
|
import { PersistenceFiles } from './PersistenceFiles.js';
|
|
6
8
|
import { PersistenceMetadata } from './PersistenceMetadata.js';
|
|
7
9
|
import { PersistenceDocuments } from './PersistenceQueries.js';
|
|
8
|
-
import { migrate } from './migration/migrate.js';
|
|
9
|
-
import { ShutdownHandler } from '../context/ShutdownHandler.js';
|
|
10
10
|
|
|
11
11
|
export async function initializePersistence(
|
|
12
12
|
ctx: InitialContext,
|
|
@@ -130,6 +130,8 @@ export async function importPersistence(
|
|
|
130
130
|
originalNamespace: importedNamespace,
|
|
131
131
|
// no-op entity events -- don't need to inform queries of changes.
|
|
132
132
|
entityEvents: new EventSubscriber(),
|
|
133
|
+
internalEvents: new EventSubscriber(),
|
|
134
|
+
globalEvents: new EventSubscriber(),
|
|
133
135
|
config: {
|
|
134
136
|
...ctx.config,
|
|
135
137
|
persistence: {
|
package/src/queries/BaseQuery.ts
CHANGED
|
@@ -57,7 +57,9 @@ export abstract class BaseQuery<T> extends Disposable {
|
|
|
57
57
|
this.isListQuery = Array.isArray(initial);
|
|
58
58
|
this._events = new EventSubscriber<BaseQueryEvents>(
|
|
59
59
|
(event: keyof BaseQueryEvents) => {
|
|
60
|
-
if (event === 'change')
|
|
60
|
+
if (event === 'change') {
|
|
61
|
+
this._allUnsubscribedHandler?.(this);
|
|
62
|
+
}
|
|
61
63
|
},
|
|
62
64
|
);
|
|
63
65
|
this.context = context;
|
|
@@ -6,6 +6,8 @@ export class QueryCache extends Disposable {
|
|
|
6
6
|
private _cache: Map<string, BaseQuery<any>> = new Map();
|
|
7
7
|
private _evictionTime;
|
|
8
8
|
private context;
|
|
9
|
+
/** A set of query keys to keep alive even if they unsubscribe */
|
|
10
|
+
private _holds = new Set<string>();
|
|
9
11
|
|
|
10
12
|
constructor({
|
|
11
13
|
evictionTime = 5 * 1000,
|
|
@@ -26,13 +28,22 @@ export class QueryCache extends Disposable {
|
|
|
26
28
|
);
|
|
27
29
|
}
|
|
28
30
|
|
|
31
|
+
get activeKeys() {
|
|
32
|
+
return Array.from(this._cache.keys());
|
|
33
|
+
}
|
|
34
|
+
|
|
29
35
|
get<T extends BaseQuery<any>>(key: string): T | null {
|
|
30
36
|
return (this._cache.get(key) as T) || null;
|
|
31
37
|
}
|
|
32
38
|
|
|
33
39
|
set<V extends BaseQuery<any>>(value: V) {
|
|
34
40
|
this._cache.set(value.key, value);
|
|
35
|
-
value[ON_ALL_UNSUBSCRIBED](this.
|
|
41
|
+
value[ON_ALL_UNSUBSCRIBED](this.enqueueQueryEviction);
|
|
42
|
+
// immediately enqueue a check to see if this query should be evicted --
|
|
43
|
+
// this basically gives code X seconds to subscribe to the query before
|
|
44
|
+
// it gets evicted.
|
|
45
|
+
this.enqueueQueryEviction(value);
|
|
46
|
+
|
|
36
47
|
return value;
|
|
37
48
|
}
|
|
38
49
|
|
|
@@ -50,13 +61,24 @@ export class QueryCache extends Disposable {
|
|
|
50
61
|
return this.set(create());
|
|
51
62
|
}
|
|
52
63
|
|
|
53
|
-
private
|
|
64
|
+
private enqueueQueryEviction = (query: BaseQuery<any>) => {
|
|
54
65
|
setTimeout(() => {
|
|
55
66
|
if (query.subscribed) return;
|
|
67
|
+
|
|
68
|
+
if (this._holds.has(query.key)) {
|
|
69
|
+
this.context.log(
|
|
70
|
+
'debug',
|
|
71
|
+
'QueryCache: keepAlive hold on query preserves after unsubscribe',
|
|
72
|
+
query.key,
|
|
73
|
+
);
|
|
74
|
+
return;
|
|
75
|
+
}
|
|
76
|
+
|
|
56
77
|
// double check before evicting... possible the cache
|
|
57
78
|
// got a different version of this query.
|
|
58
79
|
if (this._cache.get(query.key) === query) {
|
|
59
80
|
this._cache.delete(query.key);
|
|
81
|
+
this.context.log('debug', 'QueryCache: evicted query', query.key);
|
|
60
82
|
}
|
|
61
83
|
}, this._evictionTime);
|
|
62
84
|
};
|
|
@@ -81,4 +103,36 @@ export class QueryCache extends Disposable {
|
|
|
81
103
|
);
|
|
82
104
|
this._cache.forEach((q) => q.execute());
|
|
83
105
|
};
|
|
106
|
+
|
|
107
|
+
keepAlive(key: string) {
|
|
108
|
+
this._holds.add(key);
|
|
109
|
+
this.context.log('debug', 'QueryCache: keepAlive', key);
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
dropKeepAlive(key: string) {
|
|
113
|
+
this._holds.delete(key);
|
|
114
|
+
const cached = this.get(key);
|
|
115
|
+
if (!cached) return;
|
|
116
|
+
if (!cached.subscribed) {
|
|
117
|
+
this.context.log(
|
|
118
|
+
'debug',
|
|
119
|
+
'QueryCache: dropKeepAlive on unsubscribed query; queuing eviction',
|
|
120
|
+
key,
|
|
121
|
+
);
|
|
122
|
+
this.enqueueQueryEviction(cached);
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
get keepAlives() {
|
|
127
|
+
return this._holds;
|
|
128
|
+
}
|
|
84
129
|
}
|
|
130
|
+
|
|
131
|
+
export type PublicQueryCacheAPI = Pick<
|
|
132
|
+
QueryCache,
|
|
133
|
+
| 'keepAlive'
|
|
134
|
+
| 'dropKeepAlive'
|
|
135
|
+
| 'keepAlives'
|
|
136
|
+
| 'forceRefreshAll'
|
|
137
|
+
| 'activeKeys'
|
|
138
|
+
>;
|
package/src/sync/FileSync.ts
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { FileData } from '@verdant-web/common';
|
|
2
|
-
import { ServerSyncEndpointProvider } from './ServerSyncEndpointProvider.js';
|
|
3
2
|
import { Context } from '../context/context.js';
|
|
4
3
|
import { Disposable } from '../utils/Disposable.js';
|
|
4
|
+
import { ServerSyncEndpointProvider } from './ServerSyncEndpointProvider.js';
|
|
5
5
|
|
|
6
6
|
export interface FileUploadResult {
|
|
7
7
|
success: boolean;
|
|
@@ -42,7 +42,6 @@ export class FileSync extends Disposable {
|
|
|
42
42
|
this.ctx.log('debug', 'Uploading file', data.id, data.name);
|
|
43
43
|
try {
|
|
44
44
|
await this.uploadFile(data);
|
|
45
|
-
this.ctx.internalEvents.emit(`fileUploaded:${data.id}`, data);
|
|
46
45
|
} catch (e) {
|
|
47
46
|
this.ctx.log('error', 'File upload failed', e);
|
|
48
47
|
}
|
|
@@ -83,6 +82,8 @@ export class FileSync extends Disposable {
|
|
|
83
82
|
);
|
|
84
83
|
|
|
85
84
|
if (response.ok) {
|
|
85
|
+
this.ctx.internalEvents.emit(`fileUploaded:${data.id}`, data);
|
|
86
|
+
this.ctx.internalEvents.emit('fileUploaded', data);
|
|
86
87
|
this.ctx.log('info', 'File upload successful');
|
|
87
88
|
return {
|
|
88
89
|
success: true,
|