@dxos/echo-query 0.8.4-main.dedc0f3 → 0.8.4-main.e00bdcdb52

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 (63) hide show
  1. package/README.md +1 -1
  2. package/dist/lib/neutral/index.mjs +917 -0
  3. package/dist/lib/neutral/index.mjs.map +7 -0
  4. package/dist/lib/neutral/meta.json +1 -0
  5. package/dist/query-lite/index.d.ts +9975 -0
  6. package/dist/query-lite/index.d.ts.map +1 -0
  7. package/dist/query-lite/index.js +548 -0
  8. package/dist/query-lite/index.js.map +1 -0
  9. package/dist/types/src/index.d.ts +2 -0
  10. package/dist/types/src/index.d.ts.map +1 -1
  11. package/dist/types/src/parser/gen/index.d.ts +8 -0
  12. package/dist/types/src/parser/gen/index.d.ts.map +1 -0
  13. package/dist/types/src/parser/gen/query.d.ts +3 -0
  14. package/dist/types/src/parser/gen/query.d.ts.map +1 -0
  15. package/dist/types/src/parser/gen/query.terms.d.ts +2 -0
  16. package/dist/types/src/parser/gen/query.terms.d.ts.map +1 -0
  17. package/dist/types/src/parser/index.d.ts +3 -0
  18. package/dist/types/src/parser/index.d.ts.map +1 -0
  19. package/dist/types/src/parser/query-builder.d.ts +89 -0
  20. package/dist/types/src/parser/query-builder.d.ts.map +1 -0
  21. package/dist/types/src/parser/query.test.d.ts +2 -0
  22. package/dist/types/src/parser/query.test.d.ts.map +1 -0
  23. package/dist/types/src/query-lite/index.d.ts +2 -0
  24. package/dist/types/src/query-lite/index.d.ts.map +1 -0
  25. package/dist/types/src/query-lite/query-lite.d.ts +8 -0
  26. package/dist/types/src/query-lite/query-lite.d.ts.map +1 -0
  27. package/dist/types/src/sandbox/index.d.ts +2 -0
  28. package/dist/types/src/sandbox/index.d.ts.map +1 -0
  29. package/dist/types/src/sandbox/query-sandbox.d.ts +21 -0
  30. package/dist/types/src/sandbox/query-sandbox.d.ts.map +1 -0
  31. package/dist/types/src/sandbox/query-sandbox.test.d.ts +2 -0
  32. package/dist/types/src/sandbox/query-sandbox.test.d.ts.map +1 -0
  33. package/dist/types/src/sandbox/quickjs.d.ts +8 -0
  34. package/dist/types/src/sandbox/quickjs.d.ts.map +1 -0
  35. package/dist/types/src/sandbox/quickjs.test.d.ts +2 -0
  36. package/dist/types/src/sandbox/quickjs.test.d.ts.map +1 -0
  37. package/dist/types/tsconfig.tsbuildinfo +1 -1
  38. package/package.json +29 -13
  39. package/src/env.d.ts +8 -0
  40. package/src/index.ts +3 -0
  41. package/src/parser/gen/index.ts +13 -0
  42. package/src/parser/gen/query.terms.ts +27 -0
  43. package/src/parser/gen/query.ts +18 -0
  44. package/src/parser/index.ts +6 -0
  45. package/src/parser/query-builder.ts +799 -0
  46. package/src/parser/query.grammar +130 -0
  47. package/src/parser/query.test.ts +529 -0
  48. package/src/query-lite/index.ts +5 -0
  49. package/src/query-lite/query-lite.ts +744 -0
  50. package/src/sandbox/index.ts +5 -0
  51. package/src/sandbox/query-sandbox.test.ts +53 -0
  52. package/src/sandbox/query-sandbox.ts +72 -0
  53. package/src/sandbox/quickjs.test.ts +67 -0
  54. package/src/sandbox/quickjs.ts +33 -0
  55. package/dist/lib/browser/index.mjs +0 -2
  56. package/dist/lib/browser/index.mjs.map +0 -7
  57. package/dist/lib/browser/meta.json +0 -1
  58. package/dist/lib/node-esm/index.mjs +0 -2
  59. package/dist/lib/node-esm/index.mjs.map +0 -7
  60. package/dist/lib/node-esm/meta.json +0 -1
  61. package/dist/types/src/search.test.d.ts +0 -2
  62. package/dist/types/src/search.test.d.ts.map +0 -1
  63. package/src/search.test.ts +0 -49
