Skip to content
Snippets Groups Projects
Commit 757ff5e8 authored by Ariya Hidayat's avatar Ariya Hidayat
Browse files

MBQL syntax sugar (for negative filter) in the custom expression compiler (#14938)

parent 1bbb22fd
No related branches found
No related tags found
No related merge requests found
......@@ -10,6 +10,11 @@ import {
import { ExpressionCstVisitor, parse } from "./parser";
const NEGATIVE_FILTER_SHORTHANDS = {
contains: "does-not-contain",
"is-null": "not-null",
"is-empty": "not-empty",
};
class ExpressionMBQLCompilerVisitor extends ExpressionCstVisitor {
constructor(options) {
super();
......@@ -50,7 +55,10 @@ class ExpressionMBQLCompilerVisitor extends ExpressionCstVisitor {
return this.visit(ctx.expression);
}
logicalNotExpression(ctx) {
return ["not", this.visit(ctx.operands[0])];
const expr = this.visit(ctx.operands[0]);
const [fn, ...args] = expr;
const shorthand = NEGATIVE_FILTER_SHORTHANDS[fn];
return shorthand ? [shorthand, ...args] : ["not", expr];
}
relationalExpression(ctx) {
return this._collapseOperators(ctx.operands, ctx.operators);
......
......@@ -161,6 +161,13 @@ const filter = [
["or", ["not", ["between", subtotal, 3, 14]], segment],
"filter function with OR",
],
[
'NOT contains([User → Name], "John")',
["does-not-contain", userName, "John"],
"not contains",
],
["NOT isnull([Tax])", ["not-null", tax], "not null"],
["NOT isempty([Total])", ["not-empty", total], "not empty"],
];
export default [
......
......@@ -191,6 +191,15 @@ describe("metabase/lib/expressions/compile", () => {
expect(filter("[Expensive]")).toEqual(["segment", "Expensive"]);
expect(filter("NOT [Good]")).toEqual(["not", ["segment", "Good"]]);
});
it("should compile negative filters", () => {
expect(filter("NOT CONTAINS('X','Y')")).toEqual([
"does-not-contain",
"X",
"Y",
]);
expect(filter("NOT ISNULL('P')")).toEqual(["not-null", "P"]);
expect(filter("NOT ISEMPTY('Q')")).toEqual(["not-empty", "Q"]);
});
});
describe("(for an aggregation)", () => {
......
......@@ -868,7 +868,7 @@ describe("scenarios > question > filter", () => {
"source-table": PRODUCTS_ID,
filter: [
"does-not-contain",
["field-id", PRODUCTS.TITLE],
["field", PRODUCTS.TITLE, null],
"Wallet",
{ "case-sensitive": false },
],
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment