From f0ac47d34c6f8a264d9f496a7b656deef76480f1 Mon Sep 17 00:00:00 2001
From: Anton Kulyk <kuliks.anton@gmail.com>
Date: Wed, 14 Jul 2021 08:53:34 +0300
Subject: [PATCH] Fix native question looses visualisation settings if query
 fails (#17009)

* Add test IDs for visualization buttons

* Add `runNativeQuery` helper to e2e-misc-helpers

* Add repro for #16914

* Don't sync native query columns if it failed

* Fix `table.columns: []` treated as "all hidden"
---
 frontend/src/metabase-lib/lib/Question.js     |  3 +-
 .../components/view/ViewFooter.jsx            | 18 +++++++-
 .../visualizations/visualizations/Table.jsx   | 31 +++++++------
 .../e2e/helpers/e2e-misc-helpers.js           | 11 +++++
 .../native/reproductions/16914.cy.spec.js     | 44 +++++++++++++++++++
 5 files changed, 91 insertions(+), 16 deletions(-)
 create mode 100644 frontend/test/metabase/scenarios/native/reproductions/16914.cy.spec.js

diff --git a/frontend/src/metabase-lib/lib/Question.js b/frontend/src/metabase-lib/lib/Question.js
index d7ec233f69c..744da5b07ea 100644
--- a/frontend/src/metabase-lib/lib/Question.js
+++ b/frontend/src/metabase-lib/lib/Question.js
@@ -645,7 +645,8 @@ export default class Question {
 
   syncColumnsAndSettings(previous, queryResults) {
     const query = this.query();
-    if (query instanceof NativeQuery && queryResults) {
+    const isQueryResultValid = queryResults && !queryResults.error;
+    if (query instanceof NativeQuery && isQueryResultValid) {
       return this._syncNativeQuerySettings(queryResults);
     }
     const previousQuery = previous && previous.query();
diff --git a/frontend/src/metabase/query_builder/components/view/ViewFooter.jsx b/frontend/src/metabase/query_builder/components/view/ViewFooter.jsx
index d46f09d434b..23162b04221 100644
--- a/frontend/src/metabase/query_builder/components/view/ViewFooter.jsx
+++ b/frontend/src/metabase/query_builder/components/view/ViewFooter.jsx
@@ -191,14 +191,28 @@ const VizTypeButton = ({ question, result, ...props }) => {
   const icon = visualization && visualization.iconName;
 
   return (
-    <ViewButton medium p={[2, 1]} icon={icon} labelBreakpoint="sm" {...props}>
+    <ViewButton
+      medium
+      p={[2, 1]}
+      icon={icon}
+      labelBreakpoint="sm"
+      data-testid="viz-type-button"
+      {...props}
+    >
       {t`Visualization`}
     </ViewButton>
   );
 };
 
 const VizSettingsButton = ({ ...props }) => (
-  <ViewButton medium p={[2, 1]} icon="gear" labelBreakpoint="sm" {...props}>
+  <ViewButton
+    medium
+    p={[2, 1]}
+    icon="gear"
+    labelBreakpoint="sm"
+    data-testid="viz-settings-button"
+    {...props}
+  >
     {t`Settings`}
   </ViewButton>
 );
diff --git a/frontend/src/metabase/visualizations/visualizations/Table.jsx b/frontend/src/metabase/visualizations/visualizations/Table.jsx
index db1d8a15039..bb6bb400eba 100644
--- a/frontend/src/metabase/visualizations/visualizations/Table.jsx
+++ b/frontend/src/metabase/visualizations/visualizations/Table.jsx
@@ -160,6 +160,11 @@ export default class Table extends Component {
       widget: ChartSettingOrderedColumns,
       getHidden: (series, vizSettings) => vizSettings["table.pivot"],
       isValid: ([{ card, data }]) =>
+        // If "table.columns" happened to be an empty array,
+        // it will be treated as "all columns are hidden",
+        // This check ensures it's not empty,
+        // otherwise it will be overwritten by `getDefault` below
+        card.visualization_settings["table.columns"].length !== 0 &&
         _.all(
           card.visualization_settings["table.columns"],
           columnSetting =>
@@ -392,15 +397,15 @@ export default class Table extends Component {
     const { data } = this.state;
     const sort = getIn(card, ["dataset_query", "query", "order-by"]) || null;
     const isPivoted = settings["table.pivot"];
-    const isColumnsDisabled =
-      (settings["table.columns"] || []).filter(f => f.enabled).length < 1;
+    const columnSettings = settings["table.columns"] || [];
+    const areAllColumnsHidden = !columnSettings.some(f => f.enabled);
     const TableComponent = isDashboard ? TableSimple : TableInteractive;
 
     if (!data) {
       return null;
     }
 
-    if (isColumnsDisabled) {
+    if (areAllColumnsHidden) {
       return (
         <div
           className={cx(
@@ -420,16 +425,16 @@ export default class Table extends Component {
           <span className="h4 text-bold">{t`Every field is hidden right now`}</span>
         </div>
       );
-    } else {
-      return (
-        <TableComponent
-          {...this.props}
-          data={data}
-          isPivoted={isPivoted}
-          sort={sort}
-          getColumnTitle={this.getColumnTitle}
-        />
-      );
     }
+
+    return (
+      <TableComponent
+        {...this.props}
+        data={data}
+        isPivoted={isPivoted}
+        sort={sort}
+        getColumnTitle={this.getColumnTitle}
+      />
+    );
   }
 }
diff --git a/frontend/test/__support__/e2e/helpers/e2e-misc-helpers.js b/frontend/test/__support__/e2e/helpers/e2e-misc-helpers.js
index 2b941d96b11..78252c323e5 100644
--- a/frontend/test/__support__/e2e/helpers/e2e-misc-helpers.js
+++ b/frontend/test/__support__/e2e/helpers/e2e-misc-helpers.js
@@ -28,3 +28,14 @@ export function openNativeEditor(alias = "editor") {
     .as(alias)
     .should("be.visible");
 }
+
+/**
+ * Executes native query and waits for the results to load.
+ * Makes sure that the question is not "dirty" after the query successfully ran.
+ * @param {string} [xhrAlias ="dataset"]
+ */
+export function runNativeQuery(xhrAlias = "dataset") {
+  cy.get(".NativeQueryEditor .Icon-play").click();
+  cy.wait("@" + xhrAlias);
+  cy.icon("play").should("not.exist");
+}
diff --git a/frontend/test/metabase/scenarios/native/reproductions/16914.cy.spec.js b/frontend/test/metabase/scenarios/native/reproductions/16914.cy.spec.js
new file mode 100644
index 00000000000..cdedddc2e33
--- /dev/null
+++ b/frontend/test/metabase/scenarios/native/reproductions/16914.cy.spec.js
@@ -0,0 +1,44 @@
+import {
+  restore,
+  openNativeEditor,
+  runNativeQuery,
+} from "__support__/e2e/cypress";
+
+describe("issue 16914", () => {
+  beforeEach(() => {
+    restore();
+    cy.intercept("POST", "api/dataset").as("dataset");
+    cy.signInAsAdmin();
+  });
+
+  it("should recover visualization settings after a failed query (metabase#16914)", () => {
+    const FAILING_PIECE = " foo";
+    const highlightSelectedText = "{shift}{leftarrow}".repeat(
+      FAILING_PIECE.length,
+    );
+
+    openNativeEditor().type("SELECT 'a' as hidden, 'b' as visible");
+    runNativeQuery();
+
+    cy.findByTestId("viz-settings-button").click();
+    cy.findByTestId("sidebar-left")
+      .contains(/hidden/i)
+      .siblings(".Icon-close")
+      .click();
+    cy.button("Done").click();
+
+    cy.get("@editor").type(FAILING_PIECE);
+    runNativeQuery();
+
+    cy.get("@editor").type(
+      "{movetoend}" + highlightSelectedText + "{backspace}",
+    );
+    runNativeQuery();
+
+    cy.get(".Visualization").within(() => {
+      cy.findByText("Every field is hidden right now").should("not.exist");
+      cy.findByText("VISIBLE");
+      cy.findByText("HIDDEN").should("not.exist");
+    });
+  });
+});
-- 
GitLab