@instantdb/core 0.22.164 → 0.22.165

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.
Files changed (60) hide show
  1. package/__tests__/src/infiniteQuery.e2e.test.ts +384 -0
  2. package/__tests__/src/simple.e2e.test.ts +0 -1
  3. package/__tests__/src/utils/e2e.ts +1 -1
  4. package/dist/commonjs/Reactor.d.ts +1 -1
  5. package/dist/commonjs/Reactor.js +3 -3
  6. package/dist/commonjs/Reactor.js.map +1 -1
  7. package/dist/commonjs/index.d.ts +23 -2
  8. package/dist/commonjs/index.d.ts.map +1 -1
  9. package/dist/commonjs/index.js +25 -1
  10. package/dist/commonjs/index.js.map +1 -1
  11. package/dist/commonjs/infiniteQuery.d.ts +26 -0
  12. package/dist/commonjs/infiniteQuery.d.ts.map +1 -0
  13. package/dist/commonjs/infiniteQuery.js +422 -0
  14. package/dist/commonjs/infiniteQuery.js.map +1 -0
  15. package/dist/commonjs/instaql.d.ts.map +1 -1
  16. package/dist/commonjs/instaql.js +18 -5
  17. package/dist/commonjs/instaql.js.map +1 -1
  18. package/dist/commonjs/queryTypes.d.ts +2 -2
  19. package/dist/commonjs/queryTypes.d.ts.map +1 -1
  20. package/dist/commonjs/queryTypes.js.map +1 -1
  21. package/dist/commonjs/utils/Deferred.d.ts +5 -4
  22. package/dist/commonjs/utils/Deferred.d.ts.map +1 -1
  23. package/dist/commonjs/utils/Deferred.js.map +1 -1
  24. package/dist/commonjs/utils/weakHash.d.ts.map +1 -1
  25. package/dist/commonjs/utils/weakHash.js +4 -0
  26. package/dist/commonjs/utils/weakHash.js.map +1 -1
  27. package/dist/esm/Reactor.d.ts +1 -1
  28. package/dist/esm/Reactor.js +1 -1
  29. package/dist/esm/Reactor.js.map +1 -1
  30. package/dist/esm/index.d.ts +23 -2
  31. package/dist/esm/index.d.ts.map +1 -1
  32. package/dist/esm/index.js +25 -0
  33. package/dist/esm/index.js.map +1 -1
  34. package/dist/esm/infiniteQuery.d.ts +26 -0
  35. package/dist/esm/infiniteQuery.d.ts.map +1 -0
  36. package/dist/esm/infiniteQuery.js +417 -0
  37. package/dist/esm/infiniteQuery.js.map +1 -0
  38. package/dist/esm/instaql.d.ts.map +1 -1
  39. package/dist/esm/instaql.js +18 -5
  40. package/dist/esm/instaql.js.map +1 -1
  41. package/dist/esm/queryTypes.d.ts +2 -2
  42. package/dist/esm/queryTypes.d.ts.map +1 -1
  43. package/dist/esm/queryTypes.js.map +1 -1
  44. package/dist/esm/utils/Deferred.d.ts +5 -4
  45. package/dist/esm/utils/Deferred.d.ts.map +1 -1
  46. package/dist/esm/utils/Deferred.js.map +1 -1
  47. package/dist/esm/utils/weakHash.d.ts.map +1 -1
  48. package/dist/esm/utils/weakHash.js +4 -0
  49. package/dist/esm/utils/weakHash.js.map +1 -1
  50. package/dist/standalone/index.js +1731 -1432
  51. package/dist/standalone/index.umd.cjs +3 -3
  52. package/package.json +2 -2
  53. package/src/Reactor.js +1 -1
  54. package/src/index.ts +49 -0
  55. package/src/infiniteQuery.ts +573 -0
  56. package/src/instaql.ts +25 -7
  57. package/src/queryTypes.ts +1 -2
  58. package/src/utils/{Deferred.js → Deferred.ts} +4 -4
  59. package/src/utils/weakHash.ts +4 -0
  60. package/vitest.config.ts +6 -0
