Skip to content

Commit

Permalink
[FIX] pivot: Prevent faulty pivot domain to crash
Browse files Browse the repository at this point in the history
The getter `getPivotCellFromPosition` parses pivot cells domain args
but this step assumes that the domain is valid (e.g. refers to a valid
groupby,measure combination). Unfortunately, if the domain is invalid,
the parsing will crash.

Similarly to the decision made for the getter `evaluateFormula`[^1]
we wrap the parsing in a try/catch statement and in case of faulty evaluation,
return the same result as for invalid/non-existing pivots, that is an
empty Pivotcell.

[^1]: See #3371

closes #4741

Task: 4088765
Signed-off-by: Lucas Lefèvre (lul) <[email protected]>
  • Loading branch information
rrahir committed Aug 9, 2024
1 parent 29775e1 commit 693db2b
Show file tree
Hide file tree
Showing 2 changed files with 46 additions and 22 deletions.
46 changes: 25 additions & 21 deletions src/plugins/ui_core_views/pivot_ui.ts
Original file line number Diff line number Diff line change
Expand Up @@ -207,33 +207,37 @@ export class PivotUIPlugin extends UIPlugin {
const pivotRow = position.row - mainPosition.row;
return pivotCells[pivotCol][pivotRow];
}
if (functionName === "PIVOT.HEADER" && args.at(-2) === "measure") {
const domain = pivot.parseArgsToPivotDomain(
args.slice(1, -2).map((value) => ({ value } as FunctionResultObject))
);
return {
type: "MEASURE_HEADER",
domain,
measure: args.at(-1)?.toString() || "",
};
} else if (functionName === "PIVOT.HEADER") {
try {
if (functionName === "PIVOT.HEADER" && args.at(-2) === "measure") {
const domain = pivot.parseArgsToPivotDomain(
args.slice(1, -2).map((value) => ({ value } as FunctionResultObject))
);
return {
type: "MEASURE_HEADER",
domain,
measure: args.at(-1)?.toString() || "",
};
} else if (functionName === "PIVOT.HEADER") {
const domain = pivot.parseArgsToPivotDomain(
args.slice(1).map((value) => ({ value } as FunctionResultObject))
);
return {
type: "HEADER",
domain,
};
}
const [measure, ...domainArgs] = args.slice(1);
const domain = pivot.parseArgsToPivotDomain(
args.slice(1).map((value) => ({ value } as FunctionResultObject))
domainArgs.map((value) => ({ value } as FunctionResultObject))
);
return {
type: "HEADER",
type: "VALUE",
domain,
measure: measure?.toString() || "",
};
} catch (_) {
return EMPTY_PIVOT_CELL;
}
const [measure, ...domainArgs] = args.slice(1);
const domain = pivot.parseArgsToPivotDomain(
domainArgs.map((value) => ({ value } as FunctionResultObject))
);
return {
type: "VALUE",
domain,
measure: measure?.toString() || "",
};
}

getPivot(pivotId: UID) {
Expand Down
22 changes: 21 additions & 1 deletion tests/pivots/pivot_plugin.test.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import { setCellContent } from "../test_helpers/commands_helpers";
import { EMPTY_PIVOT_CELL } from "../../src/helpers/pivot/table_spreadsheet_pivot";
import { selectCell, setCellContent } from "../test_helpers/commands_helpers";
import { createModelFromGrid, toCellPosition } from "../test_helpers/helpers";
import { addPivot } from "../test_helpers/pivot_helpers";

Expand Down Expand Up @@ -28,4 +29,23 @@ describe("Pivot plugin", () => {
setCellContent(model, "G1", "=PIVOT.HEADER(1)");
expect(isSpillPivotFormula("G1")).toBe(false);
});

test("getPivotCellFromPosition doesn't throw with invalid pivot domain", () => {
// prettier-ignore
const grid = {
A1: "Customer", B1: "Price", C1: '=PIVOT.VALUE(1,"Price","5","Bob")',
A2: "Alice", B2: "10",
A3: "Bob", B3: "30",
};
const model = createModelFromGrid(grid);
addPivot(model, "A1:B3", {
columns: [],
rows: [{ name: "Customer" }],
measures: [{ name: "Price", aggregator: "sum" }],
});
selectCell(model, "C1");
expect(model.getters.getPivotCellFromPosition(model.getters.getActivePosition())).toMatchObject(
EMPTY_PIVOT_CELL
);
});
});

0 comments on commit 693db2b

Please sign in to comment.