mirror of
https://github.com/gristlabs/grist-core.git
synced 2024-10-27 20:44:07 +00:00
(core) add an option to action summarization to preserve columns entirely
Summary: Action summaries by default will drop rows in bulk changes, keeping only a few of them as examples. This diff allows overriding that, or selectively preserving some columns in their entirety. This is intended for use with webhooks. Test Plan: added test Reviewers: alexmojaki Reviewed By: alexmojaki Differential Revision: https://phab.getgrist.com/D3098
This commit is contained in:
parent
35e18cc0ad
commit
6c53f3e820
@ -12,131 +12,152 @@ import toPairs = require('lodash/toPairs');
|
|||||||
import values = require('lodash/values');
|
import values = require('lodash/values');
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The maximum number of rows in a single bulk change that will be recorded
|
* The default maximum number of rows in a single bulk change that will be recorded
|
||||||
* individually. Bulk changes that touch more than this number of rows
|
* individually. Bulk changes that touch more than this number of rows
|
||||||
* will be summarized only by the number of rows touched.
|
* will be summarized only by the number of rows touched.
|
||||||
*/
|
*/
|
||||||
const MAXIMUM_INLINE_ROWS = 10;
|
const MAXIMUM_INLINE_ROWS = 10;
|
||||||
|
|
||||||
/** helper function to access summary changes for a specific table by name */
|
|
||||||
function _forTable(summary: ActionSummary, tableId: string): TableDelta {
|
|
||||||
return summary.tableDeltas[tableId] || (summary.tableDeltas[tableId] = createEmptyTableDelta());
|
|
||||||
}
|
|
||||||
|
|
||||||
/** helper function to access summary changes for a specific cell by rowId and colId */
|
|
||||||
function _forCell(td: TableDelta, rowId: number, colId: string): CellDelta {
|
|
||||||
const cd = td.columnDeltas[colId] || (td.columnDeltas[colId] = {});
|
|
||||||
return cd[rowId] || (cd[rowId] = [null, null]);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* helper function to store detailed cell changes for a single row.
|
* Options when producing an action sumary.
|
||||||
* Direction parameter is 0 if values are prior values of cells, 1 if values are new values.
|
|
||||||
*/
|
*/
|
||||||
function _addRow(td: TableDelta, rowId: number, colValues: Action.ColValues,
|
export interface ActionSummaryOptions {
|
||||||
direction: 0|1) {
|
maximumInlineRows?: number; // Overrides the maximum number of rows in a
|
||||||
for (const [colId, colChanges] of toPairs(colValues)) {
|
// single bulk change that will be recorded individually.
|
||||||
const cell = _forCell(td, rowId, colId);
|
alwaysPreserveColIds?: string[]; // If set, all cells in these columns are preserved
|
||||||
cell[direction] = [colChanges];
|
// regardless of maximumInlineRows setting.
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/** helper function to store detailed cell changes for a set of rows */
|
class ActionSummarizer {
|
||||||
function _addRows(tableId: string, td: TableDelta, rowIds: number[],
|
|
||||||
colValues: Action.BulkColValues, direction: 0|1) {
|
|
||||||
let rows: Array<[number, number]>;
|
|
||||||
if (rowIds.length <= MAXIMUM_INLINE_ROWS || tableId.startsWith("_grist_")) {
|
|
||||||
rows = [...rowIds.entries()];
|
|
||||||
} else {
|
|
||||||
// if many rows, just take some from start and one from end as examples
|
|
||||||
rows = [...rowIds.slice(0, MAXIMUM_INLINE_ROWS - 1).entries()];
|
|
||||||
rows.push([rowIds.length - 1, rowIds[rowIds.length - 1]]);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const [colId, colChanges] of toPairs(colValues)) {
|
constructor(private _options?: ActionSummaryOptions) {}
|
||||||
rows.forEach(([idx, rowId]) => {
|
|
||||||
const cell = _forCell(td, rowId, colId);
|
|
||||||
cell[direction] = [colChanges[idx]];
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/** add a rename to a list, avoiding duplicates */
|
/** add information about an action based on the forward direction */
|
||||||
function _addRename(renames: LabelDelta[], rename: LabelDelta) {
|
public addForwardAction(summary: ActionSummary, act: DocAction) {
|
||||||
if (renames.find(r => r[0] === rename[0] && r[1] === rename[1])) { return; }
|
const tableId = act[1];
|
||||||
renames.push(rename);
|
if (Action.isAddTable(act)) {
|
||||||
}
|
summary.tableRenames.push([null, tableId]);
|
||||||
|
for (const info of act[2]) {
|
||||||
/** add information about an action based on the forward direction */
|
this._forTable(summary, tableId).columnRenames.push([null, info.id]);
|
||||||
function _addForwardAction(summary: ActionSummary, act: DocAction) {
|
}
|
||||||
const tableId = act[1];
|
} else if (Action.isRenameTable(act)) {
|
||||||
if (Action.isAddTable(act)) {
|
this._addRename(summary.tableRenames, [tableId, act[2]]);
|
||||||
summary.tableRenames.push([null, tableId]);
|
} else if (Action.isRenameColumn(act)) {
|
||||||
for (const info of act[2]) {
|
this._addRename(this._forTable(summary, tableId).columnRenames, [act[2], act[3]]);
|
||||||
_forTable(summary, tableId).columnRenames.push([null, info.id]);
|
} else if (Action.isAddColumn(act)) {
|
||||||
|
this._forTable(summary, tableId).columnRenames.push([null, act[2]]);
|
||||||
|
} else if (Action.isRemoveColumn(act)) {
|
||||||
|
this._forTable(summary, tableId).columnRenames.push([act[2], null]);
|
||||||
|
} else if (Action.isAddRecord(act)) {
|
||||||
|
const td = this._forTable(summary, tableId);
|
||||||
|
td.addRows.push(act[2]);
|
||||||
|
this._addRow(td, act[2], act[3], 1);
|
||||||
|
} else if (Action.isUpdateRecord(act)) {
|
||||||
|
const td = this._forTable(summary, tableId);
|
||||||
|
td.updateRows.push(act[2]);
|
||||||
|
this._addRow(td, act[2], act[3], 1);
|
||||||
|
} else if (Action.isBulkAddRecord(act)) {
|
||||||
|
const td = this._forTable(summary, tableId);
|
||||||
|
arrayExtend(td.addRows, act[2]);
|
||||||
|
this._addRows(tableId, td, act[2], act[3], 1);
|
||||||
|
} else if (Action.isBulkUpdateRecord(act)) {
|
||||||
|
const td = this._forTable(summary, tableId);
|
||||||
|
arrayExtend(td.updateRows, act[2]);
|
||||||
|
this._addRows(tableId, td, act[2], act[3], 1);
|
||||||
|
} else if (Action.isReplaceTableData(act)) {
|
||||||
|
const td = this._forTable(summary, tableId);
|
||||||
|
arrayExtend(td.addRows, act[2]);
|
||||||
|
this._addRows(tableId, td, act[2], act[3], 1);
|
||||||
}
|
}
|
||||||
} else if (Action.isRenameTable(act)) {
|
|
||||||
_addRename(summary.tableRenames, [tableId, act[2]]);
|
|
||||||
} else if (Action.isRenameColumn(act)) {
|
|
||||||
_addRename(_forTable(summary, tableId).columnRenames, [act[2], act[3]]);
|
|
||||||
} else if (Action.isAddColumn(act)) {
|
|
||||||
_forTable(summary, tableId).columnRenames.push([null, act[2]]);
|
|
||||||
} else if (Action.isRemoveColumn(act)) {
|
|
||||||
_forTable(summary, tableId).columnRenames.push([act[2], null]);
|
|
||||||
} else if (Action.isAddRecord(act)) {
|
|
||||||
const td = _forTable(summary, tableId);
|
|
||||||
td.addRows.push(act[2]);
|
|
||||||
_addRow(td, act[2], act[3], 1);
|
|
||||||
} else if (Action.isUpdateRecord(act)) {
|
|
||||||
const td = _forTable(summary, tableId);
|
|
||||||
td.updateRows.push(act[2]);
|
|
||||||
_addRow(td, act[2], act[3], 1);
|
|
||||||
} else if (Action.isBulkAddRecord(act)) {
|
|
||||||
const td = _forTable(summary, tableId);
|
|
||||||
arrayExtend(td.addRows, act[2]);
|
|
||||||
_addRows(tableId, td, act[2], act[3], 1);
|
|
||||||
} else if (Action.isBulkUpdateRecord(act)) {
|
|
||||||
const td = _forTable(summary, tableId);
|
|
||||||
arrayExtend(td.updateRows, act[2]);
|
|
||||||
_addRows(tableId, td, act[2], act[3], 1);
|
|
||||||
} else if (Action.isReplaceTableData(act)) {
|
|
||||||
const td = _forTable(summary, tableId);
|
|
||||||
arrayExtend(td.addRows, act[2]);
|
|
||||||
_addRows(tableId, td, act[2], act[3], 1);
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
/** add information about an action based on undo information */
|
/** add information about an action based on undo information */
|
||||||
function _addReverseAction(summary: ActionSummary, act: DocAction) {
|
public addReverseAction(summary: ActionSummary, act: DocAction) {
|
||||||
const tableId = act[1];
|
const tableId = act[1];
|
||||||
if (Action.isAddTable(act)) { // undoing, so this is a table removal
|
if (Action.isAddTable(act)) { // undoing, so this is a table removal
|
||||||
summary.tableRenames.push([tableId, null]);
|
summary.tableRenames.push([tableId, null]);
|
||||||
for (const info of act[2]) {
|
for (const info of act[2]) {
|
||||||
_forTable(summary, tableId).columnRenames.push([info.id, null]);
|
this._forTable(summary, tableId).columnRenames.push([info.id, null]);
|
||||||
|
}
|
||||||
|
} else if (Action.isAddRecord(act)) { // undoing, so this is a record removal
|
||||||
|
const td = this._forTable(summary, tableId);
|
||||||
|
td.removeRows.push(act[2]);
|
||||||
|
this._addRow(td, act[2], act[3], 0);
|
||||||
|
} else if (Action.isUpdateRecord(act)) { // undoing, so this is reversal of a record update
|
||||||
|
const td = this._forTable(summary, tableId);
|
||||||
|
this._addRow(td, act[2], act[3], 0);
|
||||||
|
} else if (Action.isBulkAddRecord(act)) { // undoing, this may be reversing a table delete
|
||||||
|
const td = this._forTable(summary, tableId);
|
||||||
|
arrayExtend(td.removeRows, act[2]);
|
||||||
|
this._addRows(tableId, td, act[2], act[3], 0);
|
||||||
|
} else if (Action.isBulkUpdateRecord(act)) { // undoing, so this is reversal of a bulk record update
|
||||||
|
const td = this._forTable(summary, tableId);
|
||||||
|
arrayExtend(td.updateRows, act[2]);
|
||||||
|
this._addRows(tableId, td, act[2], act[3], 0);
|
||||||
|
} else if (Action.isRenameTable(act)) { // undoing - sometimes renames only in undo info
|
||||||
|
this._addRename(summary.tableRenames, [act[2], tableId]);
|
||||||
|
} else if (Action.isRenameColumn(act)) { // undoing - sometimes renames only in undo info
|
||||||
|
this._addRename(this._forTable(summary, tableId).columnRenames, [act[3], act[2]]);
|
||||||
|
} else if (Action.isReplaceTableData(act)) { // undoing
|
||||||
|
const td = this._forTable(summary, tableId);
|
||||||
|
arrayExtend(td.removeRows, act[2]);
|
||||||
|
this._addRows(tableId, td, act[2], act[3], 0);
|
||||||
}
|
}
|
||||||
} else if (Action.isAddRecord(act)) { // undoing, so this is a record removal
|
}
|
||||||
const td = _forTable(summary, tableId);
|
|
||||||
td.removeRows.push(act[2]);
|
/** helper function to access summary changes for a specific table by name */
|
||||||
_addRow(td, act[2], act[3], 0);
|
private _forTable(summary: ActionSummary, tableId: string): TableDelta {
|
||||||
} else if (Action.isUpdateRecord(act)) { // undoing, so this is reversal of a record update
|
return summary.tableDeltas[tableId] || (summary.tableDeltas[tableId] = createEmptyTableDelta());
|
||||||
const td = _forTable(summary, tableId);
|
}
|
||||||
_addRow(td, act[2], act[3], 0);
|
|
||||||
} else if (Action.isBulkAddRecord(act)) { // undoing, this may be reversing a table delete
|
/** helper function to access summary changes for a specific cell by rowId and colId */
|
||||||
const td = _forTable(summary, tableId);
|
private _forCell(td: TableDelta, rowId: number, colId: string): CellDelta {
|
||||||
arrayExtend(td.removeRows, act[2]);
|
const cd = td.columnDeltas[colId] || (td.columnDeltas[colId] = {});
|
||||||
_addRows(tableId, td, act[2], act[3], 0);
|
return cd[rowId] || (cd[rowId] = [null, null]);
|
||||||
} else if (Action.isBulkUpdateRecord(act)) { // undoing, so this is reversal of a bulk record update
|
}
|
||||||
const td = _forTable(summary, tableId);
|
|
||||||
arrayExtend(td.updateRows, act[2]);
|
/**
|
||||||
_addRows(tableId, td, act[2], act[3], 0);
|
* helper function to store detailed cell changes for a single row.
|
||||||
} else if (Action.isRenameTable(act)) { // undoing - sometimes renames only in undo info
|
* Direction parameter is 0 if values are prior values of cells, 1 if values are new values.
|
||||||
_addRename(summary.tableRenames, [act[2], tableId]);
|
*/
|
||||||
} else if (Action.isRenameColumn(act)) { // undoing - sometimes renames only in undo info
|
private _addRow(td: TableDelta, rowId: number, colValues: Action.ColValues,
|
||||||
_addRename(_forTable(summary, tableId).columnRenames, [act[3], act[2]]);
|
direction: 0|1) {
|
||||||
} else if (Action.isReplaceTableData(act)) { // undoing
|
for (const [colId, colChanges] of toPairs(colValues)) {
|
||||||
const td = _forTable(summary, tableId);
|
const cell = this._forCell(td, rowId, colId);
|
||||||
arrayExtend(td.removeRows, act[2]);
|
cell[direction] = [colChanges];
|
||||||
_addRows(tableId, td, act[2], act[3], 0);
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** helper function to store detailed cell changes for a set of rows */
|
||||||
|
private _addRows(tableId: string, td: TableDelta, rowIds: number[],
|
||||||
|
colValues: Action.BulkColValues, direction: 0|1) {
|
||||||
|
const maximumInlineRows = this._options?.maximumInlineRows || MAXIMUM_INLINE_ROWS;
|
||||||
|
const limitRows: boolean = rowIds.length > maximumInlineRows && !tableId.startsWith("_grist_");
|
||||||
|
let selectedRows: Array<[number, number]> = [];
|
||||||
|
if (limitRows) {
|
||||||
|
// if many rows, just take some from start and one from end as examples
|
||||||
|
selectedRows = [...rowIds.slice(0, maximumInlineRows - 1).entries()];
|
||||||
|
selectedRows.push([rowIds.length - 1, rowIds[rowIds.length - 1]]);
|
||||||
|
}
|
||||||
|
|
||||||
|
const alwaysPreserveColIds = new Set(this._options?.alwaysPreserveColIds || []);
|
||||||
|
for (const [colId, colChanges] of toPairs(colValues)) {
|
||||||
|
const addCellToSummary = (rowId: number, idx: number) => {
|
||||||
|
const cell = this._forCell(td, rowId, colId);
|
||||||
|
cell[direction] = [colChanges[idx]];
|
||||||
|
};
|
||||||
|
if (!limitRows || alwaysPreserveColIds.has(colId)) {
|
||||||
|
rowIds.forEach(addCellToSummary);
|
||||||
|
} else {
|
||||||
|
selectedRows.forEach(([idx, rowId]) => addCellToSummary(rowId, idx));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** add a rename to a list, avoiding duplicates */
|
||||||
|
private _addRename(renames: LabelDelta[], rename: LabelDelta) {
|
||||||
|
if (renames.find(r => r[0] === rename[0] && r[1] === rename[1])) { return; }
|
||||||
|
renames.push(rename);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -144,13 +165,14 @@ function _addReverseAction(summary: ActionSummary, act: DocAction) {
|
|||||||
* Summarize the tabular changes that a LocalActionBundle results in, in a form
|
* Summarize the tabular changes that a LocalActionBundle results in, in a form
|
||||||
* that will be suitable for composition.
|
* that will be suitable for composition.
|
||||||
*/
|
*/
|
||||||
export function summarizeAction(body: LocalActionBundle): ActionSummary {
|
export function summarizeAction(body: LocalActionBundle, options?: ActionSummaryOptions): ActionSummary {
|
||||||
|
const summarizer = new ActionSummarizer(options);
|
||||||
const summary = createEmptyActionSummary();
|
const summary = createEmptyActionSummary();
|
||||||
for (const act of getEnvContent(body.stored)) {
|
for (const act of getEnvContent(body.stored)) {
|
||||||
_addForwardAction(summary, act);
|
summarizer.addForwardAction(summary, act);
|
||||||
}
|
}
|
||||||
for (const act of Array.from(body.undo).reverse()) {
|
for (const act of Array.from(body.undo).reverse()) {
|
||||||
_addReverseAction(summary, act);
|
summarizer.addReverseAction(summary, act);
|
||||||
}
|
}
|
||||||
// Name tables consistently, by their ultimate name, now we know it.
|
// Name tables consistently, by their ultimate name, now we know it.
|
||||||
for (const renames of summary.tableRenames) {
|
for (const renames of summary.tableRenames) {
|
||||||
|
Loading…
Reference in New Issue
Block a user