@malloy-publisher/server 0.0.181 → 0.0.182

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 (26) hide show
  1. package/dist/app/api-doc.yaml +91 -1
  2. package/dist/app/assets/{HomePage-B0C6gwGj.js → HomePage-or6BbD5P.js} +1 -1
  3. package/dist/app/assets/{MainPage-B53xidTF.js → MainPage-DINuSDg0.js} +2 -2
  4. package/dist/app/assets/{ModelPage-UMuQe8qY.js → ModelPage-BMcaV1YQ.js} +1 -1
  5. package/dist/app/assets/{PackagePage-BEDvm_je.js → PackagePage-DXxlQcCj.js} +1 -1
  6. package/dist/app/assets/{ProjectPage-DzN4P86H.js → ProjectPage-vfZc_Kvu.js} +1 -1
  7. package/dist/app/assets/{RouteError-Cv58zNpb.js → RouteError-r14osUo0.js} +1 -1
  8. package/dist/app/assets/{WorkbookPage-DZ1StqsX.js → WorkbookPage-HI39NTWs.js} +1 -1
  9. package/dist/app/assets/{index-D-xPyBUA.js → index-Bw1lh09G.js} +78 -78
  10. package/dist/app/assets/{index-DPThhVfX.js → index-Dd6uCk_C.js} +1 -1
  11. package/dist/app/assets/{index-M3Zo817E.js → index-JqHhhRqY.js} +1 -1
  12. package/dist/app/assets/{index.umd-DnfBsVqO.js → index.umd-lwkX_kFe.js} +1 -1
  13. package/dist/app/index.html +1 -1
  14. package/dist/server.js +323 -31
  15. package/package.json +1 -1
  16. package/src/controller/model.controller.ts +4 -1
  17. package/src/controller/query.controller.ts +5 -0
  18. package/src/mcp/resources/model_resource.ts +12 -9
  19. package/src/mcp/resources/source_resource.ts +7 -6
  20. package/src/mcp/resources/view_resource.ts +0 -1
  21. package/src/mcp/tools/execute_query_tool.ts +9 -0
  22. package/src/server.ts +21 -0
  23. package/src/service/filter.spec.ts +392 -0
  24. package/src/service/filter.ts +332 -0
  25. package/src/service/filter_integration.spec.ts +622 -0
  26. package/src/service/model.ts +180 -43
@@ -24,6 +24,12 @@ const executeQueryShape = {
24
24
  query: z.string().optional().describe("Ad-hoc Malloy query code"),
25
25
  sourceName: z.string().optional().describe("Source name for a view"),
26
26
  queryName: z.string().optional().describe("Named query or view"),
27
+ filterParams: z
28
+ .record(z.union([z.string(), z.array(z.string())]))
29
+ .optional()
30
+ .describe(
31
+ "Filter parameter values keyed by filter name. Used with sources that declare #(filter) annotations.",
32
+ ),
27
33
  };
28
34
 
29
35
  // Type inference is handled automatically by the MCP server based on the executeQueryShape
@@ -49,6 +55,7 @@ export function registerExecuteQueryTool(
49
55
  query,
50
56
  sourceName,
51
57
  queryName,
58
+ filterParams,
52
59
  } = params;
53
60
 
54
61
  logger.info("[MCP Tool executeQuery] Received params:", { params });
@@ -120,6 +127,7 @@ export function registerExecuteQueryTool(
120
127
  undefined,
121
128
  undefined,
122
129
  query,
130
+ filterParams,
123
131
  );
