@dxos/echo-query 0.8.4-main.bc674ce → 0.8.4-main.bcb3aa67d6

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/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@dxos/echo-query",
3
- "version": "0.8.4-main.bc674ce",
3
+ "version": "0.8.4-main.bcb3aa67d6",
4
4
  "description": "ECHO queries.",
5
5
  "homepage": "https://dxos.org",
6
6
  "bugs": "https://github.com/dxos/dxos/issues",
@@ -19,8 +19,7 @@
19
19
  ".": {
20
20
  "source": "./src/index.ts",
21
21
  "types": "./dist/types/src/index.d.ts",
22
- "browser": "./dist/lib/browser/index.mjs",
23
- "node": "./dist/lib/node-esm/index.mjs"
22
+ "default": "./dist/lib/neutral/index.mjs"
24
23
  },
25
24
  "./api.d.ts": "./dist/query-lite/index.d.ts"
26
25
  },
@@ -36,22 +35,22 @@
36
35
  "@lezer/common": "^1.2.2",
37
36
  "@lezer/lezer": "^1.1.2",
38
37
  "@lezer/lr": "^1.4.2",
39
- "@dxos/context": "0.8.4-main.bc674ce",
40
- "@dxos/debug": "0.8.4-main.bc674ce",
41
- "@dxos/echo": "0.8.4-main.bc674ce",
42
- "@dxos/errors": "0.8.4-main.bc674ce",
43
- "@dxos/echo-protocol": "0.8.4-main.bc674ce",
44
- "@dxos/invariant": "0.8.4-main.bc674ce",
45
- "@dxos/node-std": "0.8.4-main.bc674ce",
46
- "@dxos/util": "0.8.4-main.bc674ce",
47
- "@dxos/vendor-quickjs": "0.8.4-main.bc674ce"
38
+ "@dxos/context": "0.8.4-main.bcb3aa67d6",
39
+ "@dxos/echo": "0.8.4-main.bcb3aa67d6",
40
+ "@dxos/debug": "0.8.4-main.bcb3aa67d6",
41
+ "@dxos/echo-protocol": "0.8.4-main.bcb3aa67d6",
42
+ "@dxos/node-std": "0.8.4-main.bcb3aa67d6",
43
+ "@dxos/util": "0.8.4-main.bcb3aa67d6",
44
+ "@dxos/errors": "0.8.4-main.bcb3aa67d6",
45
+ "@dxos/invariant": "0.8.4-main.bcb3aa67d6",
46
+ "@dxos/vendor-quickjs": "0.8.4-main.bcb3aa67d6"
48
47
  },
49
48
  "devDependencies": {
50
49
  "@lezer/generator": "^1.7.1",
51
50
  "tsdown": "^0.16.7",
52
51
  "typescript": "^5.9.3",
53
- "@dxos/echo-generator": "0.8.4-main.bc674ce",
54
- "@dxos/random": "0.8.4-main.bc674ce"
52
+ "@dxos/echo-generator": "0.8.4-main.bcb3aa67d6",
53
+ "@dxos/random": "0.8.4-main.bcb3aa67d6"
55
54
  },
