@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.
- package/dist/app/api-doc.yaml +91 -1
- package/dist/app/assets/{HomePage-B0C6gwGj.js → HomePage-or6BbD5P.js} +1 -1
- package/dist/app/assets/{MainPage-B53xidTF.js → MainPage-DINuSDg0.js} +2 -2
- package/dist/app/assets/{ModelPage-UMuQe8qY.js → ModelPage-BMcaV1YQ.js} +1 -1
- package/dist/app/assets/{PackagePage-BEDvm_je.js → PackagePage-DXxlQcCj.js} +1 -1
- package/dist/app/assets/{ProjectPage-DzN4P86H.js → ProjectPage-vfZc_Kvu.js} +1 -1
- package/dist/app/assets/{RouteError-Cv58zNpb.js → RouteError-r14osUo0.js} +1 -1
- package/dist/app/assets/{WorkbookPage-DZ1StqsX.js → WorkbookPage-HI39NTWs.js} +1 -1
- package/dist/app/assets/{index-D-xPyBUA.js → index-Bw1lh09G.js} +78 -78
- package/dist/app/assets/{index-DPThhVfX.js → index-Dd6uCk_C.js} +1 -1
- package/dist/app/assets/{index-M3Zo817E.js → index-JqHhhRqY.js} +1 -1
- package/dist/app/assets/{index.umd-DnfBsVqO.js → index.umd-lwkX_kFe.js} +1 -1
- package/dist/app/index.html +1 -1
- package/dist/server.js +323 -31
- package/package.json +1 -1
- package/src/controller/model.controller.ts +4 -1
- package/src/controller/query.controller.ts +5 -0
- package/src/mcp/resources/model_resource.ts +12 -9
- package/src/mcp/resources/source_resource.ts +7 -6
- package/src/mcp/resources/view_resource.ts +0 -1
- package/src/mcp/tools/execute_query_tool.ts +9 -0
- package/src/server.ts +21 -0
- package/src/service/filter.spec.ts +392 -0
- package/src/service/filter.ts +332 -0
- package/src/service/filter_integration.spec.ts +622 -0
- 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
|
+
});
|