124
132
  const { validateRenderTags } = await import(
125
133
  "@malloydata/render-validator"
@@ -165,6 +173,7 @@ export function registerExecuteQueryTool(
165
173
  sourceName,
166
174
  queryName,
167
175
  undefined,
176
+ filterParams,
168
177
  );
169
178
  const { validateRenderTags } = await import(
170
179
  "@malloydata/render-validator"
package/src/server.ts CHANGED
@@ -817,12 +817,29 @@ app.get(
817
817
  // Express stores wildcard matches in params['0']
818
818
  const notebookPath = (req.params as Record<string, string>)["0"];
819
819
 
820
+ // Parse optional filter_params (JSON query string) and bypass_filters
821
+ let filterParams: Record<string, string | string[]> | undefined;
822
+ if (typeof req.query.filter_params === "string") {
823
+ try {
824
+ filterParams = JSON.parse(req.query.filter_params);
825
+ } catch {
826
+ res.status(400).json({
827
+ error: "Invalid filter_params: must be valid JSON",
828
+ });
829
+ return;
830
+ }
831
+ }
832
+ const bypassFilters =
833
+ req.query.bypass_filters === "true" ? true : undefined;
834
+
820
835
  res.status(200).json(
821
836
  await modelController.executeNotebookCell(
822
837
  req.params.projectName,
823
838
  req.params.packageName,
824
839
  notebookPath,
825
840
  cellIndex,
841
+ filterParams,
842
+ bypassFilters,
826
843
  ),
827
844
  );
828
845
  } catch (error) {
@@ -879,6 +896,10 @@ app.post(
879
896
  req.body.queryName as string,
880
897
  req.body.query as string,
881
898
  req.body.compactJson === true,
899
+ (req.body.filterParams ?? req.body.sourceFilters) as
900
+ | Record<string, string | string[]>
901
+ | undefined,
902
+ req.body.bypassFilters === true ? true : undefined,
882
903
  ),
883
904
  );
884
905
  } catch (error) {
@@ -0,0 +1,392 @@
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
+
175
+ // -----------------------------------------------------------------------
176
+ // buildFilterClause
177
+ // -----------------------------------------------------------------------
178
+ describe("buildFilterClause", () => {
179
+ const equalFilter: FilterDefinition = {
180
+ name: "status",
181
+ dimension: "status",
182
+ type: "equal",
183
+ implicit: false,
184
+ required: false,
185
+ };
186
+
187
+ const inFilter: FilterDefinition = {
188
+ name: "region",
189
+ dimension: "region",
190
+ type: "in",
191
+ implicit: false,
192
+ required: false,
193
+ };
194
+
195
+ const likeFilter: FilterDefinition = {
196
+ name: "name_search",
197
+ dimension: "customer_name",
198
+ type: "like",
199
+ implicit: false,
200
+ required: false,
201
+ };
202
+
203
+ const gtFilter: FilterDefinition = {
204
+ name: "start_date",
205
+ dimension: "created_at",
206
+ type: "greater_than",
207
+ implicit: false,
208
+ required: false,
209
+ };
210
+
211
+ const ltFilter: FilterDefinition = {
212
+ name: "end_date",
213
+ dimension: "created_at",
214
+ type: "less_than",
215
+ implicit: false,
216
+ required: false,
217
+ };
218
+
219
+ const requiredFilter: FilterDefinition = {
220
+ name: "tenant_id",
221
+ dimension: "tenant_id",
222
+ type: "equal",
223
+ implicit: true,
224
+ required: true,
225
+ };
226
+
227
+ it("returns empty string when no params provided", () => {
228
+ const clause = buildFilterClause([equalFilter], {});
229
+ expect(clause).toBe("");
230
+ });
231
+
232
+ it("returns empty string when param is empty string", () => {
233
+ const clause = buildFilterClause([equalFilter], { status: "" });
234
+ expect(clause).toBe("");
235
+ });
236
+
237
+ it("returns empty string when param is empty array", () => {
238
+ const clause = buildFilterClause([inFilter], { region: [] });
239
+ expect(clause).toBe("");
240
+ });
241
+
242
+ it("builds equal predicate", () => {
243
+ const clause = buildFilterClause([equalFilter], {
244
+ status: "active",
245
+ });
246
+ expect(clause).toBe("`status` = 'active'");
247
+ });
248
+
249
+ it("equal uses first element if given array", () => {
250
+ const clause = buildFilterClause([equalFilter], {
251
+ status: ["active", "pending"],
252
+ });
253
+ expect(clause).toBe("`status` = 'active'");
254
+ });
255
+
256
+ it("builds in predicate with single value", () => {
257
+ const clause = buildFilterClause([inFilter], {
258
+ region: ["US"],
259
+ });
260
+ expect(clause).toBe("`region` = 'US'");
261
+ });
262
+
263
+ it("builds in predicate with multiple values", () => {
264
+ const clause = buildFilterClause([inFilter], {
265
+ region: ["US", "EU", "APAC"],
266
+ });
267
+ expect(clause).toBe(
268
+ "(`region` = 'US' or `region` = 'EU' or `region` = 'APAC')",
269
+ );
270
+ });
271
+
272
+ it("builds like predicate with auto-wrapping (case-insensitive)", () => {
273
+ const clause = buildFilterClause([likeFilter], {
274
+ name_search: "Smith",
275
+ });
276
+ expect(clause).toBe("lower(`customer_name`) ~ '%smith%'");
277
+ });
278
+
279
+ it("builds like predicate preserving existing wildcards", () => {
280
+ const clause = buildFilterClause([likeFilter], {
281
+ name_search: "%Smith%",
282
+ });
283
+ expect(clause).toBe("lower(`customer_name`) ~ '%smith%'");
284
+ });
285
+
286
+ it("builds greater_than predicate", () => {
287
+ const clause = buildFilterClause([gtFilter], {
288
+ start_date: "2024-01-01",
289
+ });
290
+ expect(clause).toBe("`created_at` > @2024-01-01");
291
+ });
292
+
293
+ it("builds less_than predicate", () => {
294
+ const clause = buildFilterClause([ltFilter], {
295
+ end_date: "2024-12-31",
296
+ });
297
+ expect(clause).toBe("`created_at` < @2024-12-31");
298
+ });
299
+
300
+ it("combines multiple filters with AND", () => {
301
+ const params: FilterParams = {
302
+ status: "active",
303
+ region: ["US", "EU"],
304
+ };
305
+ const clause = buildFilterClause([equalFilter, inFilter], params);
306
+ expect(clause).toBe(
307
+ "`status` = 'active' and (`region` = 'US' or `region` = 'EU')",
308
+ );
309
+ });
310
+
311
+ it("skips optional filters with no value", () => {
312
+ const params: FilterParams = {
313
+ status: "active",
314
+ };
315
+ const clause = buildFilterClause([equalFilter, inFilter], params);
316
+ expect(clause).toBe("`status` = 'active'");
317
+ });
318
+
319
+ it("throws on missing required filter", () => {
320
+ expect(() => buildFilterClause([requiredFilter], {})).toThrow(
321
+ FilterValidationError,
322
+ );
323
+ expect(() => buildFilterClause([requiredFilter], {})).toThrow(
324
+ 'Required filter "tenant_id"',
325
+ );
326
+ });
327
+
328
+ it("builds clause for required filter when value provided", () => {
329
+ const clause = buildFilterClause([requiredFilter], {
330
+ tenant_id: "abc123",
331
+ });
332
+ expect(clause).toBe("`tenant_id` = 'abc123'");
333
+ });
334
+
335
+ it("escapes single quotes in values", () => {
336
+ const clause = buildFilterClause([equalFilter], {
337
+ status: "it's active",
338
+ });
339
+ expect(clause).toBe("`status` = 'it\\'s active'");
340
+ });
341
+
342
+ it("escapes backslashes in values", () => {
343
+ const clause = buildFilterClause([likeFilter], {
344
+ name_search: "foo\\bar",
345
+ });
346
+ expect(clause).toBe("lower(`customer_name`) ~ '%foo\\\\bar%'");
347
+ });
348
+
349
+ it("ignores params that don't match any filter", () => {
350
+ const clause = buildFilterClause([equalFilter], {
351
+ status: "active",
352
+ unknown_param: "ignored",
353
+ });
354
+ expect(clause).toBe("`status` = 'active'");
355
+ });
356
+ });
357
+
358
+ // -----------------------------------------------------------------------
359
+ // injectFilterRefinement
360
+ // -----------------------------------------------------------------------
361
+ describe("injectFilterRefinement", () => {
362
+ it("returns original query when clause is empty", () => {
363
+ const query = "run: orders -> summary";
364
+ expect(injectFilterRefinement(query, "")).toBe(query);
365
+ });
366
+
367
+ it("appends refinement to named view query", () => {
368
+ const query = "run: orders -> summary";
369
+ const clause = "`status` = 'active'";
370
+ expect(injectFilterRefinement(query, clause)).toBe(
371
+ "run: orders -> summary + {where: `status` = 'active'}",
372
+ );
373
+ });
374
+
375
+ it("appends refinement to ad-hoc query", () => {
376
+ const query =
377
+ "run: orders -> { group_by: status; aggregate: order_count }";
378
+ const clause = "`region` = 'US'";
379
+ expect(injectFilterRefinement(query, clause)).toBe(
380
+ "run: orders -> { group_by: status; aggregate: order_count } + {where: `region` = 'US'}",
381
+ );
382
+ });
383
+
384
+ it("trims trailing whitespace before appending", () => {
385
+ const query = "run: orders -> summary \n ";
386
+ const clause = "`status` = 'active'";
387
+ expect(injectFilterRefinement(query, clause)).toBe(
388
+ "run: orders -> summary + {where: `status` = 'active'}",
389
+ );
390
+ });
391
+ });
392
+ });