@@ -0,0 +1,744 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import type * as Schema from 'effect/Schema';
6
+
7
+ import type { Filter as Filter$, Order as Order$, Query as Query$, Ref } from '@dxos/echo';
8
+ import type { ForeignKey, QueryAST } from '@dxos/echo-protocol';
9
+ import { assertArgument } from '@dxos/invariant';
10
+ import type { DXN, ObjectId } from '@dxos/keys';
11
+
12
+ //
13
+ // Light-weight implementation of query execution.
14
+ //
15
+
16
+ // TODO(wittjosiah): The `export * as ...` syntax causes tsdown to genereate multiple files which breaks the sandbox.
17
+
18
+ class OrderClass implements Order$.Any {
19
+ private static 'variance': Order$.Any['~Order'] = {} as Order$.Any['~Order'];
20
+
21
+ static is(value: unknown): value is Order$.Any {
22
+ return typeof value === 'object' && value !== null && '~Order' in value;
23
+ }
24
+
25
+ constructor(public readonly ast: QueryAST.Order) {}
26
+
27
+ '~Order' = OrderClass.variance;
28
+ }
29
+
30
+ namespace Order1 {
31
+ export const natural: Order$.Any = new OrderClass({ kind: 'natural' });
32
+ export const property = <T>(property: keyof T & string, direction: QueryAST.OrderDirection): Order$.Order<T> =>
33
+ new OrderClass({
34
+ kind: 'property',
35
+ property,
36
+ direction,
37
+ });
38
+ export const rank = <T>(direction: QueryAST.OrderDirection = 'desc'): Order$.Order<T> =>
39
+ new OrderClass({
40
+ kind: 'rank',
41
+ direction,
42
+ });
43
+ }
44
+
45
+ const Order2: typeof Order$ = Order1;
46
+ export { Order2 as Order };
47
+
48
+ class FilterClass implements Filter$.Any {
49
+ private static 'variance': Filter$.Any['~Filter'] = {} as Filter$.Any['~Filter'];
50
+
51
+ static is(value: unknown): value is Filter$.Any {
52
+ return typeof value === 'object' && value !== null && '~Filter' in value;
53
+ }
54
+
55
+ static fromAst(ast: QueryAST.Filter): Filter$.Any {
56
+ return new FilterClass(ast);
57
+ }
58
+
59
+ static everything(): FilterClass {
60
+ return new FilterClass({
61
+ type: 'object',
62
+ typename: null,
63
+ props: {},
64
+ });
65
+ }
66
+
67
+ static nothing(): FilterClass {
68
+ return new FilterClass({
69
+ type: 'not',
70
+ filter: {
71
+ type: 'object',
72
+ typename: null,
73
+ props: {},
74
+ },
75
+ });
76
+ }
77
+
78
+ static relation() {
79
+ return new FilterClass({
80
+ type: 'object',
81
+ typename: null,
82
+ props: {},
83
+ });
84
+ }
85
+
86
+ static id(...ids: ObjectId[]): Filter$.Any {
87
+ // assertArgument(
88
+ // ids.every((id) => ObjectId.isValid(id)),
89
+ // 'ids',
90
+ // 'ids must be valid',
91
+ // );
92
+
93
+ if (ids.length === 0) {
94
+ return FilterClass.nothing();
95
+ }
96
+
97
+ return new FilterClass({
98
+ type: 'object',
99
+ typename: null,
100
+ id: ids,
101
+ props: {},
102
+ });
103
+ }
104
+
105
+ static type<S extends Schema.Schema.All>(
106
+ schema: S | string,
107
+ props?: Filter$.Props<Schema.Schema.Type<S>>,
108
+ ): Filter$.Filter<Schema.Schema.Type<S>> {
109
+ if (typeof schema !== 'string') {
110
+ throw new TypeError('expected typename as the first paramter');
111
+ }
112
+ return new FilterClass({
113
+ type: 'object',
114
+ typename: makeTypeDxn(schema),
115
+ ...propsFilterToAst(props ?? {}),
116
+ });
117
+ }
118
+
119
+ static typename(typename: string): Filter$.Any {
120
+ return new FilterClass({
121
+ type: 'object',
122
+ typename: makeTypeDxn(typename),
123
+ props: {},
124
+ });
125
+ }
126
+
127
+ static typeDXN(dxn: DXN): Filter$.Any {
128
+ return new FilterClass({
129
+ type: 'object',
130
+ typename: dxn.toString(),
131
+ props: {},
132
+ });
133
+ }
134
+
135
+ static tag(tag: string): Filter$.Any {
136
+ return new FilterClass({
137
+ type: 'tag',
138
+ tag,
139
+ });
140
+ }
141
+
142
+ static props<T>(props: Filter$.Props<T>): Filter$.Filter<T> {
143
+ return new FilterClass({
144
+ type: 'object',
145
+ typename: null,
146
+ ...propsFilterToAst(props),
147
+ });
148
+ }
149
+
150
+ static text(text: string, options?: Filter$.TextSearchOptions): Filter$.Any {
151
+ return new FilterClass({
152
+ type: 'text-search',
153
+ text,
154
+ searchKind: options?.type,
155
+ });
156
+ }
157
+
158
+ static foreignKeys<S extends Schema.Schema.All>(
159
+ schema: S | string,
160
+ keys: ForeignKey[],
161
+ ): Filter$.Filter<Schema.Schema.Type<S>> {
162
+ assertArgument(typeof schema === 'string', 'schema');
163
+ assertArgument(!schema.startsWith('dxn:'), 'schema');
164
+ return new FilterClass({
165
+ type: 'object',
166
+ typename: `dxn:type:${schema}`,
167
+ props: {},
168
+ foreignKeys: keys,
169
+ });
170
+ }
171
+
172
+ static eq<T>(value: T): Filter$.Filter<T | undefined> {
173
+ if (!isRef(value) && typeof value === 'object' && value !== null) {
174
+ throw new TypeError('Cannot use object as a value for eq filter');
175
+ }
176
+
177
+ return new FilterClass({
178
+ type: 'compare',
179
+ operator: 'eq',
180
+ value: isRef(value) ? value.noInline().encode() : value,
181
+ });
182
+ }
183
+
184
+ static neq<T>(value: T): Filter$.Filter<T | undefined> {
185
+ return new FilterClass({
186
+ type: 'compare',
187
+ operator: 'neq',
188
+ value,
189
+ });
190
+ }
191
+
192
+ static gt<T>(value: T): Filter$.Filter<T | undefined> {
193
+ return new FilterClass({
194
+ type: 'compare',
195
+ operator: 'gt',
196
+ value,
197
+ });
198
+ }
199
+
200
+ static gte<T>(value: T): Filter$.Filter<T | undefined> {
201
+ return new FilterClass({
202
+ type: 'compare',
203
+ operator: 'gte',
204
+ value,
205
+ });
206
+ }
207
+
208
+ static lt<T>(value: T): Filter$.Filter<T | undefined> {
209
+ return new FilterClass({
210
+ type: 'compare',
211
+ operator: 'lt',
212
+ value,
213
+ });
214
+ }
215
+
216
+ static lte<T>(value: T): Filter$.Filter<T | undefined> {
217
+ return new FilterClass({
218
+ type: 'compare',
219
+ operator: 'lte',
220
+ value,
221
+ });
222
+ }
223
+
224
+ static in<T>(...values: T[]): Filter$.Filter<T> {
225
+ return new FilterClass({
226
+ type: 'in',
227
+ values,
228
+ });
229
+ }
230
+
231
+ static contains<T>(value: T): Filter$.Filter<readonly T[] | undefined> {
232
+ return new FilterClass({
233
+ type: 'contains',
234
+ value,
235
+ });
236
+ }
237
+
238
+ static between<T>(from: T, to: T): Filter$.Filter<T> {
239
+ return new FilterClass({
240
+ type: 'range',
241
+ from,
242
+ to,
243
+ });
244
+ }
245
+
246
+ static updated(range: { after?: Date | number; before?: Date | number }): Filter$.Any {
247
+ return FilterClass._timeRangeFilter('updatedAt', range);
248
+ }
249
+
250
+ static created(range: { after?: Date | number; before?: Date | number }): Filter$.Any {
251
+ return FilterClass._timeRangeFilter('createdAt', range);
252
+ }
253
+
254
+ static childOf(parents: unknown | DXN | (unknown | DXN)[], options?: { transitive?: boolean }): Filter$.Any {
255
+ const items = Array.isArray(parents) ? parents : [parents];
256
+ const dxns = items.map((item) => {
257
+ if (isDxnLike(item)) {
258
+ return item.toString();
259
+ }
260
+ throw new TypeError('childOf requires DXN values in query-lite');
261
+ });
262
+ return new FilterClass({
263
+ type: 'child-of',
264
+ parents: dxns,
265
+ transitive: options?.transitive ?? true,
266
+ });
267
+ }
268
+
269
+ private static _timeRangeFilter(
270
+ field: 'updatedAt' | 'createdAt',
271
+ range: { after?: Date | number; before?: Date | number },
272
+ ): Filter$.Any {
273
+ const toMs = (d: Date | number) => (typeof d === 'number' ? d : d.getTime());
274
+ const filters: Filter$.Any[] = [];
275
+ if (range.after != null) {
276
+ filters.push(new FilterClass({ type: 'timestamp', field, operator: 'gte', value: toMs(range.after) }));
277
+ }
278
+ if (range.before != null) {
279
+ filters.push(new FilterClass({ type: 'timestamp', field, operator: 'lte', value: toMs(range.before) }));
280
+ }
281
+ if (filters.length === 0) {
282
+ return FilterClass.everything();
283
+ }
284
+ return filters.length === 1 ? filters[0] : FilterClass.and(...filters);
285
+ }
286
+
287
+ static not<F extends Filter$.Any>(filter: F): Filter$.Filter<Filter$.Type<F>> {
288
+ return new FilterClass({
289
+ type: 'not',
290
+ filter: filter.ast,
291
+ });
292
+ }
293
+
294
+ static and<Filters extends readonly Filter$.Any[]>(
295
+ ...filters: Filters
296
+ ): Filter$.Filter<Filter$.Type<Filters[number]>> {
297
+ return new FilterClass({
298
+ type: 'and',
299
+ filters: filters.map((f) => f.ast),
300
+ });
301
+ }
302
+
303
+ static or<Filters extends readonly Filter$.Any[]>(
304
+ ...filters: Filters
305
+ ): Filter$.Filter<Filter$.Type<Filters[number]>> {
306
+ return new FilterClass({
307
+ type: 'or',
308
+ filters: filters.map((f) => f.ast),
309
+ });
310
+ }
311
+
312
+ /** Returns a human-readable string representation of a Filter AST. */
313
+ static pretty(filter: Filter$.Any): string {
314
+ return prettyFilter(filter.ast);
315
+ }
316
+
317
+ private constructor(public readonly ast: QueryAST.Filter) {}
318
+
319
+ '~Filter' = FilterClass.variance;
320
+ }
321
+
322
+ export const Filter1: typeof Filter$ = FilterClass;
323
+ export { Filter1 as Filter };
324
+
325
+ /**
326
+ * All property paths inside T that are references.
327
+ */
328
+ // TODO(dmaretskyi): Filter only properties that are references (or optional references, or unions that include references).
329
+ type RefPropKey<T> = keyof T & string;
330
+
331
+ const propsFilterToAst = (predicates: Filter$.Props<any>): Pick<QueryAST.FilterObject, 'id' | 'props'> => {
332
+ let idFilter: readonly ObjectId[] | undefined;
333
+ if ('id' in predicates) {
334
+ assertArgument(
335
+ typeof predicates.id === 'string' || Array.isArray(predicates.id),
336
+ 'predicates.id',
337
+ 'invalid id filter',
338
+ );
339
+ idFilter = typeof predicates.id === 'string' ? [predicates.id] : predicates.id;
340
+ }
341
+
342
+ return {
343
+ id: idFilter,
344
+ props: Object.fromEntries(
345
+ Object.entries(predicates)
346
+ .filter(([prop, _value]) => prop !== 'id')
347
+ .map(([prop, predicate]) => [prop, processPredicate(predicate)]),
348
+ ) as Record<string, QueryAST.Filter>,
349
+ };
350
+ };
351
+
352
+ const processPredicate = (predicate: any): QueryAST.Filter => {
353
+ if (FilterClass.is(predicate)) {
354
+ return predicate.ast;
355
+ }
356
+
357
+ if (Array.isArray(predicate)) {
358
+ throw new Error('Array predicates are not yet supported.');
359
+ }
360
+
361
+ if (!isRef(predicate) && typeof predicate === 'object' && predicate !== null) {
362
+ const nestedProps = Object.fromEntries(
363
+ Object.entries(predicate).map(([key, value]) => [key, processPredicate(value)]),
364
+ );
365
+
366
+ return {
367
+ type: 'object',
368
+ typename: null,
369
+ props: nestedProps,
370
+ };
371
+ }
372
+
373
+ return FilterClass.eq(predicate).ast;
374
+ };
375
+
376
+ class QueryClass implements Query$.Any {
377
+ private static 'variance': Query$.Any['~Query'] = {} as Query$.Any['~Query'];
378
+
379
+ static is(value: unknown): value is Query$.Any {
380
+ return typeof value === 'object' && value !== null && '~Query' in value;
381
+ }
382
+
383
+ static fromAst(ast: QueryAST.Query): Query$.Any {
384
+ return new QueryClass(ast);
385
+ }
386
+
387
+ static select<F extends Filter$.Any>(filter: F): Query$.Query<Filter$.Type<F>> {
388
+ return new QueryClass({
389
+ type: 'select',
390
+ filter: filter.ast,
391
+ });
392
+ }
393
+
394
+ select(filter: Filter$.Any | Filter$.Props<any>): Query$.Any {
395
+ if (FilterClass.is(filter)) {
396
+ return new QueryClass({
397
+ type: 'filter',
398
+ selection: this.ast,
399
+ filter: filter.ast,
400
+ });
401
+ } else {
402
+ return new QueryClass({
403
+ type: 'filter',
404
+ selection: this.ast,
405
+ filter: FilterClass.props(filter).ast,
406
+ });
407
+ }
408
+ }
409
+
410
+ static type(schema: Schema.Schema.All | string, predicates?: Filter$.Props<unknown>): Query$.Any {
411
+ return new QueryClass({
412
+ type: 'select',
413
+ filter: FilterClass.type(schema, predicates).ast,
414
+ });
415
+ }
416
+
417
+ static all(...queries: Query$.Any[]): Query$.Any {
418
+ if (queries.length === 0) {
419
+ throw new TypeError(
420
+ 'Query.all combines results of multiple queries, to query all objects use Query.select(Filter.everything())',
421
+ );
422
+ }
423
+ return new QueryClass({
424
+ type: 'union',
425
+ queries: queries.map((q) => q.ast),
426
+ });
427
+ }
428
+
429
+ static without<T>(source: Query$.Query<T>, exclude: Query$.Query<T>): Query$.Query<T> {
430
+ return new QueryClass({
431
+ type: 'set-difference',
432
+ source: source.ast,
433
+ exclude: exclude.ast,
434
+ });
435
+ }
436
+
437
+ static from(source: any, options?: { includeFeeds?: boolean }): Query$.Any {
438
+ const baseQuery: QueryAST.Query = {
439
+ type: 'select',
440
+ filter: FilterClass.everything().ast,
441
+ };
442
+ const wrapper = new QueryClass(baseQuery);
443
+ return wrapper.from(source, options);
444
+ }
445
+
446
+ from(arg: any, options?: { includeFeeds?: boolean }): Query$.Any {
447
+ if (arg === 'all-accessible-spaces') {
448
+ return new QueryClass({
449
+ type: 'from',
450
+ query: this.ast,
451
+ from: {
452
+ _tag: 'scope',
453
+ scope: {
454
+ ...(options?.includeFeeds ? { allQueuesFromSpaces: true } : {}),
455
+ },
456
+ },
457
+ });
458
+ }
459
+
460
+ if (_isScopeLike(arg)) {
461
+ return new QueryClass({
462
+ type: 'from',
463
+ query: this.ast,
464
+ from: { _tag: 'scope', scope: arg },
465
+ });
466
+ }
467
+
468
+ throw new TypeError('Database and Feed objects are not supported in query-lite sandbox');
469
+ }
470
+
471
+ /** Returns a human-readable string representation of a Query AST. */
472
+ static pretty(query: Query$.Any): string {
473
+ return prettyQuery(query.ast);
474
+ }
475
+
476
+ constructor(public readonly ast: QueryAST.Query) {}
477
+
478
+ '~Query' = QueryClass.variance;
479
+
480
+ reference(key: string): Query$.Any {
481
+ return new QueryClass({
482
+ type: 'reference-traversal',
483
+ anchor: this.ast,
484
+ property: key,
485
+ });
486
+ }
487
+
488
+ referencedBy(target?: Schema.Schema.All | string, key?: string): Query$.Any {
489
+ const typename =
490
+ target !== undefined
491
+ ? (assertArgument(typeof target === 'string', 'target'),
492
+ assertArgument(!target.startsWith('dxn:'), 'target'),
493
+ target)
494
+ : null;
495
+ return new QueryClass({
496
+ type: 'incoming-references',
497
+ anchor: this.ast,
498
+ property: key ?? null,
499
+ typename,
500
+ });
501
+ }
502
+
503
+ sourceOf(relation?: Schema.Schema.All | string, predicates?: Filter$.Props<unknown> | undefined): Query$.Any {
504
+ return new QueryClass({
505
+ type: 'relation',
506
+ anchor: this.ast,
507
+ direction: 'outgoing',
508
+ filter: relation !== undefined ? FilterClass.type(relation, predicates).ast : undefined,
509
+ });
510
+ }
511
+
512
+ targetOf(relation?: Schema.Schema.All | string, predicates?: Filter$.Props<unknown> | undefined): Query$.Any {
513
+ return new QueryClass({
514
+ type: 'relation',
515
+ anchor: this.ast,
516
+ direction: 'incoming',
517
+ filter: relation !== undefined ? FilterClass.type(relation, predicates).ast : undefined,
518
+ });
519
+ }
520
+
521
+ source(): Query$.Any {
522
+ return new QueryClass({
523
+ type: 'relation-traversal',
524
+ anchor: this.ast,
525
+ direction: 'source',
526
+ });
527
+ }
528
+
529
+ target(): Query$.Any {
530
+ return new QueryClass({
531
+ type: 'relation-traversal',
532
+ anchor: this.ast,
533
+ direction: 'target',
534
+ });
535
+ }
536
+
537
+ parent(): Query$.Any {
538
+ return new QueryClass({
539
+ type: 'hierarchy-traversal',
540
+ anchor: this.ast,
541
+ direction: 'to-parent',
542
+ });
543
+ }
544
+
545
+ children(): Query$.Any {
546
+ return new QueryClass({
547
+ type: 'hierarchy-traversal',
548
+ anchor: this.ast,
549
+ direction: 'to-children',
550
+ });
551
+ }
552
+
553
+ orderBy(...order: Order$.Any[]): Query$.Any {
554
+ return new QueryClass({
555
+ type: 'order',
556
+ query: this.ast,
557
+ order: order.map((o) => o.ast),
558
+ });
559
+ }
560
+
561
+ limit(limit: number): Query$.Any {
562
+ return new QueryClass({
563
+ type: 'limit',
564
+ query: this.ast,
565
+ limit,
566
+ });
567
+ }
568
+
569
+ options(options: QueryAST.QueryOptions): Query$.Any {
570
+ return new QueryClass({
571
+ type: 'options',
572
+ query: this.ast,
573
+ options,
574
+ });
575
+ }
576
+
577
+ debugLabel(label: string): Query$.Any {
578
+ if (this.ast.type === 'options') {
579
+ return new QueryClass({
580
+ type: 'options',
581
+ query: this.ast.query,
582
+ options: { ...this.ast.options, debugLabel: label },
583
+ });
584
+ }
585
+ return new QueryClass({
586
+ type: 'options',
587
+ query: this.ast,
588
+ options: { debugLabel: label },
589
+ });
590
+ }
591
+ }
592
+
593
+ export const Query1: typeof Query$ = QueryClass;
594
+ export { Query1 as Query };
595
+
596
+ const RefTypeId: unique symbol = Symbol('@dxos/echo-query/Ref');
597
+ const isRef = (obj: any): obj is Ref.Ref<any> => {
598
+ return obj && typeof obj === 'object' && RefTypeId in obj;
599
+ };
600
+
601
+ const makeTypeDxn = (typename: string) => {
602
+ assertArgument(typeof typename === 'string', 'typename');
603
+ assertArgument(!typename.startsWith('dxn:'), 'typename');
604
+ return `dxn:type:${typename}`;
605
+ };
606
+
607
+ const isDxnLike = (value: unknown): value is DXN => {
608
+ return (
609
+ typeof value === 'object' &&
610
+ value !== null &&
611
+ 'toString' in value &&
612
+ typeof value.toString === 'function' &&
613
+ value.toString().startsWith('dxn:')
614
+ );
615
+ };
616
+
617
+ const SCOPE_KEYS = new Set(['spaceIds', 'queues', 'allQueuesFromSpaces']);
618
+
619
+ const _isScopeLike = (value: unknown): value is QueryAST.Scope => {
620
+ if (typeof value !== 'object' || value === null || Array.isArray(value)) {
621
+ return false;
622
+ }
623
+ return Object.keys(value).every((key) => SCOPE_KEYS.has(key));
624
+ };
625
+
626
+ const prettyFilter = (filter: QueryAST.Filter): string => {
627
+ switch (filter.type) {
628
+ case 'object': {
629
+ const parts: string[] = [];
630
+ if (filter.typename !== null) {
631
+ parts.push(JSON.stringify(filter.typename));
632
+ }
633
+ const propEntries = Object.entries(filter.props);
634
+ if (propEntries.length > 0) {
635
+ const propsStr = propEntries.map(([k, v]) => `${k}: ${prettyFilter(v)}`).join(', ');
636
+ parts.push(`{ ${propsStr} }`);
637
+ }
638
+ if (filter.id !== undefined) {
639
+ parts.push(`id: [${filter.id.join(', ')}]`);
640
+ }
641
+ return parts.length > 0 ? `Filter.type(${parts.join(', ')})` : 'Filter.everything()';
642
+ }
643
+ case 'compare':
644
+ return `Filter.${filter.operator}(${JSON.stringify(filter.value)})`;
645
+ case 'in':
646
+ return `Filter.in(${filter.values.map((v) => JSON.stringify(v)).join(', ')})`;
647
+ case 'contains':
648
+ return `Filter.contains(${JSON.stringify(filter.value)})`;
649
+ case 'range':
650
+ return `Filter.between(${JSON.stringify(filter.from)}, ${JSON.stringify(filter.to)})`;
651
+ case 'text-search':
652
+ return `Filter.text(${JSON.stringify(filter.text)})`;
653
+ case 'tag':
654
+ return `Filter.tag(${JSON.stringify(filter.tag)})`;
655
+ case 'child-of':
656
+ return `Filter.childOf([${filter.parents.map((parent) => JSON.stringify(parent)).join(', ')}], { transitive: ${filter.transitive} })`;
657
+ case 'timestamp':
658
+ return `Filter.${filter.field}.${filter.operator}(${filter.value})`;
659
+ case 'not':
660
+ return `Filter.not(${prettyFilter(filter.filter)})`;
661
+ case 'and':
662
+ return `Filter.and(${filter.filters.map(prettyFilter).join(', ')})`;
663
+ case 'or':
664
+ return `Filter.or(${filter.filters.map(prettyFilter).join(', ')})`;
665
+ }
666
+ };
667
+
668
+ const prettyQuery = (query: QueryAST.Query): string => {
669
+ switch (query.type) {
670
+ case 'select':
671
+ return `Query.select(${prettyFilter(query.filter)})`;
672
+ case 'filter':
673
+ return `${prettyQuery(query.selection)}.select(${prettyFilter(query.filter)})`;
674
+ case 'reference-traversal':
675
+ return `${prettyQuery(query.anchor)}.reference(${JSON.stringify(query.property)})`;
676
+ case 'incoming-references': {
677
+ const args: string[] = [];
678
+ if (query.typename !== null) {
679
+ args.push(JSON.stringify(query.typename));
680
+ }
681
+ if (query.property !== null) {
682
+ args.push(JSON.stringify(query.property));
683
+ }
684
+ return `${prettyQuery(query.anchor)}.referencedBy(${args.join(', ')})`;
685
+ }
686
+ case 'relation': {
687
+ const method =
688
+ query.direction === 'outgoing' ? 'sourceOf' : query.direction === 'incoming' ? 'targetOf' : 'relationOf';
689
+ const filterStr = query.filter !== undefined ? prettyFilter(query.filter) : '';
690
+ return `${prettyQuery(query.anchor)}.${method}(${filterStr})`;
691
+ }
692
+ case 'relation-traversal':
693
+ return `${prettyQuery(query.anchor)}.${query.direction}()`;
694
+ case 'hierarchy-traversal':
695
+ return query.direction === 'to-parent'
696
+ ? `${prettyQuery(query.anchor)}.parent()`
697
+ : `${prettyQuery(query.anchor)}.children()`;
698
+ case 'union':
699
+ return `Query.all(${query.queries.map(prettyQuery).join(', ')})`;
700
+ case 'set-difference':
701
+ return `Query.without(${prettyQuery(query.source)}, ${prettyQuery(query.exclude)})`;
702
+ case 'order': {
703
+ const orders = query.order.map((o) => {
704
+ if (o.kind === 'natural') {
705
+ return 'Order.natural';
706
+ }
707
+ if (o.kind === 'rank') {
708
+ return `Order.rank(${JSON.stringify(o.direction)})`;
709
+ }
710
+ return `Order.property(${JSON.stringify(o.property)}, ${JSON.stringify(o.direction)})`;
711
+ });
712
+ return `${prettyQuery(query.query)}.orderBy(${orders.join(', ')})`;
713
+ }
714
+ case 'options': {
715
+ const parts: string[] = [];
716
+ if (query.options.deleted !== undefined) {
717
+ parts.push(`deleted: ${JSON.stringify(query.options.deleted)}`);
718
+ }
719
+ if (query.options.debugLabel !== undefined) {
720
+ parts.push(`debugLabel: ${JSON.stringify(query.options.debugLabel)}`);
721
+ }
722
+ return `${prettyQuery(query.query)}.options({ ${parts.join(', ')} })`;
723
+ }
724
+ case 'from': {
725
+ if (query.from._tag === 'scope') {
726
+ const scope = query.from.scope;
727
+ const parts: string[] = [];
728
+ if (scope.spaceIds !== undefined) {
729
+ parts.push(`spaceIds: [${scope.spaceIds.join(', ')}]`);
730
+ }
731
+ if (scope.queues !== undefined) {
732
+ parts.push(`queues: [${scope.queues.join(', ')}]`);
733
+ }
734
+ if (scope.allQueuesFromSpaces !== undefined) {
735
+ parts.push(`allQueuesFromSpaces: ${scope.allQueuesFromSpaces}`);
736
+ }
737
+ return `${prettyQuery(query.query)}.from({ ${parts.join(', ')} })`;
738
+ }
739
+ return `${prettyQuery(query.query)}.from(${prettyQuery(query.from.query)})`;
740
+ }
741
+ case 'limit':
742
+ return `${prettyQuery(query.query)}.limit(${query.limit})`;
743
+ }
744
+ };