(core) Fix issue with lodash's map interpreting objects with length as array-like

Summary:
Here's a series of badness that easily leads to a crash, in reverse order:
- Lodash's map() function interprets an object with a .length property as an array.
- Some very old code generated human-friendly descriptions of user actions,
  applying map() to parts of them. It so happens that this generated description
  isn't even used.
- If a user action is encountered with a sufficiently large length propery,
  map() would exhaust the server memory.

Fixed by removing old unneeded code, and replacing some other occurrences of
lodash's map() with native equivalents.

Test Plan: Tested manually on a local reproduction of the issue.

Reviewers: paulfitz

Reviewed By: paulfitz

Subscribers: paulfitz

Differential Revision: https://phab.getgrist.com/D3938
This commit is contained in:
Dmitry S 2023-07-01 14:31:21 -04:00
parent b0aa17c932
commit 2b581ab7dc
6 changed files with 16 additions and 119 deletions

View File

@ -6,7 +6,6 @@ import * as dispose from 'app/client/lib/dispose';
import dom from 'app/client/lib/dom'; import dom from 'app/client/lib/dom';
import {timeFormat} from 'app/common/timeFormat'; import {timeFormat} from 'app/common/timeFormat';
import * as ko from 'knockout'; import * as ko from 'knockout';
import map = require('lodash/map');
import koArray from 'app/client/lib/koArray'; import koArray from 'app/client/lib/koArray';
import {KoArray} from 'app/client/lib/koArray'; import {KoArray} from 'app/client/lib/koArray';
@ -17,8 +16,8 @@ import {GristDoc} from 'app/client/components/GristDoc';
import {ActionGroup} from 'app/common/ActionGroup'; import {ActionGroup} from 'app/common/ActionGroup';
import {ActionSummary, asTabularDiffs, defunctTableName, getAffectedTables, import {ActionSummary, asTabularDiffs, defunctTableName, getAffectedTables,
LabelDelta} from 'app/common/ActionSummary'; LabelDelta} from 'app/common/ActionSummary';
import {CellDelta} from 'app/common/TabularDiff'; import {CellDelta, TabularDiff} from 'app/common/TabularDiff';
import {IDomComponent} from 'grainjs'; import {DomContents, IDomComponent} from 'grainjs';
import {makeT} from 'app/client/lib/localization'; import {makeT} from 'app/client/lib/localization';
/** /**
@ -141,12 +140,12 @@ export class ActionLog extends dispose.Disposable implements IDomComponent {
* @param {string} txt - a textual description of the action * @param {string} txt - a textual description of the action
* @param {ActionGroupWithState} ag - the full action information we have * @param {ActionGroupWithState} ag - the full action information we have
*/ */
public renderTabularDiffs(sum: ActionSummary, txt: string, ag?: ActionGroupWithState) { public renderTabularDiffs(sum: ActionSummary, txt: string, ag?: ActionGroupWithState): HTMLElement {
const act = asTabularDiffs(sum); const act = asTabularDiffs(sum);
const editDom = dom('div', const editDom = dom('div',
this._renderTableSchemaChanges(sum, ag), this._renderTableSchemaChanges(sum, ag),
this._renderColumnSchemaChanges(sum, ag), this._renderColumnSchemaChanges(sum, ag),
map(act, (tdiff, table) => { Object.entries(act).map(([table, tdiff]: [string, TabularDiff]) => {
if (tdiff.cells.length === 0) { return dom('div'); } if (tdiff.cells.length === 0) { return dom('div'); }
return dom('table.action_log_table', return dom('table.action_log_table',
koDom.show(() => this._showForTable(table, ag)), koDom.show(() => this._showForTable(table, ag)),
@ -238,7 +237,7 @@ export class ActionLog extends dispose.Disposable implements IDomComponent {
'Loading...'), 'Loading...'),
koDom.foreach(this._displayStack, (ag: ActionGroupWithState) => { koDom.foreach(this._displayStack, (ag: ActionGroupWithState) => {
const timestamp = ag.time ? timeFormat("D T", new Date(ag.time)) : ""; const timestamp = ag.time ? timeFormat("D T", new Date(ag.time)) : "";
let desc = ag.desc || ""; let desc: DomContents = ag.desc || "";
if (ag.actionSummary) { if (ag.actionSummary) {
desc = this.renderTabularDiffs(ag.actionSummary, desc, ag); desc = this.renderTabularDiffs(ag.actionSummary, desc, ag);
} }

View File

@ -3,7 +3,6 @@
const _ = require('underscore'); const _ = require('underscore');
const ko = require('knockout'); const ko = require('knockout');
const moment = require('moment-timezone'); const moment = require('moment-timezone');
const {getSelectionDesc} = require('app/common/DocActions');
const {nativeCompare, roundDownToMultiple, waitObs} = require('app/common/gutil'); const {nativeCompare, roundDownToMultiple, waitObs} = require('app/common/gutil');
const gutil = require('app/common/gutil'); const gutil = require('app/common/gutil');
const MANUALSORT = require('app/common/gristTypes').MANUALSORT; const MANUALSORT = require('app/common/gristTypes').MANUALSORT;
@ -646,20 +645,7 @@ BaseView.prototype.sendPasteActions = function(cutCallback, actions) {
// If the cut occurs on an edit restricted cell, there may be no cut action. // If the cut occurs on an edit restricted cell, there may be no cut action.
if (cutAction) { actions.unshift(cutAction); } if (cutAction) { actions.unshift(cutAction); }
} }
return this.gristDoc.docData.sendActions(actions, return this.gristDoc.docData.sendActions(actions);
this._getPasteDesc(actions[actions.length - 1], cutAction));
};
/**
* Returns a string which describes a cut/copy action.
*/
BaseView.prototype._getPasteDesc = function(pasteAction, optCutAction) {
if (optCutAction) {
return `Moved ${getSelectionDesc(optCutAction, true)} to ` +
`${getSelectionDesc(pasteAction, true)}.`;
} else {
return `Pasted data to ${getSelectionDesc(pasteAction, true)}.`;
}
}; };
BaseView.prototype.buildDom = function() { BaseView.prototype.buildDom = function() {

View File

@ -8,7 +8,6 @@ import {safeJsonParse} from 'app/common/gutil';
import type {TableData} from 'app/common/TableData'; import type {TableData} from 'app/common/TableData';
import {tsvEncode} from 'app/common/tsvFormat'; import {tsvEncode} from 'app/common/tsvFormat';
import {dom} from 'grainjs'; import {dom} from 'grainjs';
import map = require('lodash/map');
import zipObject = require('lodash/zipObject'); import zipObject = require('lodash/zipObject');
const G = getBrowserGlobals('document', 'DOMParser'); const G = getBrowserGlobals('document', 'DOMParser');
@ -134,8 +133,11 @@ export function parsePasteHtml(data: string): RichPasteObject[][] {
} }
// Helper function to add css style properties to an html tag // Helper function to add css style properties to an html tag
function _styleAttr(style: object) { function _styleAttr(style: object|undefined) {
return map(style, (value, prop) => `${prop}: ${value};`).join(' '); if (typeof style !== 'object') {
return '';
}
return Object.entries(style).map(([prop, value]) => `${prop}: ${value};`).join(' ');
} }
/** /**

View File

@ -14,8 +14,6 @@ export interface AllCellVersions {
} }
export type CellVersions = Partial<AllCellVersions>; export type CellVersions = Partial<AllCellVersions>;
import map = require('lodash/map');
export type AddRecord = ['AddRecord', string, number, ColValues]; export type AddRecord = ['AddRecord', string, number, ColValues];
export type BulkAddRecord = ['BulkAddRecord', string, number[], BulkColValues]; export type BulkAddRecord = ['BulkAddRecord', string, number[], BulkColValues];
export type RemoveRecord = ['RemoveRecord', string, number]; export type RemoveRecord = ['RemoveRecord', string, number];
@ -150,21 +148,6 @@ export type UserAction = Array<string|number|object|boolean|null|undefined>;
// Actions that trigger formula calculations in the data engine // Actions that trigger formula calculations in the data engine
export const CALCULATING_USER_ACTIONS = new Set(['Calculate', 'UpdateCurrentTime', 'RespondToRequests']); export const CALCULATING_USER_ACTIONS = new Set(['Calculate', 'UpdateCurrentTime', 'RespondToRequests']);
/**
* Gives a description for an action which involves setting values to a selection.
* @param {Array} action - The (Bulk)AddRecord/(Bulk)UpdateRecord action to describe.
* @param {Boolean} optExcludeVals - Indicates whether the values should be excluded from
* the description.
*/
export function getSelectionDesc(action: UserAction, optExcludeVals: boolean): string {
const table = action[1];
const rows = action[2];
const colValues: number[] = action[3] as any; // TODO: better typing - but code may evaporate
const columns = map(colValues, (values, col) => optExcludeVals ? col : `${col}: ${values}`);
const s = typeof rows === 'object' ? 's' : '';
return `table ${table}, row${s} ${rows}; ${columns.join(", ")}`;
}
export function getNumRows(action: DocAction): number { export function getNumRows(action: DocAction): number {
return !isDataAction(action) ? 0 return !isDataAction(action) ? 0
: Array.isArray(action[2]) ? action[2].length : Array.isArray(action[2]) ? action[2].length

View File

@ -185,6 +185,8 @@ export const ThemeColors = t.iface([], {
"left-panel-page-options-selected-hover-bg": "string", "left-panel-page-options-selected-hover-bg": "string",
"left-panel-page-initials-fg": "string", "left-panel-page-initials-fg": "string",
"left-panel-page-initials-bg": "string", "left-panel-page-initials-bg": "string",
"left-panel-page-emoji-fg": "string",
"left-panel-page-emoji-outline": "string",
"right-panel-tab-fg": "string", "right-panel-tab-fg": "string",
"right-panel-tab-bg": "string", "right-panel-tab-bg": "string",
"right-panel-tab-icon": "string", "right-panel-tab-icon": "string",

View File

@ -16,9 +16,7 @@
import {LocalActionBundle} from 'app/common/ActionBundle'; import {LocalActionBundle} from 'app/common/ActionBundle';
import {ActionGroup, MinimalActionGroup} from 'app/common/ActionGroup'; import {ActionGroup, MinimalActionGroup} from 'app/common/ActionGroup';
import {createEmptyActionSummary} from 'app/common/ActionSummary'; import {createEmptyActionSummary} from 'app/common/ActionSummary';
import {getSelectionDesc, UserAction} from 'app/common/DocActions';
import {DocState} from 'app/common/UserAPI'; import {DocState} from 'app/common/UserAPI';
import toPairs = require('lodash/toPairs');
import {summarizeAction} from 'app/common/ActionSummarizer'; import {summarizeAction} from 'app/common/ActionSummarizer';
export interface ActionGroupOptions { export interface ActionGroupOptions {
@ -163,81 +161,6 @@ export abstract class ActionHistory {
} }
/**
* Old helper to display the actionGroup in a human-readable way. Being maintained
* to avoid having to change too much at once.
*/
export function humanDescription(actions: UserAction[]): string {
const action = actions[0];
if (!action) { return ""; }
let output = '';
// Common names for various action parameters
const name = action[0];
const table = action[1];
const rows = action[2];
const colId = action[2];
const columns: any = action[3]; // TODO - better typing - but code may evaporate
switch (name) {
case 'UpdateRecord':
case 'BulkUpdateRecord':
case 'AddRecord':
case 'BulkAddRecord':
output = name + ' ' + getSelectionDesc(action, columns);
break;
case 'ApplyUndoActions':
// Currently cannot display information about what action was undone, as the action comes
// with the description of the "undo" message, which might be very different
// Also, cannot currently properly log redos as they are not distinguished from others in any way
// TODO: make an ApplyRedoActions type for redoing actions
output = 'Undo Previous Action';
break;
case 'InitNewDoc':
output = 'Initialized new Document';
break;
case 'AddColumn':
output = 'Added column ' + colId + ' to ' + table;
break;
case 'RemoveColumn':
output = 'Removed column ' + colId + ' from ' + table;
break;
case 'RemoveRecord':
case 'BulkRemoveRecord':
output = 'Removed record(s) ' + rows + ' from ' + table;
break;
case 'EvalCode':
output = 'Evaluated Code ' + action[1];
break;
case 'AddTable':
output = 'Added table ' + table;
break;
case 'RemoveTable':
output = 'Removed table ' + table;
break;
case 'ModifyColumn':
// TODO: The Action Log currently only logs user actions,
// But ModifyColumn/Rename Column are almost always triggered from the client
// through a meta-table UpdateRecord.
// so, this is a case where making use of explicit sandbox engine 'looged' actions
// may be useful
output = 'Modify column ' + colId + ", ";
for (const [col, val] of toPairs(columns)) {
output += col + ": " + val + ", ";
}
output += ' in table ' + table;
break;
case 'RenameColumn': {
const newColId = action[3];
output = 'Renamed Column ' + colId + ' to ' + newColId + ' in ' + table;
break;
}
default:
output = name + ' [No Description]';
}
// A period for good grammar
output += '.';
return output;
}
/** /**
* Convert an ActionBundle into an ActionGroup. ActionGroups are the representation of * Convert an ActionBundle into an ActionGroup. ActionGroups are the representation of
* actions on the client. * actions on the client.
@ -260,7 +183,9 @@ export function asActionGroup(history: ActionHistory,
return { return {
actionNum: act.actionNum, actionNum: act.actionNum,
actionHash: act.actionHash || "", actionHash: act.actionHash || "",
desc: info.desc || humanDescription(act.userActions), // Desc is a human-readable description of the user action set in a few places by client-side
// code, but is mostly (or maybe completely) unused.
desc: info.desc,
actionSummary: summarize ? summarizeAction(act) : createEmptyActionSummary(), actionSummary: summarize ? summarizeAction(act) : createEmptyActionSummary(),
fromSelf, fromSelf,
linkId: info.linkId, linkId: info.linkId,