(core) TypeTransform race condition fix

Summary:
TypeTransformation was flaky. Probably after upgrading AceEditor we introduced a race condition between updating the revised formula and doing the transformation. Now we explicitly make sure that the formula is updated.

I also fixed some other flaky tests.

Test Plan: Updated

Reviewers: paulfitz

Reviewed By: paulfitz

Subscribers: paulfitz

Differential Revision: https://phab.getgrist.com/D3984
This commit is contained in:
Jarosław Sadziński 2023-08-02 12:37:00 +02:00
parent 8110a26873
commit 4cfa033078
5 changed files with 43 additions and 15 deletions

View File

@ -4,6 +4,7 @@
* and TypeTransform. * and TypeTransform.
*/ */
import * as commands from 'app/client/components/commands'; import * as commands from 'app/client/components/commands';
import * as AceEditor from 'app/client/components/AceEditor';
import {GristDoc} from 'app/client/components/GristDoc'; import {GristDoc} from 'app/client/components/GristDoc';
import {ColumnRec} from 'app/client/models/entities/ColumnRec'; import {ColumnRec} from 'app/client/models/entities/ColumnRec';
import {ViewFieldRec} from 'app/client/models/entities/ViewFieldRec'; import {ViewFieldRec} from 'app/client/models/entities/ViewFieldRec';
@ -88,6 +89,13 @@ export class ColumnTransform extends Disposable {
* @param {String} optInit - Optional initial value for the editor. * @param {String} optInit - Optional initial value for the editor.
*/ */
protected buildEditorDom(optInit?: string) { protected buildEditorDom(optInit?: string) {
if (!this.editor) {
this.editor = this.autoDispose(AceEditor.create({
gristDoc: this.gristDoc,
observable: this.transformColumn.formula,
saveValueOnBlurEvent: false,
}));
}
return this.editor.buildDom((aceObj: any) => { return this.editor.buildDom((aceObj: any) => {
this.editor.adjustContentToWidth(); this.editor.adjustContentToWidth();
this.editor.attachSaveCommand(); this.editor.attachSaveCommand();
@ -238,6 +246,7 @@ export class ColumnTransform extends Disposable {
{...this.origWidgetOptions as object, ...this._fieldBuilder.options.peek()} : {...this.origWidgetOptions as object, ...this._fieldBuilder.options.peek()} :
this._fieldBuilder.options.peek(); this._fieldBuilder.options.peek();
return [ return [
...this.previewActions(),
[ [
'CopyFromColumn', 'CopyFromColumn',
this._tableData.tableId, this._tableData.tableId,
@ -268,4 +277,23 @@ export class ColumnTransform extends Disposable {
protected isFinalizing(): boolean { protected isFinalizing(): boolean {
return this._isFinalizing; return this._isFinalizing;
} }
protected preview() {
if (!this.editor) { return; }
return this.editor.writeObservable();
}
/**
* Generates final actions before executing the transform. Used only when the editor was created.
*/
protected previewActions(): UserAction[] {
if (!this.editor) { return []; }
const formula = this.editor.getValue();
const oldFormula = this.transformColumn.formula();
if (formula === oldFormula) { return []; }
if (!formula && !oldFormula) { return []; }
return [
['UpdateRecord', '_grist_Tables_column', this.transformColumn.getRowId(), {formula}]
];
}
} }

View File

@ -5,7 +5,6 @@
*/ */
// Client libraries // Client libraries
import * as AceEditor from 'app/client/components/AceEditor';
import {ColumnTransform} from 'app/client/components/ColumnTransform'; import {ColumnTransform} from 'app/client/components/ColumnTransform';
import {GristDoc} from 'app/client/components/GristDoc'; import {GristDoc} from 'app/client/components/GristDoc';
import {cssButtonRow} from 'app/client/ui/RightPanelStyles'; import {cssButtonRow} from 'app/client/ui/RightPanelStyles';
@ -26,10 +25,6 @@ export class FormulaTransform extends ColumnTransform {
* Build the transform menu for a formula transform * Build the transform menu for a formula transform
*/ */
public buildDom() { public buildDom() {
this.editor = this.autoDispose(AceEditor.create({
gristDoc: this.gristDoc,
observable: this.transformColumn.formula,
}));
return [ return [
dom('div.transform_menu', dom('div.transform_menu',
dom('div.transform_editor', dom('div.transform_editor',
@ -40,7 +35,7 @@ export class FormulaTransform extends ColumnTransform {
cssButtonRow( cssButtonRow(
basicButton(dom.on('click', () => this.cancel()), basicButton(dom.on('click', () => this.cancel()),
'Cancel', testId("formula-transform-cancel")), 'Cancel', testId("formula-transform-cancel")),
basicButton(dom.on('click', () => this.editor.writeObservable()), basicButton(dom.on('click', () => this.preview()),
'Preview', 'Preview',
dom.cls('disabled', this.formulaUpToDate), dom.cls('disabled', this.formulaUpToDate),
{ title: 'Update formula (Shift+Enter)' }, { title: 'Update formula (Shift+Enter)' },

View File

@ -5,7 +5,6 @@
* to be pre-entered for certain transforms (to Reference / Date) which the user can modify via dropdown menus. * to be pre-entered for certain transforms (to Reference / Date) which the user can modify via dropdown menus.
*/ */
import * as AceEditor from 'app/client/components/AceEditor';
import {ColumnTransform} from 'app/client/components/ColumnTransform'; import {ColumnTransform} from 'app/client/components/ColumnTransform';
import {GristDoc} from 'app/client/components/GristDoc'; import {GristDoc} from 'app/client/components/GristDoc';
import * as TypeConversion from 'app/client/components/TypeConversion'; import * as TypeConversion from 'app/client/components/TypeConversion';
@ -50,12 +49,7 @@ export class TypeTransform extends ColumnTransform {
public buildDom() { public buildDom() {
// An observable to disable all buttons before the dom get removed. // An observable to disable all buttons before the dom get removed.
const disableButtons = Observable.create(null, false); const disableButtons = Observable.create(null, false);
this._reviseTypeChange.set(false); this._reviseTypeChange.set(false);
this.editor = this.autoDispose(AceEditor.create({
gristDoc: this.gristDoc,
observable: this.transformColumn.formula,
}));
return dom('div', return dom('div',
testId('type-transform-top'), testId('type-transform-top'),
dom.maybe(this._transformWidget, transformWidget => transformWidget.buildTransformConfigDom()), dom.maybe(this._transformWidget, transformWidget => transformWidget.buildTransformConfigDom()),
@ -71,7 +65,7 @@ export class TypeTransform extends ColumnTransform {
), ),
dom.domComputed(this._reviseTypeChange, revising => { dom.domComputed(this._reviseTypeChange, revising => {
if (revising) { if (revising) {
return basicButton(dom.on('click', () => this.editor.writeObservable()), return basicButton(dom.on('click', () => this.preview()),
t('Preview'), testId("type-transform-update"), t('Preview'), testId("type-transform-update"),
dom.cls('disabled', (use) => use(disableButtons) || use(this.formulaUpToDate)), dom.cls('disabled', (use) => use(disableButtons) || use(this.formulaUpToDate)),
{ title: t('Update formula (Shift+Enter)') } { title: t('Update formula (Shift+Enter)') }

View File

@ -42,6 +42,7 @@ describe("GridOptions.ntest", function() {
await gu.supportOldTimeyTestCode(); await gu.supportOldTimeyTestCode();
await gu.useFixtureDoc(cleanup, "World-v10.grist", true); await gu.useFixtureDoc(cleanup, "World-v10.grist", true);
await $('.test-gristdoc').wait(); await $('.test-gristdoc').wait();
await gu.hideBanners();
}); });
beforeEach(async function() { beforeEach(async function() {
@ -109,6 +110,7 @@ describe("GridOptions.ntest", function() {
await driver.navigate().refresh(); await driver.navigate().refresh();
//await $.injectIntoPage(); //await $.injectIntoPage();
await gu.waitForDocToLoad(); await gu.waitForDocToLoad();
await gu.hideBanners();
await assertHVZ(0, true, true, true); // all on await assertHVZ(0, true, true, true); // all on
await assertHVZ(1, false, false, false); // all off await assertHVZ(1, false, false, false); // all off
await assertHVZ(2, false, true, true); // -h +v +z await assertHVZ(2, false, true, true); // -h +v +z

View File

@ -976,6 +976,14 @@ export async function confirm(save = true, remember = false) {
} }
} }
/** Hides all top banners by injecting css style */
export async function hideBanners() {
const style = `.test-banner-element { display: none !important; }`;
await driver.executeScript(`const style = document.createElement('style');
style.innerHTML = ${JSON.stringify(style)};
document.head.appendChild(style);`);
}
/** /**
* Returns the left-panel item for the given page, given by a full string name, or a RegExp. * Returns the left-panel item for the given page, given by a full string name, or a RegExp.
* You may simply click it to switch to that page. * You may simply click it to switch to that page.
@ -1582,10 +1590,11 @@ export async function sendKeys(...keys: string[]) {
} }
/** /**
* Clears active input by sending HOME + SHIFT END + DELETE. * Clears active input/textarea by sending CTRL HOME + CTRL + SHIFT END + DELETE.
*/ */
export async function clearInput() { export async function clearInput() {
return sendKeys(Key.HOME, Key.chord(Key.SHIFT, Key.END), Key.DELETE); const ctrl = await modKey();
return sendKeys(Key.chord(ctrl, Key.HOME), Key.chord(ctrl, Key.SHIFT, Key.END), Key.DELETE);
} }
/** /**