@malloy-publisher/server 0.0.178 → 0.0.180-dev-v1

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 (74) hide show
  1. package/build.ts +1 -1
  2. package/dist/app/api-doc.yaml +505 -52
  3. package/dist/app/assets/HomePage-Dn3E4CuB.js +1 -0
  4. package/dist/app/assets/{MainPage-JYvDXOkC.js → MainPage-BzB3yoqi.js} +2 -2
  5. package/dist/app/assets/{ModelPage-TEQrhaqq.js → ModelPage-C9O_sAXT.js} +1 -1
  6. package/dist/app/assets/PackagePage-DcxKEjBX.js +1 -0
  7. package/dist/app/assets/ProjectPage-BDj307rF.js +1 -0
  8. package/dist/app/assets/{RouteError-DnSZEzkT.js → RouteError-DAShbVCG.js} +1 -1
  9. package/dist/app/assets/{WorkbookPage-DjQ8u5DD.js → WorkbookPage-Cs_XYEaB.js} +1 -1
  10. package/dist/app/assets/core-CjeTkq8O.es-BqRc6yhC.js +148 -0
  11. package/dist/app/assets/engine-oniguruma-C4vnmooL.es-jdkXmgTr.js +1 -0
  12. package/dist/app/assets/github-light-JYsPkUQd.es-DAi9KRSo.js +1 -0
  13. package/dist/app/assets/index-15BOvhp0.js +456 -0
  14. package/dist/app/assets/{index--80Q7qw1.js → index-Bb2jqquW.js} +1 -1
  15. package/dist/app/assets/{index-CZ4G_NMp.js → index-D68X76-7.js} +168 -166
  16. package/dist/app/assets/index.umd-DGBekgSu.js +1145 -0
  17. package/dist/app/assets/json-71t8ZF9g.es-BQoSv7ci.js +1 -0
  18. package/dist/app/assets/sql-DCkt643-.es-COK4E0Yg.js +1 -0
  19. package/dist/app/assets/typescript-buWNZFwO.es-Dj6nwHGl.js +1 -0
  20. package/dist/app/index.html +1 -1
  21. package/dist/instrumentation.js +10567 -10584
  22. package/dist/server.js +16973 -15367
  23. package/package.json +14 -12
  24. package/src/controller/connection.controller.ts +27 -20
  25. package/src/controller/manifest.controller.ts +29 -0
  26. package/src/controller/materialization.controller.ts +125 -0
  27. package/src/controller/model.controller.ts +4 -3
  28. package/src/controller/package.controller.ts +53 -2
  29. package/src/controller/query.controller.ts +5 -0
  30. package/src/errors.ts +24 -0
  31. package/src/mcp/resources/model_resource.ts +12 -9
  32. package/src/mcp/resources/source_resource.ts +7 -6
  33. package/src/mcp/resources/view_resource.ts +0 -1
  34. package/src/mcp/tools/execute_query_tool.ts +9 -0
  35. package/src/server.ts +217 -5
  36. package/src/service/connection.ts +1 -4
  37. package/src/service/db_utils.spec.ts +4 -2
  38. package/src/service/db_utils.ts +6 -2
  39. package/src/service/filter.spec.ts +447 -0
  40. package/src/service/filter.ts +337 -0
  41. package/src/service/filter_integration.spec.ts +825 -0
  42. package/src/service/manifest_service.spec.ts +201 -0
  43. package/src/service/manifest_service.ts +106 -0
  44. package/src/service/materialization_service.spec.ts +648 -0
  45. package/src/service/materialization_service.ts +929 -0
  46. package/src/service/materialized_table_gc.spec.ts +383 -0
  47. package/src/service/materialized_table_gc.ts +279 -0
  48. package/src/service/model.ts +221 -47
  49. package/src/service/package.ts +50 -0
  50. package/src/service/project_store.ts +21 -2
  51. package/src/service/quoting.ts +41 -0
  52. package/src/service/resolve_project.ts +13 -0
  53. package/src/storage/DatabaseInterface.ts +103 -1
  54. package/src/storage/{StorageManager.spec.ts → StorageManager.mock.ts} +9 -0
  55. package/src/storage/StorageManager.ts +119 -1
  56. package/src/storage/duckdb/DuckDBManifestStore.ts +70 -0
  57. package/src/storage/duckdb/DuckDBRepository.ts +99 -9
  58. package/src/storage/duckdb/ManifestRepository.ts +119 -0
  59. package/src/storage/duckdb/MaterializationRepository.ts +249 -0
  60. package/src/storage/duckdb/manifest_store.spec.ts +133 -0
  61. package/src/storage/duckdb/schema.ts +59 -1
  62. package/src/storage/ducklake/DuckLakeManifestStore.ts +146 -0
  63. package/tests/fixtures/persist-test/data/orders.csv +5 -0
  64. package/tests/fixtures/persist-test/persist_test.malloy +11 -0
  65. package/tests/fixtures/persist-test/publisher.json +5 -0
  66. package/tests/fixtures/publisher.config.json +15 -0
  67. package/tests/harness/rest_e2e.ts +68 -0
  68. package/tests/integration/materialization/materialization_lifecycle.integration.spec.ts +470 -0
  69. package/tests/integration/mcp/mcp_execute_query_tool.integration.spec.ts +2 -2
  70. package/dist/app/assets/HomePage-CwUkFsA8.js +0 -1
  71. package/dist/app/assets/PackagePage-CgE-izLw.js +0 -1
  72. package/dist/app/assets/ProjectPage-PiMPpFX8.js +0 -1
  73. package/dist/app/assets/index-BJUsHnGO.js +0 -467
  74. package/dist/app/assets/index.umd-Cf-wqh-R.js +0 -1145
