From 79965714c478d75ffef2cc43343f523434f0534b Mon Sep 17 00:00:00 2001 From: dpsutton <dan@dpsutton.com> Date: Wed, 15 Jun 2022 10:35:44 -0500 Subject: [PATCH] Ensure :error is present for frontend (#23346) * Ensure :error is present for frontend The frontend checks for a string at "error" to determine if there's an error or not. Any error that lacks a message (`(.getMessage e) = nil`) will report as a successful query with no results. This was hard to reproduce at first because java 14+ added a nice error message to NPEs in [JEP-358](https://openjdk.org/jeps/358). ```java ;; java 11 jshell> HashSet x = null; x ==> null jshell> x.contains(3); | Exception java.lang.NullPointerException | at (#2:1) jshell> ;; java 17 jshell> HashSet x = null; x ==> null jshell> x.contains(3); | Exception java.lang.NullPointerException: Cannot invoke "java.util.HashSet.contains(Object)" because "REPL.$JShell$11.x" is null | at (#4:1) ``` ```clojure ;; java 17 catch-exceptions=> (let [x nil] (x 3)) Execution error (NullPointerException) at metabase.query-processor.middleware.catch-exceptions/eval130385 (REPL:41). Cannot invoke "clojure.lang.IFn.invoke(Object)" because "x" is null ;; java 11 catch-exceptions=> (let [x nil] (x 3)) Execution error (NullPointerException) at metabase.driver.sql.parameters.substitution/eval118508 (REPL:17). null ``` Here are the responses to the FE edited to the relevant bits: ```javascript // objects are edited for clarity/brevity: // from java-11 { "error_type": "qp", "status": "failed", "class": "class java.lang.NullPointerException", "stacktrace": [...], "error": null, // <---- the FE ignores all of the other data and only sees this } // vs java-17 { "error_type": "qp", "status": "failed", "class": "class java.lang.NullPointerException", "stacktrace": [...], "error": "Cannot invoke \"clojure.lang.IFn.invoke(Object)\" because \"to_period\" is null",, } ``` Also, note that we were still logging > ERROR middleware.catch-exceptions :: Error processing query: null because we were looking for `(:error format-exception)` instead of `(:error formatted-exception)`. `format-exception` is a multimethod and lookup on non-associative things will happily return nil. * Tests --- .../middleware/catch_exceptions.clj | 10 ++- .../middleware/catch_exceptions_test.clj | 73 +++++++++++-------- 2 files changed, 51 insertions(+), 32 deletions(-) diff --git a/src/metabase/query_processor/middleware/catch_exceptions.clj b/src/metabase/query_processor/middleware/catch_exceptions.clj index dd28df493fb..06e67124caa 100644 --- a/src/metabase/query_processor/middleware/catch_exceptions.clj +++ b/src/metabase/query_processor/middleware/catch_exceptions.clj @@ -154,9 +154,15 @@ (letfn [(raisef* [e context] ;; format the Exception and return it (let [formatted-exception (format-exception* query e @extra-info)] - (log/error (str (trs "Error processing query: {0}" (:error format-exception)) + (log/error (str (trs "Error processing query: {0}" + (or (:error formatted-exception) + ;; log in server locale, respond in user locale + (trs "Error running query"))) "\n" (u/pprint-to-str formatted-exception))) - (qp.context/resultf formatted-exception context)))] + ;; ensure always a message on the error otherwise FE thinks query was successful. (#23258, #23281) + (qp.context/resultf (update formatted-exception + :error (fnil identity (trs "Error running query"))) + context)))] (try (qp query rff (assoc context :raisef raisef*)) (catch Throwable e diff --git a/test/metabase/query_processor/middleware/catch_exceptions_test.clj b/test/metabase/query_processor/middleware/catch_exceptions_test.clj index f1145a999ac..f4cc08fd5bb 100644 --- a/test/metabase/query_processor/middleware/catch_exceptions_test.clj +++ b/test/metabase/query_processor/middleware/catch_exceptions_test.clj @@ -104,37 +104,50 @@ :metadata (update :stacktrace boolean))))))) -(deftest include-query-execution-info-test - (testing "Should include info from QueryExecution if added to the thrown/raised Exception" +(deftest catch-exceptions-test + (testing "include-query-execution-info-test" + (testing "Should include info from QueryExecution if added to the thrown/raised Exception" + (tu.log/suppress-output + (is (= {:status :failed + :class java.lang.Exception + :error "Something went wrong" + :stacktrace true + :card_id 300 + :json_query {} + :row_count 0 + :data {:cols []} + :a 100 + :b 200} + (-> (mt/test-qp-middleware catch-exceptions/catch-exceptions + {} {} [] + {:runf (fn [_ _ context] + (qp.context/raisef (ex-info "Something went wrong." + {:query-execution {:a 100 + :b 200 + :card_id 300 + ;; these keys should all get removed + :result_rows 400 + :hash 500 + :executor_id 500 + :dashboard_id 700 + :pulse_id 800 + :native 900}} + (Exception. "Something went wrong")) + context))}) + :metadata + (update :stacktrace boolean))))))) + (testing "Should always include :error (#23258, #23281)" (tu.log/suppress-output - (is (= {:status :failed - :class java.lang.Exception - :error "Something went wrong" - :stacktrace true - :card_id 300 - :json_query {} - :row_count 0 - :data {:cols []} - :a 100 - :b 200} - (-> (mt/test-qp-middleware catch-exceptions/catch-exceptions - {} {} [] - {:runf (fn [_ _ context] - (qp.context/raisef (ex-info "Something went wrong." - {:query-execution {:a 100 - :b 200 - :card_id 300 - ;; these keys should all get removed - :result_rows 400 - :hash 500 - :executor_id 500 - :dashboard_id 700 - :pulse_id 800 - :native 900}} - (Exception. "Something went wrong")) - context))}) - :metadata - (update :stacktrace boolean))))))) + (testing "Uses error message if present" + (is (= "Something went wrong" + (-> (fn [] (throw (Exception. "Something went wrong"))) + (catch-exceptions) + :error)))) + (testing "Has a default if no message on error" + (is (= "Error running query" + (-> (fn [] (throw (ex-info nil {}))) + (catch-exceptions) + :error))))))) (deftest permissions-test (data/with-temp-copy-of-db -- GitLab