@@ -0,0 +1,422 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.getInfiniteQueryInitialSnapshot = exports.subscribeInfiniteQuery = void 0;
4
+ const index_ts_1 = require("./index.js");
5
+ const error_ts_1 = require("./utils/error.js");
6
+ // Example for {order: {value: "asc"}}
7
+ //
8
+ // 0
9
+ // <------------------|------------------------------------------------------>
10
+ // <- starter sub ->
11
+ //
12
+ // Bootstrap phase: until the limit (4 in this example) items are reached, the
13
+ // starter subscription is the only subscription and it writes to the forwardChunks map with the key PRE_BOOTSTRAP_CURSOR.
14
+ //
15
+ // When the limit is reached it automatically becomes a real forward chunk and has a definite start and end.
16
+ // A new reverse chunk gets added to watch for any new items at the start of the list.
17
+ //
18
+ // 0 1 2 3
19
+ // <------------------|------------------------------------------------------>
20
+ // <- starter sub ->
21
+ //
22
+ // ↓ BECOMES ↓
23
+ //
24
+ // 0 1 2 3
25
+ // <------------------|------------------------------------------------------>
26
+ // <-reverse chunk][forward chunk ]
27
+ //
28
+ // 0 1 2 3 4
29
+ // <------------------|------------------------------------------------------>
30
+ // <-reverse chunk][forward chunk ]
31
+ // When item 4 is added, the forward chunk subscription gets updated so that
32
+ // hasNextPage is `true`. This tells the user that a new page can be loaded.
33
+ //
34
+ // User clicks: loadNextPage
35
+ // 0 1 2 3 4
36
+ // <------------------|------------------------------------------------------>
37
+ // <-reverse chunk][ frozen forward chunk ][ new forward chunk ]
38
+ //
39
+ // More numbers get added
40
+ // 0 1 2 3 4 5 6 7 8
41
+ // <------------------|------------------------------------------------------>
42
+ // <-reverse chunk][ frozen forward chunk ][ forward chunk ] ^
43
+ // hasNextPage=true^
44
+ //
45
+ //
46
+ // User clicks: loadNextPage
47
+ //
48
+ // 0 1 2 3 4 5 6 7 8
49
+ // <------------------|------------------------------------------------------>
50
+ // <-reverse chunk][ frozen forward chunk ][ frozen forward chunk ][ new chunk
51
+ //
52
+ // The reverse chunks work in the same way as the forward chunks but the order in the query is reversed.
53
+ // When a reverse chunks recieves an update it will check to see if more can be loaded and it will
54
+ // automatically freeze the chunk and add a new one. i.e. : works the same as if
55
+ // loadNextPage was automatically clicked when hasNextPage became true.
56
+ //
57
+ // Chunks are indexed by their starting point cursor, for forward chunks this is the "[" point.
58
+ // Their starting point cursor is inclusive in the query and exclusive from the following query
59
+ const makeCursorKey = (cursor) => JSON.stringify(cursor);
60
+ const parseCursorKey = (cursorKey) => JSON.parse(cursorKey);
61
+ const chunkHasEndCursor = (chunk) => {
62
+ return !!chunk.endCursor;
63
+ };
64
+ const readCanLoadNextPage = (forwardChunks) => {
65
+ const chunksInOrder = Array.from(forwardChunks.values());
66
+ if (chunksInOrder.length === 0)
67
+ return false;
68
+ return chunksInOrder[chunksInOrder.length - 1]?.hasMore || false;
69
+ };
70
+ // Chunk sub key is used to create keys to keep track of the subscriptions
71
+ // while the chunk maps are keyed by the cursor, here we disinguish between
72
+ // forward and reverse because the first 2 chunks will have the same starting
73
+ // cursor.
74
+ const chunkSubKey = (direction, cursor) => `${direction}:${JSON.stringify(cursor)}`;
75
+ const reverseOrder = (order) => {
76
+ if (!order) {
77
+ return {
78
+ serverCreatedAt: 'asc',
79
+ };
80
+ }
81
+ const key = Object.keys(order).at(0);
82
+ if (!key) {
83
+ return {
84
+ serverCreatedAt: 'asc',
85
+ };
86
+ }
87
+ return {
88
+ [key]: order[key] === 'asc' ? 'desc' : 'asc',
89
+ };
90
+ };
91
+ const normalizeChunks = (forwardChunks, reverseChunks) => {
92
+ const chunks = [
93
+ ...Array.from(reverseChunks.values()).slice().reverse(),
94
+ ...Array.from(forwardChunks.values()),
95
+ ];
96
+ const data = [
97
+ ...Array.from(reverseChunks.values())
98
+ .slice()
99
+ .reverse()
100
+ .flatMap((chunk) => chunk.data.slice().reverse()),
101
+ ...Array.from(forwardChunks.values()).flatMap((chunk) => chunk.data),
102
+ ];
103
+ return { chunks, data };
104
+ };
105
+ const PRE_BOOTSTRAP_CURSOR = ['bootstrap', 'bootstrap', 'bootstrap', 1];
106
+ const subscribeInfiniteQuery = (db, fullQuery, cb, opts) => {
107
+ const { entityName, entityQuery: query } = splitAndValidateQuery(fullQuery);
108
+ const pageSize = query.$?.limit || 10;
109
+ const entity = entityName;
110
+ const forwardChunks = new Map();
111
+ const reverseChunks = new Map();
112
+ // Keeps track of all subscriptions (besides starter sub)
113
+ const allUnsubs = new Map();
114
+ let hasKickstarted = false;
115
+ let isActive = true;
116
+ let lastReverseAdvancedChunkKey = null;
117
+ let starterUnsub = null;
118
+ const sendError = (err) => {
119
+ cb({ error: err, data: undefined, canLoadNextPage: false });
120
+ };
121
+ const pushUpdate = () => {
122
+ if (!isActive)
123
+ return;
124
+ const { chunks, data } = normalizeChunks(forwardChunks, reverseChunks);
125
+ cb({
126
+ data: { [entity]: data },
127
+ // @ts-expect-error hidden debug variable
128
+ chunks,
129
+ canLoadNextPage: readCanLoadNextPage(forwardChunks),
130
+ });
131
+ };
132
+ const setForwardChunk = (startCursor, chunk) => {
133
+ forwardChunks.set(makeCursorKey(startCursor), chunk);
134
+ pushUpdate();
135
+ };
136
+ const setReverseChunk = (startCursor, chunk) => {
137
+ reverseChunks.set(makeCursorKey(startCursor), chunk);
138
+ maybeAdvanceReverse();
139
+ pushUpdate();
140
+ };
141
+ const freezeReverse = (chunkKey, chunk) => {
142
+ const startCursor = parseCursorKey(chunkKey);
143
+ const currentSub = allUnsubs.get(chunkSubKey('reverse', startCursor));
144
+ currentSub?.();
145
+ const nextSub = db.subscribeQuery({
146
+ [entity]: {
147
+ ...query,
148
+ $: {
149
+ after: startCursor,
150
+ before: chunk.endCursor,
151
+ beforeInclusive: true,
152
+ where: query.$?.where,
153
+ fields: query.$?.fields,
154
+ order: reverseOrder(query.$?.order),
155
+ },
156
+ },
157
+ }, (frozenData) => {
158
+ if (frozenData.error) {
159
+ return sendError(frozenData.error);
160
+ }
161
+ const rows = frozenData.data[entity];
162
+ const pageInfo = frozenData.pageInfo[entity];
163
+ (0, error_ts_1.assert)(rows && pageInfo, 'Expected query subscription to contain rows and pageInfo');
164
+ setReverseChunk(startCursor, {
165
+ data: rows,
166
+ status: 'frozen',
167
+ hasMore: pageInfo.hasNextPage,
168
+ endCursor: pageInfo.endCursor,
169
+ });
170
+ }, opts);
171
+ allUnsubs.set(chunkSubKey('reverse', startCursor), nextSub);
172
+ };
173
+ const pushNewReverse = (startCursor) => {
174
+ const querySub = db.subscribeQuery({
175
+ [entity]: {
176
+ ...query,
177
+ $: {
178
+ limit: pageSize,
179
+ after: startCursor,
180
+ where: query.$?.where,
181
+ fields: query.$?.fields,
182
+ order: reverseOrder(query.$?.order),
183
+ },
184
+ },
185
+ }, (windowData) => {
186
+ if (windowData.error) {
187
+ return sendError(windowData.error);
188
+ }
189
+ const rows = windowData.data[entity];
190
+ const pageInfo = windowData.pageInfo[entity];
191
+ (0, error_ts_1.assert)(rows && pageInfo, 'Expected rows and pageInfo');
192
+ setReverseChunk(startCursor, {
193
+ data: rows,
194
+ status: 'bootstrapping',
195
+ hasMore: pageInfo.hasNextPage,
196
+ endCursor: pageInfo.endCursor,
197
+ });
198
+ }, opts);
199
+ allUnsubs.set(chunkSubKey('reverse', startCursor), querySub);
200
+ };
201
+ const pushNewForward = (startCursor, afterInclusive = false) => {
202
+ const querySub = db.subscribeQuery({
203
+ [entity]: {
204
+ ...query,
205
+ $: {
206
+ limit: pageSize,
207
+ after: startCursor,
208
+ afterInclusive,
209
+ where: query.$?.where,
210
+ fields: query.$?.fields,
211
+ order: query.$?.order,
212
+ },
213
+ },
214
+ }, (windowData) => {
215
+ if (windowData.error) {
216
+ return sendError(windowData.error);
217
+ }
218
+ const rows = windowData.data[entity];
219
+ const pageInfo = windowData.pageInfo[entity];
220
+ (0, error_ts_1.assert)(rows && pageInfo, 'Page info and rows');
221
+ setForwardChunk(startCursor, {
222
+ data: rows,
223
+ status: 'bootstrapping',
224
+ hasMore: pageInfo.hasNextPage,
225
+ endCursor: pageInfo.endCursor,
226
+ afterInclusive,
227
+ });
228
+ }, opts);
229
+ allUnsubs.set(chunkSubKey('forward', startCursor), querySub);
230
+ };
231
+ const freezeForward = (startCursor) => {
232
+ const key = makeCursorKey(startCursor);
233
+ const currentSub = allUnsubs.get(chunkSubKey('forward', startCursor));
234
+ currentSub?.();
235
+ const chunk = forwardChunks.get(key);
236
+ if (!chunk?.endCursor)
237
+ return;
238
+ const nextSub = db.subscribeQuery({
239
+ [entity]: {
240
+ ...query,
241
+ $: {
242
+ after: startCursor,
243
+ afterInclusive: chunk.afterInclusive,
244
+ before: chunk.endCursor,
245
+ beforeInclusive: true,
246
+ where: query.$?.where,
247
+ fields: query.$?.fields,
248
+ order: query.$?.order,
249
+ },
250
+ },
251
+ }, (frozenData) => {
252
+ if (frozenData.error) {
253
+ return sendError(frozenData.error);
254
+ }
255
+ const rows = frozenData.data[entity];
256
+ const pageInfo = frozenData.pageInfo[entity];
257
+ (0, error_ts_1.assert)(rows && pageInfo, 'Expected rows and pageInfo');
258
+ setForwardChunk(startCursor, {
259
+ data: rows,
260
+ status: 'frozen',
261
+ hasMore: pageInfo.hasNextPage,
262
+ endCursor: pageInfo.endCursor,
263
+ afterInclusive: chunk.afterInclusive,
264
+ });
265
+ }, opts);
266
+ allUnsubs.set(chunkSubKey('forward', startCursor), nextSub);
267
+ };
268
+ // Consider order: {val: "asc"} with pageItems = 4
269
+ // A reverse chunk captures all the new items coming in before us.
270
+ // If we hit 4 then we freeze the current chunk and create a new reverse chunk
271
+ const maybeAdvanceReverse = () => {
272
+ const tailEntry = Array.from(reverseChunks.entries()).at(-1);
273
+ if (!tailEntry)
274
+ return;
275
+ const [chunkKey, chunk] = tailEntry;
276
+ // If a chunk has more, then it must have an endCursor
277
+ if (!chunk?.hasMore)
278
+ return;
279
+ if (!chunkHasEndCursor(chunk))
280
+ return;
281
+ // maybeAdvanceReverse can run multiple times if multiple changes are made
282
+ // to the reverse chunk
283
+ // This prevents adding the same new reverse frame twice
284
+ const advanceKey = `${chunkKey}:${makeCursorKey(chunk.endCursor)}`;
285
+ if (advanceKey == lastReverseAdvancedChunkKey)
286
+ return;
287
+ lastReverseAdvancedChunkKey = advanceKey;
288
+ freezeReverse(chunkKey, chunk);
289
+ pushNewReverse(chunk.endCursor);
290
+ };
291
+ const loadNextPage = () => {
292
+ const tailEntry = Array.from(forwardChunks.entries()).at(-1);
293
+ if (!tailEntry)
294
+ return;
295
+ const [chunkKey, chunk] = tailEntry;
296
+ // If the chunk has more items after it, it must have an end cursor, and we can
297
+ // load more items
298
+ // if (!chunk?.hasMore) return;
299
+ if (!chunk.endCursor)
300
+ return;
301
+ freezeForward(parseCursorKey(chunkKey));
302
+ pushNewForward(chunk.endCursor);
303
+ };
304
+ starterUnsub = db.subscribeQuery({
305
+ [entity]: {
306
+ ...query,
307
+ $: {
308
+ limit: pageSize,
309
+ where: query.$?.where,
310
+ fields: query.$?.fields,
311
+ order: query.$?.order,
312
+ },
313
+ },
314
+ }, async (starterData) => {
315
+ if (hasKickstarted)
316
+ return;
317
+ if (starterData.error) {
318
+ return sendError(starterData.error);
319
+ }
320
+ const pageInfo = starterData.pageInfo[entity];
321
+ const rows = starterData?.data?.[entity];
322
+ (0, error_ts_1.assert)(rows && pageInfo, 'Expected rows and pageInfo');
323
+ if (rows.length < pageSize) {
324
+ // If the rows are less than the page size, then we don't need to
325
+ // create forward and reverse chunks.
326
+ // We just treat the starter query as a forward chunk
327
+ setForwardChunk(PRE_BOOTSTRAP_CURSOR, {
328
+ data: rows,
329
+ status: 'pre-bootstrap',
330
+ });
331
+ return;
332
+ }
333
+ // Consider a query with no items; the server will return a result with
334
+ // no start cursor. If we add {pageSize} optimistic updates we can
335
+ // get here and still have no startCursor. By returning we are skipping
336
+ // the optimistic update and just waiting for the result from the
337
+ // server.
338
+ const initialForwardCursor = pageInfo.startCursor;
339
+ if (!initialForwardCursor) {
340
+ return;
341
+ }
342
+ forwardChunks.delete(makeCursorKey(PRE_BOOTSTRAP_CURSOR));
343
+ pushNewForward(initialForwardCursor, true);
344
+ pushNewReverse(pageInfo.startCursor);
345
+ hasKickstarted = true;
346
+ // Flush the initial boostrap querysub data
347
+ // because immediately unsubscribing will never save it for offline in idb
348
+ await db._reactor.querySubs.flush();
349
+ // Unsubscribe the starter subscription
350
+ starterUnsub?.();
351
+ starterUnsub = null;
352
+ }, opts);
353
+ const unsubscribe = () => {
354
+ if (!isActive)
355
+ return;
356
+ isActive = false;
357
+ starterUnsub?.();
358
+ starterUnsub = null;
359
+ for (const unsub of allUnsubs.values()) {
360
+ unsub?.();
361
+ }
362
+ allUnsubs.clear();
363
+ };
364
+ return {
365
+ unsubscribe,
366
+ loadNextPage,
367
+ };
368
+ };
369
+ exports.subscribeInfiniteQuery = subscribeInfiniteQuery;
370
+ const getInfiniteQueryInitialSnapshot = (db, fullQuery, opts) => {
371
+ if (!fullQuery) {
372
+ return {
373
+ canLoadNextPage: false,
374
+ data: undefined,
375
+ error: undefined,
376
+ };
377
+ }
378
+ const { entityName, entityQuery } = splitAndValidateQuery(fullQuery);
379
+ const pageSize = entityQuery.$?.limit || 10;
380
+ let coercedQuery = fullQuery
381
+ ? (0, index_ts_1.coerceQuery)({
382
+ [entityName]: {
383
+ ...entityQuery,
384
+ $: {
385
+ limit: pageSize,
386
+ where: entityQuery.$?.where,
387
+ fields: entityQuery.$?.fields,
388
+ order: entityQuery.$?.order,
389
+ },
390
+ },
391
+ })
392
+ : null;
393
+ if (opts && 'ruleParams' in opts) {
394
+ coercedQuery = {
395
+ $$ruleParams: opts.ruleParams,
396
+ ...fullQuery,
397
+ };
398
+ }
399
+ const queryResult = db._reactor.getPreviousResult(coercedQuery);
400
+ return {
401
+ canLoadNextPage: false,
402
+ data: queryResult?.data || undefined,
403
+ error: undefined,
404
+ };
405
+ };
406
+ exports.getInfiniteQueryInitialSnapshot = getInfiniteQueryInitialSnapshot;
407
+ /**
408
+ * @throws QueryValidationError
409
+ * @param fullQuery a ValidQuery with one key (entity)
410
+ */
411
+ const splitAndValidateQuery = (fullQuery) => {
412
+ const entityNames = Object.keys(fullQuery);
413
+ if (entityNames.length !== 1) {
414
+ throw new index_ts_1.QueryValidationError('subscribeInfiniteQuery expects exactly one entity');
415
+ }
416
+ const [entityName, entityQuery] = Object.entries(fullQuery)[0];
417
+ if (!entityName || !entityQuery) {
418
+ throw new index_ts_1.QueryValidationError('No query provided for infinite query');
419
+ }
420
+ return { entityName, entityQuery };
421
+ };
422
+ //# sourceMappingURL=infiniteQuery.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"infiniteQuery.js","sourceRoot":"","sources":["../../src/infiniteQuery.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n coerceQuery,\n QueryValidationError,\n type InstantCoreDatabase,\n type ValidQuery,\n} from './index.ts';\nimport {\n InstaQLResponse,\n InstaQLOptions,\n Cursor,\n Order,\n} from './queryTypes.ts';\nimport { InstantSchemaDef } from './schemaTypes.ts';\nimport { assert } from './utils/error.ts';\n\n// Example for {order: {value: \"asc\"}}\n//\n// 0\n// <------------------|------------------------------------------------------>\n// <- starter sub ->\n//\n// Bootstrap phase: until the limit (4 in this example) items are reached, the\n// starter subscription is the only subscription and it writes to the forwardChunks map with the key PRE_BOOTSTRAP_CURSOR.\n//\n// When the limit is reached it automatically becomes a real forward chunk and has a definite start and end.\n// A new reverse chunk gets added to watch for any new items at the start of the list.\n//\n// 0 1 2 3\n// <------------------|------------------------------------------------------>\n// <- starter sub ->\n//\n// ↓ BECOMES ↓\n//\n// 0 1 2 3\n// <------------------|------------------------------------------------------>\n// <-reverse chunk][forward chunk ]\n//\n// 0 1 2 3 4\n// <------------------|------------------------------------------------------>\n// <-reverse chunk][forward chunk ]\n// When item 4 is added, the forward chunk subscription gets updated so that\n// hasNextPage is `true`. This tells the user that a new page can be loaded.\n//\n// User clicks: loadNextPage\n// 0 1 2 3 4\n// <------------------|------------------------------------------------------>\n// <-reverse chunk][ frozen forward chunk ][ new forward chunk ]\n//\n// More numbers get added\n// 0 1 2 3 4 5 6 7 8\n// <------------------|------------------------------------------------------>\n// <-reverse chunk][ frozen forward chunk ][ forward chunk ] ^\n// hasNextPage=true^\n//\n//\n// User clicks: loadNextPage\n//\n// 0 1 2 3 4 5 6 7 8\n// <------------------|------------------------------------------------------>\n// <-reverse chunk][ frozen forward chunk ][ frozen forward chunk ][ new chunk\n//\n// The reverse chunks work in the same way as the forward chunks but the order in the query is reversed.\n// When a reverse chunks recieves an update it will check to see if more can be loaded and it will\n// automatically freeze the chunk and add a new one. i.e. : works the same as if\n// loadNextPage was automatically clicked when hasNextPage became true.\n//\n// Chunks are indexed by their starting point cursor, for forward chunks this is the \"[\" point.\n// Their starting point cursor is inclusive in the query and exclusive from the following query\n\nconst makeCursorKey = (cursor: Cursor) => JSON.stringify(cursor);\nconst parseCursorKey = (cursorKey: string) => JSON.parse(cursorKey) as Cursor;\n\nexport type ChunkStatus = 'pre-bootstrap' | 'bootstrapping' | 'frozen';\ntype Chunk = {\n status: ChunkStatus;\n data: any[];\n hasMore?: boolean;\n endCursor?: Cursor;\n afterInclusive?: boolean;\n};\n\ntype ChunkWithEndCursor = Chunk & { endCursor: Cursor };\n\nconst chunkHasEndCursor = (chunk: Chunk): chunk is ChunkWithEndCursor => {\n return !!chunk.endCursor;\n};\n\nexport interface InfiniteQuerySubscription {\n unsubscribe: () => void;\n loadNextPage: () => void;\n}\n\nconst readCanLoadNextPage = (forwardChunks: Map<string, Chunk>) => {\n const chunksInOrder = Array.from(forwardChunks.values());\n if (chunksInOrder.length === 0) return false;\n return chunksInOrder[chunksInOrder.length - 1]?.hasMore || false;\n};\n\n// Chunk sub key is used to create keys to keep track of the subscriptions\n// while the chunk maps are keyed by the cursor, here we disinguish between\n// forward and reverse because the first 2 chunks will have the same starting\n// cursor.\nconst chunkSubKey = (direction: 'forward' | 'reverse', cursor: Cursor) =>\n `${direction}:${JSON.stringify(cursor)}`;\n\nconst reverseOrder = <\n Schema extends InstantSchemaDef<any, any, any>,\n Entity extends keyof Schema['entities'],\n>(\n order?: Order<Schema, Entity>,\n): Order<Schema, Entity> => {\n if (!order) {\n return {\n serverCreatedAt: 'asc',\n } satisfies Order<Schema, Entity>;\n }\n const key = Object.keys(order).at(0);\n if (!key) {\n return {\n serverCreatedAt: 'asc',\n } satisfies Order<Schema, Entity>;\n }\n return {\n [key]: order[key as keyof typeof order] === 'asc' ? 'desc' : 'asc',\n } as Order<Schema, Entity>;\n};\n\nconst normalizeChunks = (\n forwardChunks: Map<string, Chunk>,\n reverseChunks: Map<string, Chunk>,\n): { chunks: Chunk[]; data: any[] } => {\n const chunks = [\n ...Array.from(reverseChunks.values()).slice().reverse(),\n ...Array.from(forwardChunks.values()),\n ];\n\n const data = [\n ...Array.from(reverseChunks.values())\n .slice()\n .reverse()\n .flatMap((chunk) => chunk.data.slice().reverse()),\n ...Array.from(forwardChunks.values()).flatMap((chunk) => chunk.data),\n ];\n return { chunks, data };\n};\n\nconst PRE_BOOTSTRAP_CURSOR: Cursor = ['bootstrap', 'bootstrap', 'bootstrap', 1];\n\nexport type InfiniteQueryCallbackResponse<\n Schema extends InstantSchemaDef<any, any, any>,\n Query extends Record<string, any>,\n UseDatesLocal extends boolean,\n> =\n | {\n error: { message: string };\n data: undefined;\n canLoadNextPage: boolean;\n }\n | {\n error: undefined;\n data: InstaQLResponse<Schema, Query, UseDatesLocal>;\n canLoadNextPage: boolean;\n };\n\nexport const subscribeInfiniteQuery = <\n Schema extends InstantSchemaDef<any, any, any>,\n Q extends ValidQuery<Q, Schema>,\n UseDates extends boolean,\n>(\n db: InstantCoreDatabase<Schema, UseDates>,\n fullQuery: Q,\n cb: (resp: InfiniteQueryCallbackResponse<Schema, Q, UseDates>) => void,\n opts?: InstaQLOptions,\n): InfiniteQuerySubscription => {\n const { entityName, entityQuery: query } = splitAndValidateQuery(fullQuery);\n\n const pageSize = query.$?.limit || 10;\n const entity = entityName;\n\n const forwardChunks = new Map<string, Chunk>();\n const reverseChunks = new Map<string, Chunk>();\n // Keeps track of all subscriptions (besides starter sub)\n const allUnsubs = new Map<string, () => void>();\n\n let hasKickstarted = false;\n let isActive = true;\n let lastReverseAdvancedChunkKey: string | null = null;\n let starterUnsub: (() => void) | null = null;\n\n const sendError = (err: { message: string }) => {\n cb({ error: err, data: undefined, canLoadNextPage: false });\n };\n\n const pushUpdate = () => {\n if (!isActive) return;\n\n const { chunks, data } = normalizeChunks(forwardChunks, reverseChunks);\n cb({\n data: { [entity]: data } as InstaQLResponse<\n Schema,\n typeof query,\n UseDates\n >,\n // @ts-expect-error hidden debug variable\n chunks,\n canLoadNextPage: readCanLoadNextPage(forwardChunks),\n });\n };\n\n const setForwardChunk = (startCursor: Cursor, chunk: Chunk) => {\n forwardChunks.set(makeCursorKey(startCursor), chunk);\n pushUpdate();\n };\n\n const setReverseChunk = (startCursor: Cursor, chunk: Chunk) => {\n reverseChunks.set(makeCursorKey(startCursor), chunk);\n maybeAdvanceReverse();\n pushUpdate();\n };\n\n const freezeReverse = (chunkKey: string, chunk: ChunkWithEndCursor) => {\n const startCursor = parseCursorKey(chunkKey);\n const currentSub = allUnsubs.get(chunkSubKey('reverse', startCursor));\n currentSub?.();\n\n const nextSub = db.subscribeQuery(\n {\n [entity]: {\n ...query,\n $: {\n after: startCursor,\n before: chunk.endCursor,\n beforeInclusive: true,\n where: query.$?.where,\n fields: query.$?.fields,\n order: reverseOrder(query.$?.order),\n },\n },\n } as unknown as Q,\n (frozenData) => {\n if (frozenData.error) {\n return sendError(frozenData.error);\n }\n\n const rows = frozenData.data[entity];\n const pageInfo = frozenData.pageInfo[entity];\n assert(\n rows && pageInfo,\n 'Expected query subscription to contain rows and pageInfo',\n );\n\n setReverseChunk(startCursor, {\n data: rows,\n status: 'frozen',\n hasMore: pageInfo.hasNextPage,\n endCursor: pageInfo.endCursor,\n });\n },\n opts,\n );\n\n allUnsubs.set(chunkSubKey('reverse', startCursor), nextSub);\n };\n\n const pushNewReverse = (startCursor: Cursor) => {\n const querySub = db.subscribeQuery(\n {\n [entity]: {\n ...query,\n $: {\n limit: pageSize,\n after: startCursor,\n where: query.$?.where,\n fields: query.$?.fields,\n order: reverseOrder(query.$?.order),\n },\n },\n } as unknown as Q,\n (windowData) => {\n if (windowData.error) {\n return sendError(windowData.error);\n }\n\n const rows = windowData.data[entity];\n const pageInfo = windowData.pageInfo[entity];\n assert(rows && pageInfo, 'Expected rows and pageInfo');\n\n setReverseChunk(startCursor, {\n data: rows,\n status: 'bootstrapping',\n hasMore: pageInfo.hasNextPage,\n endCursor: pageInfo.endCursor,\n });\n },\n opts,\n );\n\n allUnsubs.set(chunkSubKey('reverse', startCursor), querySub);\n };\n\n const pushNewForward = (startCursor: Cursor, afterInclusive = false) => {\n const querySub = db.subscribeQuery(\n {\n [entity]: {\n ...query,\n $: {\n limit: pageSize,\n after: startCursor,\n afterInclusive,\n where: query.$?.where,\n fields: query.$?.fields,\n order: query.$?.order,\n },\n },\n } as unknown as Q,\n (windowData) => {\n if (windowData.error) {\n return sendError(windowData.error);\n }\n\n const rows = windowData.data[entity];\n const pageInfo = windowData.pageInfo[entity];\n assert(rows && pageInfo, 'Page info and rows');\n\n setForwardChunk(startCursor, {\n data: rows,\n status: 'bootstrapping',\n hasMore: pageInfo.hasNextPage,\n endCursor: pageInfo.endCursor,\n afterInclusive,\n });\n },\n opts,\n );\n\n allUnsubs.set(chunkSubKey('forward', startCursor), querySub);\n };\n\n const freezeForward = (startCursor: Cursor) => {\n const key = makeCursorKey(startCursor);\n const currentSub = allUnsubs.get(chunkSubKey('forward', startCursor));\n currentSub?.();\n\n const chunk = forwardChunks.get(key);\n if (!chunk?.endCursor) return;\n\n const nextSub = db.subscribeQuery(\n {\n [entity]: {\n ...query,\n $: {\n after: startCursor,\n afterInclusive: chunk.afterInclusive,\n before: chunk.endCursor,\n beforeInclusive: true,\n where: query.$?.where,\n fields: query.$?.fields,\n order: query.$?.order,\n },\n },\n } as unknown as Q,\n (frozenData) => {\n if (frozenData.error) {\n return sendError(frozenData.error);\n }\n\n const rows = frozenData.data[entity];\n const pageInfo = frozenData.pageInfo[entity];\n assert(rows && pageInfo, 'Expected rows and pageInfo');\n\n setForwardChunk(startCursor, {\n data: rows,\n status: 'frozen',\n hasMore: pageInfo.hasNextPage,\n endCursor: pageInfo.endCursor,\n afterInclusive: chunk.afterInclusive,\n });\n },\n opts,\n );\n\n allUnsubs.set(chunkSubKey('forward', startCursor), nextSub);\n };\n\n // Consider order: {val: \"asc\"} with pageItems = 4\n // A reverse chunk captures all the new items coming in before us.\n // If we hit 4 then we freeze the current chunk and create a new reverse chunk\n const maybeAdvanceReverse = () => {\n const tailEntry = Array.from(reverseChunks.entries()).at(-1);\n if (!tailEntry) return;\n\n const [chunkKey, chunk] = tailEntry;\n\n // If a chunk has more, then it must have an endCursor\n if (!chunk?.hasMore) return;\n if (!chunkHasEndCursor(chunk)) return;\n\n // maybeAdvanceReverse can run multiple times if multiple changes are made\n // to the reverse chunk\n // This prevents adding the same new reverse frame twice\n const advanceKey = `${chunkKey}:${makeCursorKey(chunk.endCursor)}`;\n if (advanceKey == lastReverseAdvancedChunkKey) return;\n lastReverseAdvancedChunkKey = advanceKey;\n\n freezeReverse(chunkKey, chunk);\n pushNewReverse(chunk.endCursor);\n };\n\n const loadNextPage = () => {\n const tailEntry = Array.from(forwardChunks.entries()).at(-1);\n if (!tailEntry) return;\n\n const [chunkKey, chunk] = tailEntry;\n\n // If the chunk has more items after it, it must have an end cursor, and we can\n // load more items\n // if (!chunk?.hasMore) return;\n if (!chunk.endCursor) return;\n\n freezeForward(parseCursorKey(chunkKey));\n pushNewForward(chunk.endCursor);\n };\n\n starterUnsub = db.subscribeQuery(\n {\n [entity]: {\n ...query,\n $: {\n limit: pageSize,\n where: query.$?.where,\n fields: query.$?.fields,\n order: query.$?.order,\n },\n },\n } as unknown as Q,\n async (starterData) => {\n if (hasKickstarted) return;\n if (starterData.error) {\n return sendError(starterData.error);\n }\n const pageInfo = starterData.pageInfo[entity];\n\n const rows = starterData?.data?.[entity];\n assert(rows && pageInfo, 'Expected rows and pageInfo');\n\n if (rows.length < pageSize) {\n // If the rows are less than the page size, then we don't need to\n // create forward and reverse chunks.\n // We just treat the starter query as a forward chunk\n setForwardChunk(PRE_BOOTSTRAP_CURSOR, {\n data: rows,\n status: 'pre-bootstrap',\n });\n return;\n }\n\n // Consider a query with no items; the server will return a result with\n // no start cursor. If we add {pageSize} optimistic updates we can\n // get here and still have no startCursor. By returning we are skipping\n // the optimistic update and just waiting for the result from the\n // server.\n const initialForwardCursor = pageInfo.startCursor;\n if (!initialForwardCursor) {\n return;\n }\n forwardChunks.delete(makeCursorKey(PRE_BOOTSTRAP_CURSOR));\n\n pushNewForward(initialForwardCursor, true);\n pushNewReverse(pageInfo.startCursor);\n hasKickstarted = true;\n\n // Flush the initial boostrap querysub data\n // because immediately unsubscribing will never save it for offline in idb\n await db._reactor.querySubs.flush();\n\n // Unsubscribe the starter subscription\n starterUnsub?.();\n starterUnsub = null;\n },\n opts,\n );\n\n const unsubscribe = () => {\n if (!isActive) return;\n isActive = false;\n starterUnsub?.();\n starterUnsub = null;\n for (const unsub of allUnsubs.values()) {\n unsub?.();\n }\n allUnsubs.clear();\n };\n\n return {\n unsubscribe,\n loadNextPage,\n };\n};\n\nexport const getInfiniteQueryInitialSnapshot = <\n Schema extends InstantSchemaDef<any, any, any>,\n Q extends ValidQuery<Q, Schema>,\n UseDates extends boolean,\n>(\n db: InstantCoreDatabase<Schema, UseDates>,\n fullQuery: Q | null,\n opts?: InstaQLOptions,\n):\n | InfiniteQueryCallbackResponse<Schema, Q, UseDates>\n | {\n canLoadNextPage: false;\n data: undefined;\n error: undefined;\n } => {\n if (!fullQuery) {\n return {\n canLoadNextPage: false,\n data: undefined,\n error: undefined,\n };\n }\n const { entityName, entityQuery } = splitAndValidateQuery(fullQuery);\n\n const pageSize = entityQuery.$?.limit || 10;\n\n let coercedQuery = fullQuery\n ? coerceQuery({\n [entityName]: {\n ...entityQuery,\n $: {\n limit: pageSize,\n where: entityQuery.$?.where,\n fields: entityQuery.$?.fields,\n order: entityQuery.$?.order,\n },\n },\n })\n : null;\n\n if (opts && 'ruleParams' in opts) {\n coercedQuery = {\n $$ruleParams: opts.ruleParams,\n ...fullQuery,\n };\n }\n const queryResult = db._reactor.getPreviousResult(coercedQuery);\n\n return {\n canLoadNextPage: false,\n data: queryResult?.data || undefined,\n error: undefined,\n };\n};\n\n/**\n * @throws QueryValidationError\n * @param fullQuery a ValidQuery with one key (entity)\n */\nconst splitAndValidateQuery = (fullQuery: Record<string, any>) => {\n const entityNames = Object.keys(fullQuery);\n if (entityNames.length !== 1) {\n throw new QueryValidationError(\n 'subscribeInfiniteQuery expects exactly one entity',\n );\n }\n\n const [entityName, entityQuery] = Object.entries(fullQuery)[0];\n\n if (!entityName || !entityQuery) {\n throw new QueryValidationError('No query provided for infinite query');\n }\n return { entityName, entityQuery };\n};\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"instaql.d.ts","sourceRoot":"","sources":["../../src/instaql.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,CAAC,MAAM,YAAY,CAAC;AAqlBhC,wBAAgB,YAAY,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,KAAA,EAAE,IAAI,EAAE,MAAM,EAAE,GAAG,KAAA,EAAE,QAAQ,KAAA,UAa1E;AAyRD,MAAM,CAAC,OAAO,UAAU,KAAK,CAC3B,EACE,KAAK,EACL,UAAU,EACV,QAAQ,EACR,SAAS,GACV,EAAE;IACD,KAAK,EAAE,CAAC,CAAC,KAAK,CAAC;IACf,UAAU,EAAE,CAAC,CAAC,UAAU,CAAC;IACzB,QAAQ,CAAC,EAAE,GAAG,CAAC;IACf,SAAS,CAAC,EAAE,GAAG,CAAC;CACjB,EACD,CAAC,KAAA;UAiBqB,GAAG;eAAa,GAAG;gBAAc,GAAG;EAU3D"}
1
+ {"version":3,"file":"instaql.d.ts","sourceRoot":"","sources":["../../src/instaql.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,CAAC,MAAM,YAAY,CAAC;AAqlBhC,wBAAgB,YAAY,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,KAAA,EAAE,IAAI,EAAE,MAAM,EAAE,GAAG,KAAA,EAAE,QAAQ,KAAA,UAa1E;AA2SD,MAAM,CAAC,OAAO,UAAU,KAAK,CAC3B,EACE,KAAK,EACL,UAAU,EACV,QAAQ,EACR,SAAS,GACV,EAAE;IACD,KAAK,EAAE,CAAC,CAAC,KAAK,CAAC;IACf,UAAU,EAAE,CAAC,CAAC,UAAU,CAAC;IACzB,QAAQ,CAAC,EAAE,GAAG,CAAC;IACf,SAAS,CAAC,EAAE,GAAG,CAAC;CACjB,EACD,CAAC,KAAA;UAiBqB,GAAG;eAAa,GAAG;gBAAc,GAAG;EAU3D"}
@@ -423,17 +423,24 @@ function comparableDate(x) {
423
423
  }
424
424
  return new Date(x).getTime();
425
425
  }