@@ -0,0 +1,447 @@
1
+ import { describe, expect, it } from "bun:test";
2
+ import {
3
+ buildFilterClause,
4
+ injectFilterRefinement,
5
+ parseFilterAnnotation,
6
+ parseFilters,
7
+ FilterValidationError,
8
+ type FilterDefinition,
9
+ type FilterParams,
10
+ } from "./filter";
11
+
12
+ describe("service/filter", () => {
13
+ // -----------------------------------------------------------------------
14
+ // parseFilterAnnotation
15
+ // -----------------------------------------------------------------------
16
+ describe("parseFilterAnnotation", () => {
17
+ it("returns null for non-filter annotations", () => {
18
+ expect(parseFilterAnnotation("#(doc) Some docs")).toBeNull();
19
+ expect(parseFilterAnnotation("# bar_chart")).toBeNull();
20
+ expect(parseFilterAnnotation("")).toBeNull();
21
+ });
22
+
23
+ it("parses a minimal annotation (dimension + type)", () => {
24
+ const result = parseFilterAnnotation(
25
+ "#(filter) dimension=status type=equal",
26
+ );
27
+ expect(result).toEqual({
28
+ name: "status",
29
+ dimension: "status",
30
+ type: "equal",
31
+ implicit: false,
32
+ required: false,
33
+ });
34
+ });
35
+
36
+ it("parses all fields including name, implicit, required", () => {
37
+ const result = parseFilterAnnotation(
38
+ '#(filter) name="Customer ID" dimension=customer_id type=equal implicit required',
39
+ );
40
+ expect(result).toEqual({
41
+ name: "Customer ID",
42
+ dimension: "customer_id",
43
+ type: "equal",
44
+ implicit: true,
45
+ required: true,
46
+ });
47
+ });
48
+
49
+ it("parses type=in", () => {
50
+ const result = parseFilterAnnotation(
51
+ "#(filter) dimension=region type=in",
52
+ );
53
+ expect(result).toEqual({
54
+ name: "region",
55
+ dimension: "region",
56
+ type: "in",
57
+ implicit: false,
58
+ required: false,
59
+ });
60
+ });
61
+
62
+ it("parses type=like", () => {
63
+ const result = parseFilterAnnotation(
64
+ "#(filter) dimension=name type=like",
65
+ );
66
+ expect(result!.type).toBe("like");
67
+ });
68
+
69
+ it("parses type=greater_than", () => {
70
+ const result = parseFilterAnnotation(
71
+ "#(filter) dimension=created_at type=greater_than",
72
+ );
73
+ expect(result!.type).toBe("greater_than");
74
+ });
75
+
76
+ it("parses type=less_than", () => {
77
+ const result = parseFilterAnnotation(
78
+ "#(filter) dimension=created_at type=less_than",
79
+ );
80
+ expect(result!.type).toBe("less_than");
81
+ });
82
+
83
+ it("parses required without implicit", () => {
84
+ const result = parseFilterAnnotation(
85
+ "#(filter) dimension=tenant_id type=equal required",
86
+ );
87
+ expect(result).toEqual({
88
+ name: "tenant_id",
89
+ dimension: "tenant_id",
90
+ type: "equal",
91
+ implicit: false,
92
+ required: true,
93
+ });
94
+ });
95
+
96
+ it("handles single-quoted name values", () => {
97
+ const result = parseFilterAnnotation(
98
+ "#(filter) name='My Filter' dimension=col type=equal",
99
+ );
100
+ expect(result!.name).toBe("My Filter");
101
+ });
102
+
103
+ it("handles extra whitespace", () => {
104
+ const result = parseFilterAnnotation(
105
+ " #(filter) dimension=status type=equal required ",
106
+ );
107
+ expect(result).toEqual({
108
+ name: "status",
109
+ dimension: "status",
110
+ type: "equal",
111
+ implicit: false,
112
+ required: true,
113
+ });
114
+ });
115
+
116
+ it("throws on missing dimension", () => {
117
+ expect(() => parseFilterAnnotation("#(filter) type=equal")).toThrow(
118
+ "missing required 'dimension'",
119
+ );
120
+ });
121
+
122
+ it("throws on missing type", () => {
123
+ expect(() =>
124
+ parseFilterAnnotation("#(filter) dimension=status"),
125
+ ).toThrow("missing required 'type'");
126
+ });
127
+
128
+ it("throws on invalid type", () => {
129
+ expect(() =>
130
+ parseFilterAnnotation("#(filter) dimension=status type=banana"),
131
+ ).toThrow('Invalid filter type "banana"');
132
+ });
133
+
134
+ it("throws on unknown parameter", () => {
135
+ expect(() =>
136
+ parseFilterAnnotation(
137
+ "#(filter) dimension=status type=equal foo=bar",
138
+ ),
139
+ ).toThrow('Unknown filter parameter "foo"');
140
+ });
141
+
142
+ it("throws on unknown flag", () => {
143
+ expect(() =>
144
+ parseFilterAnnotation(
145
+ "#(filter) dimension=status type=equal banana",
146
+ ),
147
+ ).toThrow('Unknown filter flag "banana"');
148
+ });
149
+ });
150
+
151
+ // -----------------------------------------------------------------------
152
+ // parseFilters
153
+ // -----------------------------------------------------------------------
154
+ describe("parseFilters", () => {
155
+ it("extracts filter annotations from a mixed list", () => {
156
+ const annotations = [
157
+ "#(doc) This is a source for orders",
158
+ "#(filter) dimension=status type=equal",
159
+ "# bar_chart",
160
+ "#(filter) dimension=region type=in required",
161
+ ];
162
+ const filters = parseFilters(annotations);
163
+ expect(filters).toHaveLength(2);
164
+ expect(filters[0].dimension).toBe("status");
165
+ expect(filters[1].dimension).toBe("region");
166
+ expect(filters[1].required).toBe(true);
167
+ });
168
+
169
+ it("returns empty array when no filter annotations", () => {
170
+ const filters = parseFilters(["#(doc) some docs", "# hidden"]);
171
+ expect(filters).toHaveLength(0);
172
+ });
173
+
174
+ it("deduplicates by name, later annotations win (extend pattern)", () => {
175
+ const annotations = [
176
+ // Base source annotations (come first in blockNotes via inherits chain)
177
+ "#(filter) name=Manufacturer dimension=Manufacturer type=in",
178
+ "#(filter) name=Subject dimension=Subject type=like",
179
+ '#(filter) name="Major Recall" dimension="Major Recall" type=equal',
180
+ // Extending source annotations (come later, should win)
181
+ "#(filter) name=Manufacturer dimension=Manufacturer type=equal required",
182
+ "#(filter) name=Subject dimension=Subject type=like",
183
+ ];
184
+ const filters = parseFilters(annotations);
185
+ // 3 unique names: Manufacturer, Subject, Major Recall
186
+ expect(filters).toHaveLength(3);
187
+
188
+ // Manufacturer: child overrides base (in → equal, gains required)
189
+ const mfr = filters.find((f) => f.name === "Manufacturer");
190
+ expect(mfr).toBeDefined();
191
+ expect(mfr!.type).toBe("equal");
192
+ expect(mfr!.required).toBe(true);
193
+
194
+ // Subject: child re-declares identically, no visible change
195
+ const subj = filters.find((f) => f.name === "Subject");
196
+ expect(subj).toBeDefined();
197
+ expect(subj!.type).toBe("like");
198
+ expect(subj!.required).toBeFalsy();
199
+
200
+ // Major Recall: only on base, preserved in child
201
+ const major = filters.find((f) => f.name === "Major Recall");
202
+ expect(major).toBeDefined();
203
+ expect(major!.type).toBe("equal");
204
+ expect(major!.dimension).toBe("Major Recall");
205
+ });
206
+
207
+ it("child can remove required flag by overriding", () => {
208
+ const annotations = [
209
+ "#(filter) name=status dimension=status type=equal required",
210
+ "#(filter) name=status dimension=status type=equal",
211
+ ];
212
+ const filters = parseFilters(annotations);
213
+ expect(filters).toHaveLength(1);
214
+ expect(filters[0].name).toBe("status");
215
+ expect(filters[0].required).toBeFalsy();
216
+ });
217
+
218
+ it("child can change filter type by overriding", () => {
219
+ const annotations = [
220
+ "#(filter) name=category dimension=category type=in",
221
+ "#(filter) name=category dimension=category type=equal required",
222
+ ];
223
+ const filters = parseFilters(annotations);
224
+ expect(filters).toHaveLength(1);
225
+ expect(filters[0].type).toBe("equal");
226
+ expect(filters[0].required).toBe(true);
227
+ });
228
+ });
229
+
230
+ // -----------------------------------------------------------------------
231
+ // buildFilterClause
232
+ // -----------------------------------------------------------------------
233
+ describe("buildFilterClause", () => {
234
+ const equalFilter: FilterDefinition = {
235
+ name: "status",
236
+ dimension: "status",
237
+ type: "equal",
238
+ implicit: false,
239
+ required: false,
240
+ };
241
+
242
+ const inFilter: FilterDefinition = {
243
+ name: "region",
244
+ dimension: "region",
245
+ type: "in",
246
+ implicit: false,
247
+ required: false,
248
+ };
249
+
250
+ const likeFilter: FilterDefinition = {
251
+ name: "name_search",
252
+ dimension: "customer_name",
253
+ type: "like",
254
+ implicit: false,
255
+ required: false,
256
+ };
257
+
258
+ const gtFilter: FilterDefinition = {
259
+ name: "start_date",
260
+ dimension: "created_at",
261
+ type: "greater_than",
262
+ implicit: false,
263
+ required: false,
264
+ };
265
+
266
+ const ltFilter: FilterDefinition = {
267
+ name: "end_date",
268
+ dimension: "created_at",
269
+ type: "less_than",
270
+ implicit: false,
271
+ required: false,
272
+ };
273
+
274
+ const requiredFilter: FilterDefinition = {
275
+ name: "tenant_id",
276
+ dimension: "tenant_id",
277
+ type: "equal",
278
+ implicit: true,
279
+ required: true,
280
+ };
281
+
282
+ it("returns empty string when no params provided", () => {
283
+ const clause = buildFilterClause([equalFilter], {});
284
+ expect(clause).toBe("");
285
+ });
286
+
287
+ it("returns empty string when param is empty string", () => {
288
+ const clause = buildFilterClause([equalFilter], { status: "" });
289
+ expect(clause).toBe("");
290
+ });
291
+
292
+ it("returns empty string when param is empty array", () => {
293
+ const clause = buildFilterClause([inFilter], { region: [] });
294
+ expect(clause).toBe("");
295
+ });
296
+
297
+ it("builds equal predicate", () => {
298
+ const clause = buildFilterClause([equalFilter], {
299
+ status: "active",
300
+ });
301
+ expect(clause).toBe("`status` = 'active'");
302
+ });
303
+
304
+ it("equal uses first element if given array", () => {
305
+ const clause = buildFilterClause([equalFilter], {
306
+ status: ["active", "pending"],
307
+ });
308
+ expect(clause).toBe("`status` = 'active'");
309
+ });
310
+
311
+ it("builds in predicate with single value", () => {
312
+ const clause = buildFilterClause([inFilter], {
313
+ region: ["US"],
314
+ });
315
+ expect(clause).toBe("`region` = 'US'");
316
+ });
317
+
318
+ it("builds in predicate with multiple values", () => {
319
+ const clause = buildFilterClause([inFilter], {
320
+ region: ["US", "EU", "APAC"],
321
+ });
322
+ expect(clause).toBe(
323
+ "(`region` = 'US' or `region` = 'EU' or `region` = 'APAC')",
324
+ );
325
+ });
326
+
327
+ it("builds like predicate with auto-wrapping (case-insensitive)", () => {
328
+ const clause = buildFilterClause([likeFilter], {
329
+ name_search: "Smith",
330
+ });
331
+ expect(clause).toBe("lower(`customer_name`) ~ '%smith%'");
332
+ });
333
+
334
+ it("builds like predicate preserving existing wildcards", () => {
335
+ const clause = buildFilterClause([likeFilter], {
336
+ name_search: "%Smith%",
337
+ });
338
+ expect(clause).toBe("lower(`customer_name`) ~ '%smith%'");
339
+ });
340
+
341
+ it("builds greater_than predicate", () => {
342
+ const clause = buildFilterClause([gtFilter], {
343
+ start_date: "2024-01-01",
344
+ });
345
+ expect(clause).toBe("`created_at` > @2024-01-01");
346
+ });
347
+
348
+ it("builds less_than predicate", () => {
349
+ const clause = buildFilterClause([ltFilter], {
350
+ end_date: "2024-12-31",
351
+ });
352
+ expect(clause).toBe("`created_at` < @2024-12-31");
353
+ });
354
+
355
+ it("combines multiple filters with AND", () => {
356
+ const params: FilterParams = {
357
+ status: "active",
358
+ region: ["US", "EU"],
359
+ };
360
+ const clause = buildFilterClause([equalFilter, inFilter], params);
361
+ expect(clause).toBe(
362
+ "`status` = 'active' and (`region` = 'US' or `region` = 'EU')",
363
+ );
364
+ });
365
+
366
+ it("skips optional filters with no value", () => {
367
+ const params: FilterParams = {
368
+ status: "active",
369
+ };
370
+ const clause = buildFilterClause([equalFilter, inFilter], params);
371
+ expect(clause).toBe("`status` = 'active'");
372
+ });
373
+
374
+ it("throws on missing required filter", () => {
375
+ expect(() => buildFilterClause([requiredFilter], {})).toThrow(
376
+ FilterValidationError,
377
+ );
378
+ expect(() => buildFilterClause([requiredFilter], {})).toThrow(
379
+ 'Required filter "tenant_id"',
380
+ );
381
+ });
382
+
383
+ it("builds clause for required filter when value provided", () => {
384
+ const clause = buildFilterClause([requiredFilter], {
385
+ tenant_id: "abc123",
386
+ });
387
+ expect(clause).toBe("`tenant_id` = 'abc123'");
388
+ });
389
+
390
+ it("escapes single quotes in values", () => {
391
+ const clause = buildFilterClause([equalFilter], {
392
+ status: "it's active",
393
+ });
394
+ expect(clause).toBe("`status` = 'it\\'s active'");
395
+ });
396
+
397
+ it("escapes backslashes in values", () => {
398
+ const clause = buildFilterClause([likeFilter], {
399
+ name_search: "foo\\bar",
400
+ });
401
+ expect(clause).toBe("lower(`customer_name`) ~ '%foo\\\\bar%'");
402
+ });
403
+
404
+ it("ignores params that don't match any filter", () => {
405
+ const clause = buildFilterClause([equalFilter], {
406
+ status: "active",
407
+ unknown_param: "ignored",
408
+ });
409
+ expect(clause).toBe("`status` = 'active'");
410
+ });
411
+ });
412
+
413
+ // -----------------------------------------------------------------------
414
+ // injectFilterRefinement
415
+ // -----------------------------------------------------------------------
416
+ describe("injectFilterRefinement", () => {
417
+ it("returns original query when clause is empty", () => {
418
+ const query = "run: orders -> summary";
419
+ expect(injectFilterRefinement(query, "")).toBe(query);
420
+ });
421
+
422
+ it("appends refinement to named view query", () => {
423
+ const query = "run: orders -> summary";
424
+ const clause = "`status` = 'active'";
425
+ expect(injectFilterRefinement(query, clause)).toBe(
426
+ "run: orders -> summary + {where: `status` = 'active'}",
427
+ );
428
+ });
429
+
430
+ it("appends refinement to ad-hoc query", () => {
431
+ const query =
432
+ "run: orders -> { group_by: status; aggregate: order_count }";
433
+ const clause = "`region` = 'US'";
434
+ expect(injectFilterRefinement(query, clause)).toBe(
435
+ "run: orders -> { group_by: status; aggregate: order_count } + {where: `region` = 'US'}",
436
+ );
437
+ });
438
+
439
+ it("trims trailing whitespace before appending", () => {
440
+ const query = "run: orders -> summary \n ";
441
+ const clause = "`status` = 'active'";
442
+ expect(injectFilterRefinement(query, clause)).toBe(
443
+ "run: orders -> summary + {where: `status` = 'active'}",
444
+ );
445
+ });
446
+ });
447
+ });