From ad04744b4ac62dee78ef4d12f4389ca7d4b81d72 Mon Sep 17 00:00:00 2001 From: George Gevoian Date: Thu, 28 Apr 2022 08:43:31 -0700 Subject: [PATCH] (core) Fix import bug when skipping non-text columns Summary: Skipping columns during incremental imports wasn't working for certain column types, such as numeric columns. The column's default value was being used instead (e.g. 0), overwriting values in the destination table. Test Plan: Browser tests. Reviewers: jarek Reviewed By: jarek Subscribers: alexmojaki Differential Revision: https://phab.getgrist.com/D3402 --- app/client/components/Importer.ts | 95 +++++++++++++++++------------ app/client/widgets/FormulaEditor.ts | 2 +- app/server/lib/ActiveDocImport.ts | 29 +++++++-- 3 files changed, 83 insertions(+), 43 deletions(-) diff --git a/app/client/components/Importer.ts b/app/client/components/Importer.ts index 987a6f78..c66e4e8f 100644 --- a/app/client/components/Importer.ts +++ b/app/client/components/Importer.ts @@ -193,17 +193,22 @@ export class Importer extends DisposableWithEvents { return new Map(fields.map(f => [use(use(f.column).colId), use(use(f.column).label)])); }); - // List of destination fields that aren't mapped to a source column. - private _unmatchedFields = Computed.create(this, this._sourceInfoSelected, (use, info) => { + // Transform section columns of the selected source. + private _transformSectionCols = Computed.create(this, this._sourceInfoSelected, (use, info) => { if (!info) { return null; } const transformSection = use(info.transformSection); if (!transformSection || use(transformSection._isDeleted)) { return null; } const fields = use(use(transformSection.viewFields).getObservable()); - return fields - .filter(f => use(use(f.column).formula).trim() === '') - .map(f => f.column().label()); + return fields.map(f => use(f.column)); + }); + + // List of destination fields that aren't mapped to a source column. + private _unmatchedFields = Computed.create(this, this._transformSectionCols, (use, cols) => { + if (!cols) { return null; } + + return cols.filter(c => use(c.formula).trim() === '').map(c => c.label()); }); // null tells to use the built-in file picker. @@ -490,6 +495,20 @@ export class Importer extends DisposableWithEvents { * @param {SourceInfo} info The source to update the diff for. */ private async _updateImportDiff(info: SourceInfo) { + const {updateExistingRecords, mergeCols} = this._mergeOptions[info.hiddenTableId]!; + const isMerging = info.destTableId && updateExistingRecords.get() && mergeCols.get().length > 0; + if (!isMerging && this._gristDoc.comparison) { + // If we're not merging but diffing is enabled, disable it; since `comparison` isn't + // currently observable, we'll wrap the modification around the `_isLoadingDiff` + // flag, which will force the preview table to re-render with diffing disabled. + this._isLoadingDiff.set(true); + this._gristDoc.comparison = null; + this._isLoadingDiff.set(false); + } + + // If we're not merging, no diff is shown, so don't schedule an update for one. + if (!isMerging) { return; } + this._hasScheduledDiffUpdate = true; this._isLoadingDiff.set(true); await this._debouncedUpdateDiff(info); @@ -508,23 +527,17 @@ export class Importer extends DisposableWithEvents { // Reset the flag tracking scheduled updates since the debounced update has started. this._hasScheduledDiffUpdate = false; - const mergeOptions = this._mergeOptions[info.hiddenTableId]!; - if (!mergeOptions.updateExistingRecords.get() || mergeOptions.mergeCols.get().length === 0) { - // We can simply disable document comparison mode when merging isn't configured. - this._gristDoc.comparison = null; - } else { - // Request a diff of the current source and wait for a response. - const genImportDiffPromise = this._docComm.generateImportDiff(info.hiddenTableId, - this._createTransformRule(info), this._getMergeOptionsForSource(info)!); - this._lastGenImportDiffPromise = genImportDiffPromise; - const diff = await genImportDiffPromise; + // Request a diff of the current source and wait for a response. + const genImportDiffPromise = this._docComm.generateImportDiff(info.hiddenTableId, + this._createTransformRule(info), this._getMergeOptionsForSource(info)!); + this._lastGenImportDiffPromise = genImportDiffPromise; + const diff = await genImportDiffPromise; - // If the request is superseded by a newer request, or the Importer is disposed, do nothing. - if (this.isDisposed() || genImportDiffPromise !== this._lastGenImportDiffPromise) { return; } + // If the request is superseded by a newer request, or the Importer is disposed, do nothing. + if (this.isDisposed() || genImportDiffPromise !== this._lastGenImportDiffPromise) { return; } - // Otherwise, put the document in comparison mode with the diff data. - this._gristDoc.comparison = diff; - } + // Put the document in comparison mode with the diff data. + this._gristDoc.comparison = diff; // If more updates where scheduled since we started the update, leave the loading spinner up. if (!this._hasScheduledDiffUpdate) { @@ -595,7 +608,7 @@ export class Importer extends DisposableWithEvents { this._cancelPendingDiffRequests(); this._sourceInfoSelected.set(info); - await this._updateDiff(info); + await this._updateImportDiff(info); }), testId('importer-source'), ); @@ -675,14 +688,20 @@ export class Importer extends DisposableWithEvents { const sourceColIdsAndLabels = [...this._sourceColLabelsById.get()!.entries()]; return [ menuItem( - () => this._gristDoc.clearColumns([sourceColId]), + async () => { + await this._gristDoc.clearColumns([sourceColId]); + await this._updateImportDiff(info); + }, 'Skip', testId('importer-column-match-menu-item') ), menuDivider(), ...sourceColIdsAndLabels.map(([id, label]) => menuItem( - () => this._setColumnFormula(sourceColId, '$' + id), + async () => { + await this._setColumnFormula(sourceColId, '$' + id); + await this._updateImportDiff(info); + }, label, testId('importer-column-match-menu-item') ), @@ -699,7 +718,11 @@ export class Importer extends DisposableWithEvents { dom.domComputed(use => dom.create( this._buildColMappingFormula.bind(this), use(field.column), - (elem: Element) => this._activateFormulaEditor(elem, field), + (elem: Element) => this._activateFormulaEditor( + elem, + field, + () => this._updateImportDiff(info), + ), 'Skip' )), testId('importer-column-match-source-destination'), @@ -719,18 +742,6 @@ export class Importer extends DisposableWithEvents { } const gridView = this._createPreview(previewSection); - - // When changes are made to the preview table, update the import diff. - gridView.listenTo(gridView.sortedRows, 'rowNotify', async () => { - // If we aren't showing a diff, there is no need to do anything. - if (!info.destTableId || !updateExistingRecords.get() || mergeCols.get().length === 0) { - return; - } - - // Otherwise, update the diff and rebuild the preview table. - await this._updateImportDiff(info); - }); - return cssPreviewGrid( dom.maybe(use1 => SKIP_TABLE === use1(info.destTableId), () => cssOverlay(testId("importer-preview-overlay"))), @@ -782,13 +793,21 @@ export class Importer extends DisposableWithEvents { /** * Opens a formula editor for `field` over `refElem`. */ - private _activateFormulaEditor(refElem: Element, field: ViewFieldRec) { - // TODO: Set active section to hidden table section, so editor autocomplete is accurate. + private _activateFormulaEditor(refElem: Element, field: ViewFieldRec, onSave: () => Promise) { + const vsi = this._gristDoc.viewModel.activeSection().viewInstance(); + const editRow = vsi?.moveEditRowToCursor(); const editorHolder = openFormulaEditor({ gristDoc: this._gristDoc, field, refElem, + editRow, setupCleanup: this._setupFormulaEditorCleanup.bind(this), + onSave: async (column, formula) => { + if (formula === column.formula.peek()) { return; } + + await column.updateColValues({formula}); + await onSave(); + } }); this._formulaEditorHolder.autoDispose(editorHolder); } diff --git a/app/client/widgets/FormulaEditor.ts b/app/client/widgets/FormulaEditor.ts index 664245ee..98746155 100644 --- a/app/client/widgets/FormulaEditor.ts +++ b/app/client/widgets/FormulaEditor.ts @@ -249,7 +249,7 @@ function _isInIdentifier(line: string, column: number) { export function openFormulaEditor(options: { gristDoc: GristDoc, field: ViewFieldRec, - // Associated formula from a diffrent column (for example style rule). + // Associated formula from a different column (for example style rule). column?: ColumnRec, // Needed to get exception value, if any. editRow?: DataRowModel, diff --git a/app/server/lib/ActiveDocImport.ts b/app/server/lib/ActiveDocImport.ts index 0f39ee74..0b285ff9 100644 --- a/app/server/lib/ActiveDocImport.ts +++ b/app/server/lib/ActiveDocImport.ts @@ -137,8 +137,7 @@ export class ActiveDocImport { mergeCols = stripPrefixes(mergeCols); // Get column differences between `hiddenTableId` and `destTableId` for rows that exist in both tables. - const srcAndDestColIds: [string, string[]][] = - destCols.map(c => [c.colId!, [c.colId!.slice(IMPORT_TRANSFORM_COLUMN_PREFIX.length)]]); + const srcAndDestColIds: [string, string[]][] = destCols.map(c => [c.colId!, stripPrefixes([c.colId!])]); const srcToDestColIds = new Map(srcAndDestColIds); const comparisonResult = await this._getTableComparison(hiddenTableId, destTableId!, srcToDestColIds, mergeCols); @@ -153,6 +152,11 @@ export class ActiveDocImport { // Retrieve the function used to reconcile differences between source and destination. const merge = getMergeFunction(mergeStrategy); + // Destination columns with a blank formula (i.e. skipped columns). + const skippedColumnIds = new Set( + stripPrefixes(destCols.filter(c => c.formula.trim() === '').map(c => c.colId!)) + ); + const numResultRows = comparisonResult[hiddenTableId + '.id'].length; for (let i = 0; i < numResultRows; i++) { const srcRowId = comparisonResult[hiddenTableId + '.id'][i] as number; @@ -172,7 +176,12 @@ export class ActiveDocImport { // Exclude unchanged cell values from the comparison. if (srcVal === destVal) { continue; } - updatedRecords[srcColId][srcRowId] = [[destVal], [merge(srcVal, destVal)]]; + const shouldSkip = skippedColumnIds.has(matchingDestColId); + updatedRecords[srcColId][srcRowId] = [ + [destVal], + // For skipped columns, always use the destination value. + [shouldSkip ? destVal : merge(srcVal, destVal)] + ]; } } @@ -419,7 +428,9 @@ export class ActiveDocImport { const columnData: BulkColValues = {}; const srcColIds = srcCols.map(c => c.id as string); - const destCols = transformRule.destCols; + + // Only include destination columns that weren't skipped. + const destCols = transformRule.destCols.filter(c => c.formula.trim() !== ''); for (const destCol of destCols) { const formula = destCol.formula.trim(); if (!formula) { continue; } @@ -487,6 +498,16 @@ export class ActiveDocImport { const updatedRecords: BulkColValues = {}; const updatedRecordIds: number[] = []; + // Destination columns with a blank formula (i.e. skipped columns). + const skippedColumnIds = new Set( + stripPrefixes(destCols.filter(c => c.formula.trim() === '').map(c => c.colId!)) + ); + + // Remove all skipped columns from the map. + srcToDestColIds.forEach((destColIds, srcColId) => { + srcToDestColIds.set(srcColId, destColIds.filter(id => !skippedColumnIds.has(id))); + }); + const destColIds = flatten([...srcToDestColIds.values()]); for (const id of destColIds) { newRecords[id] = [];