56
55
  "publishConfig": {
57
56
  "access": "public"
@@ -51,10 +51,10 @@ describe('query', () => {
51
51
  },
52
52
  // Type
53
53
  {
54
- input: 'type:dxos.org/type/Person',
54
+ input: 'type:org.dxos.type.person',
55
55
  expected: [
56
56
  'Query',
57
- // type:dxos.org/type/Person
57
+ // type:org.dxos.type.person
58
58
  'Filter',
59
59
  'TypeFilter',
60
60
  'TypeKeyword',
@@ -111,10 +111,10 @@ describe('query', () => {
111
111
  ],
112
112
  },
113
113
  {
114
- input: 'type:dxos.org/type/Person OR type:dxos.org/type/Organization',
114
+ input: 'type:org.dxos.type.person OR type:org.dxos.type.organization',
115
115
  expected: [
116
116
  'Query',
117
- // type:dxos.org/type/Person
117
+ // type:org.dxos.type.person
118
118
  'Filter',
119
119
  'TypeFilter',
120
120
  'TypeKeyword',
@@ -122,7 +122,7 @@ describe('query', () => {
122
122
  'Identifier',
123
123
  // OR
124
124
  'Or',
125
- // type:dxos.org/type/Organization
125
+ // type:org.dxos.type.organization
126
126
  'Filter',
127
127
  'TypeFilter',
128
128
  'TypeKeyword',
@@ -131,11 +131,11 @@ describe('query', () => {
131
131
  ],
132
132
  },
133
133
  {
134
- input: '(type:dxos.org/type/Person OR type:dxos.org/type/Organization) AND { name: "DXOS" }',
134
+ input: '(type:org.dxos.type.person OR type:org.dxos.type.organization) AND { name: "DXOS" }',
135
135
  expected: [
136
136
  'Query',
137
137
  '(',
138
- // type:dxos.org/type/Person
138
+ // type:org.dxos.type.person
139
139
  'Filter',
140
140
  'TypeFilter',
141
141
  'TypeKeyword',
@@ -143,7 +143,7 @@ describe('query', () => {
143
143
  'Identifier',
144
144
  // OR
145
145
  'Or',
146
- // type:dxos.org/type/Organization
146
+ // type:org.dxos.type.organization
147
147
  'Filter',
148
148
  'TypeFilter',
149
149
  'TypeKeyword',
@@ -164,10 +164,10 @@ describe('query', () => {
164
164
  ],
165
165
  },
166
166
  {
167
- input: 'type:dxos.org/type/Person -> type:dxos.org/type/Organization',
167
+ input: 'type:org.dxos.type.person -> type:org.dxos.type.organization',
168
168
  expected: [
169
169
  'Query',
170
- // type:dxos.org/type/Person
170
+ // type:org.dxos.type.person
171
171
  'Filter',
172
172
  'TypeFilter',
173
173
  'TypeKeyword',
@@ -175,7 +175,7 @@ describe('query', () => {
175
175
  'Identifier',
176
176
  'Relation',
177
177
  'ArrowRight',
178
- // type:dxos.org/type/Organization
178
+ // type:org.dxos.type.organization
179
179
  'Filter',
180
180
  'TypeFilter',
181
181
  'TypeKeyword',
@@ -184,10 +184,10 @@ describe('query', () => {
184
184
  ],
185
185
  },
186
186
  {
187
- input: 'type:dxos.org/type/Organization <- type:dxos.org/type/Person',
187
+ input: 'type:org.dxos.type.organization <- type:org.dxos.type.person',
188
188
  expected: [
189
189
  'Query',
190
- // type:dxos.org/type/Organization
190
+ // type:org.dxos.type.organization
191
191
  'Filter',
192
192
  'TypeFilter',
193
193
  'TypeKeyword',
@@ -195,7 +195,7 @@ describe('query', () => {
195
195
  'Identifier',
196
196
  'Relation',
197
197
  'ArrowLeft',
198
- // type:dxos.org/type/Person
198
+ // type:org.dxos.type.person
199
199
  'Filter',
200
200
  'TypeFilter',
201
201
  'TypeKeyword',
@@ -206,7 +206,7 @@ describe('query', () => {
206
206
  {
207
207
  // Persons for Organizations with name "DXOS"
208
208
  // TODO(burdon): Filter relations.
209
- input: '((type:dxos.org/type/Organization AND { name: "DXOS" }) -> type:dxos.org/type/Person)',
209
+ input: '((type:org.dxos.type.organization AND { name: "DXOS" }) -> type:org.dxos.type.person)',
210
210
  expected: [
211
211
  'Query',
212
212
  '(',
@@ -238,7 +238,7 @@ describe('query', () => {
238
238
  ],
239
239
  },
240
240
  {
241
- input: 'type:dxos.org/type/Person and { name: "DXOS" }',
241
+ input: 'type:org.dxos.type.person and { name: "DXOS" }',
242
242
  expected: [
243
243
  'Query',
244
244
  'Filter',
@@ -259,7 +259,7 @@ describe('query', () => {
259
259
  ],
260
260
  },
261
261
  {
262
- input: 'x = ( type: dxos.org/type/Person )',
262
+ input: 'x = ( type: org.dxos.type.person )',
263
263
  expected: [
264
264
  'Query',
265
265
  'Assignment',
@@ -305,9 +305,9 @@ describe('query', () => {
305
305
  const tests: Test[] = [
306
306
  // Types
307
307
  {
308
- input: 'type:dxos.org/type/Person',
308
+ input: 'type:org.dxos.type.person',
309
309
  expected: {
310
- filter: Filter.typename('dxos.org/type/Person'),
310
+ filter: Filter.typename('org.dxos.type.person'),
311
311
  },
312
312
  },
313
313
  // Tags
@@ -357,32 +357,32 @@ describe('query', () => {
357
357
  },
358
358
  },
359
359
  {
360
- input: 'type:dxos.org/type/Person OR type:dxos.org/type/Organization',
360
+ input: 'type:org.dxos.type.person OR type:org.dxos.type.organization',
361
361
  expected: {
362
- filter: Filter.or(Filter.typename('dxos.org/type/Person'), Filter.typename('dxos.org/type/Organization')),
362
+ filter: Filter.or(Filter.typename('org.dxos.type.person'), Filter.typename('org.dxos.type.organization')),
363
363
  },
364
364
  },
365
365
  {
366
- input: '(type:dxos.org/type/Person OR type:dxos.org/type/Organization) AND { name: "DXOS" }',
366
+ input: '(type:org.dxos.type.person OR type:org.dxos.type.organization) AND { name: "DXOS" }',
367
367
  expected: {
368
368
  filter: Filter.and(
369
- Filter.or(Filter.typename('dxos.org/type/Person'), Filter.typename('dxos.org/type/Organization')),
369
+ Filter.or(Filter.typename('org.dxos.type.person'), Filter.typename('org.dxos.type.organization')),
370
370
  Filter.props({ name: 'DXOS' }),
371
371
  ),
372
372
  },
373
373
  },
374
374
  {
375
- input: 'type:dxos.org/type/Person and { name: "DXOS" }',
375
+ input: 'type:org.dxos.type.person and { name: "DXOS" }',
376
376
  expected: {
377
- filter: Filter.and(Filter.typename('dxos.org/type/Person'), Filter.props({ name: 'DXOS' })),
377
+ filter: Filter.and(Filter.typename('org.dxos.type.person'), Filter.props({ name: 'DXOS' })),
378
378
  },
379
379
  },
380
380
  // Assignment
381
381
  {
382
- input: 'x = ( type:dxos.org/type/Person )',
382
+ input: 'x = ( type:org.dxos.type.person )',
383
383
  expected: {
384
384
  name: 'x',
385
- filter: Filter.typename('dxos.org/type/Person'),
385
+ filter: Filter.typename('org.dxos.type.person'),
386
386
  },
387
387
  },
388
388
  {
@@ -401,9 +401,9 @@ describe('query', () => {
401
401
  //
402
402
  // {
403
403
  // input: '',
404
- // expected: Query.select(Filter.typename('dxos.org/type/Person', { jobTitle: 'investor' }))
404
+ // expected: Query.select(Filter.typename('org.dxos.type.person', { jobTitle: 'investor' }))
405
405
  // .reference('organization')
406
- // .targetOf(Relation.of('dxos.org/relation/HasSubject')) // TODO(burdon): Invert?
406
+ // .targetOf(Relation.of('org.dxos.relation.hasSubject')) // TODO(burdon): Invert?
407
407
  // .source(),
408
408
  // },
409
409
  ];
@@ -16,7 +16,7 @@ import type { DXN, ObjectId } from '@dxos/keys';
16
16
  // TODO(wittjosiah): The `export * as ...` syntax causes tsdown to genereate multiple files which breaks the sandbox.
17
17
 
18
18
  class OrderClass implements Order$.Any {
19
- private static variance: Order$.Any['~Order'] = {} as Order$.Any['~Order'];
19
+ private static 'variance': Order$.Any['~Order'] = {} as Order$.Any['~Order'];
20
20
 
21
21
  static is(value: unknown): value is Order$.Any {
22
22
  return typeof value === 'object' && value !== null && '~Order' in value;
@@ -46,7 +46,7 @@ const Order2: typeof Order$ = Order1;
46
46
  export { Order2 as Order };
47
47
 
48
48
  class FilterClass implements Filter$.Any {
49
- private static variance: Filter$.Any['~Filter'] = {} as Filter$.Any['~Filter'];
49
+ private static 'variance': Filter$.Any['~Filter'] = {} as Filter$.Any['~Filter'];
50
50
 
51
51
  static is(value: unknown): value is Filter$.Any {
52
52
  return typeof value === 'object' && value !== null && '~Filter' in value;
@@ -221,7 +221,7 @@ class FilterClass implements Filter$.Any {
221
221
  });
222
222
  }
223
223
 
224
- static in<T>(...values: T[]): Filter$.Filter<T | undefined> {
224
+ static in<T>(...values: T[]): Filter$.Filter<T> {
225
225
  return new FilterClass({
226
226
  type: 'in',
227
227
  values,
@@ -235,7 +235,7 @@ class FilterClass implements Filter$.Any {
235
235
  });
236
236
  }
237
237
 
238
- static between<T>(from: T, to: T): Filter$.Filter<unknown> {
238
+ static between<T>(from: T, to: T): Filter$.Filter<T> {
239
239
  return new FilterClass({
240
240
  type: 'range',
241
241
  from,
@@ -243,6 +243,32 @@ class FilterClass implements Filter$.Any {
243
243
  });
244
244
  }
245
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 #timeRangeFilter(
255
+ field: 'updatedAt' | 'createdAt',
256
+ range: { after?: Date | number; before?: Date | number },
257
+ ): Filter$.Any {
258
+ const toMs = (d: Date | number) => (typeof d === 'number' ? d : d.getTime());
259
+ const filters: Filter$.Any[] = [];
260
+ if (range.after != null) {
261
+ filters.push(new FilterClass({ type: 'timestamp', field, operator: 'gte', value: toMs(range.after) }));
262
+ }
263
+ if (range.before != null) {
264
+ filters.push(new FilterClass({ type: 'timestamp', field, operator: 'lte', value: toMs(range.before) }));
265
+ }
266
+ if (filters.length === 0) {
267
+ return FilterClass.everything();
268
+ }
269
+ return filters.length === 1 ? filters[0] : FilterClass.and(...filters);
270
+ }
271
+
246
272
  static not<F extends Filter$.Any>(filter: F): Filter$.Filter<Filter$.Type<F>> {
247
273
  return new FilterClass({
248
274
  type: 'not',
@@ -268,6 +294,11 @@ class FilterClass implements Filter$.Any {
268
294
  });
269
295
  }
270
296
 
297
+ /** Returns a human-readable string representation of a Filter AST. */
298
+ static pretty(filter: Filter$.Any): string {
299
+ return prettyFilter(filter.ast);
300
+ }
301
+
271
302
  private constructor(public readonly ast: QueryAST.Filter) {}
272
303
 
273
304
  '~Filter' = FilterClass.variance;
@@ -328,7 +359,7 @@ const processPredicate = (predicate: any): QueryAST.Filter => {
328
359
  };
329
360
 
330
361
  class QueryClass implements Query$.Any {
331
- private static variance: Query$.Any['~Query'] = {} as Query$.Any['~Query'];
362
+ private static 'variance': Query$.Any['~Query'] = {} as Query$.Any['~Query'];
332
363
 
333
364
  static is(value: unknown): value is Query$.Any {
334
365
  return typeof value === 'object' && value !== null && '~Query' in value;
@@ -345,6 +376,22 @@ class QueryClass implements Query$.Any {
345
376
  });
346
377
  }
347
378
 
379
+ select(filter: Filter$.Any | Filter$.Props<any>): Query$.Any {
380
+ if (FilterClass.is(filter)) {
381
+ return new QueryClass({
382
+ type: 'filter',
383
+ selection: this.ast,
384
+ filter: filter.ast,
385
+ });
386
+ } else {
387
+ return new QueryClass({
388
+ type: 'filter',
389
+ selection: this.ast,
390
+ filter: FilterClass.props(filter).ast,
391
+ });
392
+ }
393
+ }
394
+
348
395
  static type(schema: Schema.Schema.All | string, predicates?: Filter$.Props<unknown>): Query$.Any {
349
396
  return new QueryClass({
350
397
  type: 'select',
@@ -372,26 +419,49 @@ class QueryClass implements Query$.Any {
372
419
  });
373
420
  }
374
421
 
375
- constructor(public readonly ast: QueryAST.Query) {}
376
-
377
- '~Query' = QueryClass.variance;
422
+ static from(source: any, options?: { includeFeeds?: boolean }): Query$.Any {
423
+ const baseQuery: QueryAST.Query = {
424
+ type: 'select',
425
+ filter: FilterClass.everything().ast,
426
+ };
427
+ const wrapper = new QueryClass(baseQuery);
428
+ return wrapper.from(source, options);
429
+ }
378
430
 
379
- select(filter: Filter$.Any | Filter$.Props<any>): Query$.Any {
380
- if (FilterClass.is(filter)) {
431
+ from(arg: any, options?: { includeFeeds?: boolean }): Query$.Any {
432
+ if (arg === 'all-accessible-spaces') {
381
433
  return new QueryClass({
382
- type: 'filter',
383
- selection: this.ast,
384
- filter: filter.ast,
434
+ type: 'from',
435
+ query: this.ast,
436
+ from: {
437
+ _tag: 'scope',
438
+ scope: {
439
+ ...(options?.includeFeeds ? { allQueuesFromSpaces: true } : {}),
440
+ },
441
+ },
385
442
  });
386
- } else {
443
+ }
444
+
445
+ if (_isScopeLike(arg)) {
387
446
  return new QueryClass({
388
- type: 'filter',
389
- selection: this.ast,
390
- filter: FilterClass.props(filter).ast,
447
+ type: 'from',
448
+ query: this.ast,
449
+ from: { _tag: 'scope', scope: arg },
391
450
  });
392
451
  }
452
+
453
+ throw new TypeError('Database and Feed objects are not supported in query-lite sandbox');
454
+ }
455
+
456
+ /** Returns a human-readable string representation of a Query AST. */
457
+ static pretty(query: Query$.Any): string {
458
+ return prettyQuery(query.ast);
393
459
  }
394
460
 
461
+ constructor(public readonly ast: QueryAST.Query) {}
462
+
463
+ '~Query' = QueryClass.variance;
464
+
395
465
  reference(key: string): Query$.Any {
396
466
  return new QueryClass({
397
467
  type: 'reference-traversal',
@@ -415,21 +485,21 @@ class QueryClass implements Query$.Any {
415
485
  });
416
486
  }
417
487
 
418
- sourceOf(relation: Schema.Schema.All | string, predicates?: Filter$.Props<unknown> | undefined): Query$.Any {
488
+ sourceOf(relation?: Schema.Schema.All | string, predicates?: Filter$.Props<unknown> | undefined): Query$.Any {
419
489
  return new QueryClass({
420
490
  type: 'relation',
421
491
  anchor: this.ast,
422
492
  direction: 'outgoing',
423
- filter: FilterClass.type(relation, predicates).ast,
493
+ filter: relation !== undefined ? FilterClass.type(relation, predicates).ast : undefined,
424
494
  });
425
495
  }
426
496
 
427
- targetOf(relation: Schema.Schema.All | string, predicates?: Filter$.Props<unknown> | undefined): Query$.Any {
497
+ targetOf(relation?: Schema.Schema.All | string, predicates?: Filter$.Props<unknown> | undefined): Query$.Any {
428
498
  return new QueryClass({
429
499
  type: 'relation',
430
500
  anchor: this.ast,
431
501
  direction: 'incoming',
432
- filter: FilterClass.type(relation, predicates).ast,
502
+ filter: relation !== undefined ? FilterClass.type(relation, predicates).ast : undefined,
433
503
  });
434
504
  }
435
505
 
@@ -449,6 +519,22 @@ class QueryClass implements Query$.Any {
449
519
  });
450
520
  }
451
521
 
522
+ parent(): Query$.Any {
523
+ return new QueryClass({
524
+ type: 'hierarchy-traversal',
525
+ anchor: this.ast,
526
+ direction: 'to-parent',
527
+ });
528
+ }
529
+
530
+ children(): Query$.Any {
531
+ return new QueryClass({
532
+ type: 'hierarchy-traversal',
533
+ anchor: this.ast,
534
+ direction: 'to-children',
535
+ });
536
+ }
537
+
452
538
  orderBy(...order: Order$.Any[]): Query$.Any {
453
539
  return new QueryClass({
454
540
  type: 'order',
@@ -487,3 +573,127 @@ const makeTypeDxn = (typename: string) => {
487
573
  assertArgument(!typename.startsWith('dxn:'), 'typename');
488
574
  return `dxn:type:${typename}`;
489
575
  };
576
+
577
+ const SCOPE_KEYS = new Set(['spaceIds', 'queues', 'allQueuesFromSpaces']);
578
+
579
+ const _isScopeLike = (value: unknown): value is QueryAST.Scope => {
580
+ if (typeof value !== 'object' || value === null || Array.isArray(value)) {
581
+ return false;
582
+ }
583
+ return Object.keys(value).every((key) => SCOPE_KEYS.has(key));
584
+ };
585
+
586
+ const prettyFilter = (filter: QueryAST.Filter): string => {
587
+ switch (filter.type) {
588
+ case 'object': {
589
+ const parts: string[] = [];
590
+ if (filter.typename !== null) {
591
+ parts.push(JSON.stringify(filter.typename));
592
+ }
593
+ const propEntries = Object.entries(filter.props);
594
+ if (propEntries.length > 0) {
595
+ const propsStr = propEntries.map(([k, v]) => `${k}: ${prettyFilter(v)}`).join(', ');
596
+ parts.push(`{ ${propsStr} }`);
597
+ }
598
+ if (filter.id !== undefined) {
599
+ parts.push(`id: [${filter.id.join(', ')}]`);
600
+ }
601
+ return parts.length > 0 ? `Filter.type(${parts.join(', ')})` : 'Filter.everything()';
602
+ }
603
+ case 'compare':
604
+ return `Filter.${filter.operator}(${JSON.stringify(filter.value)})`;
605
+ case 'in':
606
+ return `Filter.in(${filter.values.map((v) => JSON.stringify(v)).join(', ')})`;
607
+ case 'contains':
608
+ return `Filter.contains(${JSON.stringify(filter.value)})`;
609
+ case 'range':
610
+ return `Filter.between(${JSON.stringify(filter.from)}, ${JSON.stringify(filter.to)})`;
611
+ case 'text-search':
612
+ return `Filter.text(${JSON.stringify(filter.text)})`;
613
+ case 'tag':
614
+ return `Filter.tag(${JSON.stringify(filter.tag)})`;
615
+ case 'timestamp':
616
+ return `Filter.${filter.field}.${filter.operator}(${filter.value})`;
617
+ case 'not':
618
+ return `Filter.not(${prettyFilter(filter.filter)})`;
619
+ case 'and':
620
+ return `Filter.and(${filter.filters.map(prettyFilter).join(', ')})`;
621
+ case 'or':
622
+ return `Filter.or(${filter.filters.map(prettyFilter).join(', ')})`;
623
+ }
624
+ };
625
+
626
+ const prettyQuery = (query: QueryAST.Query): string => {
627
+ switch (query.type) {
628
+ case 'select':
629
+ return `Query.select(${prettyFilter(query.filter)})`;
630
+ case 'filter':
631
+ return `${prettyQuery(query.selection)}.select(${prettyFilter(query.filter)})`;
632
+ case 'reference-traversal':
633
+ return `${prettyQuery(query.anchor)}.reference(${JSON.stringify(query.property)})`;
634
+ case 'incoming-references': {
635
+ const args: string[] = [];
636
+ if (query.typename !== null) {
637
+ args.push(JSON.stringify(query.typename));
638
+ }
639
+ if (query.property !== null) {
640
+ args.push(JSON.stringify(query.property));
641
+ }
642
+ return `${prettyQuery(query.anchor)}.referencedBy(${args.join(', ')})`;
643
+ }
644
+ case 'relation': {
645
+ const method =
646
+ query.direction === 'outgoing' ? 'sourceOf' : query.direction === 'incoming' ? 'targetOf' : 'relationOf';
647
+ const filterStr = query.filter !== undefined ? prettyFilter(query.filter) : '';
648
+ return `${prettyQuery(query.anchor)}.${method}(${filterStr})`;
649
+ }
650
+ case 'relation-traversal':
651
+ return `${prettyQuery(query.anchor)}.${query.direction}()`;
652
+ case 'hierarchy-traversal':
653
+ return query.direction === 'to-parent'
654
+ ? `${prettyQuery(query.anchor)}.parent()`
655
+ : `${prettyQuery(query.anchor)}.children()`;
656
+ case 'union':
657
+ return `Query.all(${query.queries.map(prettyQuery).join(', ')})`;
658
+ case 'set-difference':
659
+ return `Query.without(${prettyQuery(query.source)}, ${prettyQuery(query.exclude)})`;
660
+ case 'order': {
661
+ const orders = query.order.map((o) => {
662
+ if (o.kind === 'natural') {
663
+ return 'Order.natural';
664
+ }
665
+ if (o.kind === 'rank') {
666
+ return `Order.rank(${JSON.stringify(o.direction)})`;
667
+ }
668
+ return `Order.property(${JSON.stringify(o.property)}, ${JSON.stringify(o.direction)})`;
669
+ });
670
+ return `${prettyQuery(query.query)}.orderBy(${orders.join(', ')})`;
671
+ }
672
+ case 'options': {
673
+ const parts: string[] = [];
674
+ if (query.options.deleted !== undefined) {
675
+ parts.push(`deleted: ${JSON.stringify(query.options.deleted)}`);
676
+ }
677
+ return `${prettyQuery(query.query)}.options({ ${parts.join(', ')} })`;
678
+ }
679
+ case 'from': {
680
+ if (query.from._tag === 'scope') {
681
+ const scope = query.from.scope;
682
+ const parts: string[] = [];
683
+ if (scope.spaceIds !== undefined) {
684
+ parts.push(`spaceIds: [${scope.spaceIds.join(', ')}]`);
685
+ }
686
+ if (scope.queues !== undefined) {
687
+ parts.push(`queues: [${scope.queues.join(', ')}]`);
688
+ }
689
+ if (scope.allQueuesFromSpaces !== undefined) {
690
+ parts.push(`allQueuesFromSpaces: ${scope.allQueuesFromSpaces}`);
691
+ }
692
+ return `${prettyQuery(query.query)}.from({ ${parts.join(', ')} })`;
693
+ }
694
+ return `${prettyQuery(query.query)}.from(${prettyQuery(query.from.query)})`;
695
+ }
696
+ case 'limit':
697
+ return `${prettyQuery(query.query)}.limit(${query.limit})`;
698
+ }
699
+ };
@@ -16,37 +16,37 @@ describe('QuerySandbox', () => {
16
16
 
17
17
  test('works', { timeout: 10_000 }, async () => {
18
18
  const ast = sandbox.eval(trim`
19
- Query.select(Filter.typename('dxos.org/type/Person'))
19
+ Query.select(Filter.typename('org.dxos.type.person'))
20
20
  `);
21
- expect(ast).toEqual(Query.select(Filter.typename('dxos.org/type/Person')).ast);
21
+ expect(ast).toEqual(Query.select(Filter.typename('org.dxos.type.person')).ast);
22
22
  });
23
23
 
24
24
  test('works with just Filter passed in', () => {
25
25
  const ast = sandbox.eval(trim`
26
- Filter.typename('dxos.org/type/Person')
26
+ Filter.typename('org.dxos.type.person')
27
27
  `);
28
- expect(ast).toEqual(Query.select(Filter.typename('dxos.org/type/Person')).ast);
28
+ expect(ast).toEqual(Query.select(Filter.typename('org.dxos.type.person')).ast);
29
29
  });
30
30
 
31
31
  test('Order', () => {
32
32
  const ast = sandbox.eval(trim`
33
- Query.type('dxos.org/type/Person').orderBy(Order.property('name', 'desc'))
33
+ Query.type('org.dxos.type.person').orderBy(Order.property('name', 'desc'))
34
34
  `);
35
- expect(ast).toEqual(Query.type('dxos.org/type/Person').orderBy(Order.property('name', 'desc')).ast);
35
+ expect(ast).toEqual(Query.type('org.dxos.type.person').orderBy(Order.property('name', 'desc')).ast);
36
36
  });
37
37
 
38
38
  test('traversal', () => {
39
39
  const ast = sandbox.eval(trim`
40
- Query.select(Filter.type('dxos.org/type/Person', { jobTitle: 'investor' }))
40
+ Query.select(Filter.type('org.dxos.type.person', { jobTitle: 'investor' }))
41
41
  .reference('organization')
42
- .targetOf('dxos.org/relation/HasSubject')
42
+ .targetOf('org.dxos.relation.hasSubject')
43
43
  .source()
44
44
  `);
45
45
 
46
46
  expect(ast).toEqual(
47
- Query.select(Filter.type('dxos.org/type/Person', { jobTitle: 'investor' }))
47
+ Query.select(Filter.type('org.dxos.type.person', { jobTitle: 'investor' }))
48
48
  .reference('organization')
49
- .targetOf('dxos.org/relation/HasSubject')
49
+ .targetOf('org.dxos.relation.hasSubject')
50
50
  .source().ast,
51
51
  );
52
52
  });
@@ -49,7 +49,7 @@ export class QuerySandbox extends Resource {
49
49
 
50
50
  /**
51
51
  * Evaluates the query code.
52
- * @param queryCode Example: `Query.select(Filter.typename('dxos.org/type/Person'))`
52
+ * @param queryCode Example: `Query.select(Filter.typename('org.dxos.type.person'))`
53
53
  */
54
54
  eval(queryCode: string): QueryAST.Query {
55
55
  using context = this.#runtime.newContext();