426
- function isBefore(startCursor, orderAttr, direction, idVec) {
427
- const [c_e, _c_a, c_v, c_t] = startCursor;
428
- const compareVal = direction === 'desc' ? 1 : -1;
426
+ function compareToCursor(cursor, orderAttr, idVec) {
427
+ const [c_e, _c_a, c_v, c_t] = cursor;
429
428
  if (orderAttr['forward-identity']?.[2] === 'id') {
430
- return compareOrderTriples(idVec, [c_e, c_t], null) === compareVal;
429
+ return compareOrderTriples(idVec, [c_e, c_t], null);
431
430
  }
432
431
  const [e, v] = idVec;
433
432
  const dataType = orderAttr['checked-data-type'];
434
433
  const v_new = dataType === 'date' ? comparableDate(v) : v;
435
434
  const c_v_new = dataType === 'date' ? comparableDate(c_v) : c_v;
436
- return (compareOrderTriples([e, v_new], [c_e, c_v_new], dataType) === compareVal);
435
+ return compareOrderTriples([e, v_new], [c_e, c_v_new], dataType);
436
+ }
437
+ function isBefore(startCursor, orderAttr, direction, idVec) {
438
+ const cmp = compareToCursor(startCursor, orderAttr, idVec);
439
+ return direction === 'desc' ? cmp > 0 : cmp < 0;
440
+ }
441
+ function isAfter(endCursor, orderAttr, direction, idVec) {
442
+ const cmp = compareToCursor(endCursor, orderAttr, idVec);
443
+ return direction === 'desc' ? cmp < 0 : cmp > 0;
437
444
  }
438
445
  function orderAttrFromCursor(attrsStore, cursor) {
439
446
  const cursorAttrId = cursor[1];
@@ -479,6 +486,7 @@ function runDataloadAndReturnObjects(store, attrsStore, { etype, pageInfo, dq, f
479
486
  const direction = determineDirection(form);
480
487
  let idVecs = (0, datalog_js_1.query)(store, dq);
481
488
  const startCursor = pageInfo?.['start-cursor'];
489
+ const endCursor = pageInfo?.['end-cursor'];
482
490
  const orderAttr = getOrderAttr(attrsStore, etype, startCursor, order);
483
491
  if (orderAttr && orderAttr?.['forward-identity']?.[2] !== 'id') {
484
492
  const isDate = orderAttr['checked-data-type'] === 'date';
@@ -513,6 +521,11 @@ function runDataloadAndReturnObjects(store, attrsStore, { etype, pageInfo, dq, f
513
521
  isBefore(startCursor, orderAttr, direction, idVec)) {
514
522
  continue;
515
523
  }
524
+ if (endCursor &&
525
+ orderAttr &&
526
+ isAfter(endCursor, orderAttr, direction, idVec)) {
527
+ continue;
528
+ }
516
529
  const obj = s.getAsObject(store, attrs, id);
517
530
  if (obj) {
518
531
  objects[id] = obj;