diff --git a/src/metabase/async/streaming_response.clj b/src/metabase/async/streaming_response.clj index 76050dd399341b7b58faed054a367a00455f77e7..015d682bfc0c1b075f53fb3f93e4ac4c40372368 100644 --- a/src/metabase/async/streaming_response.clj +++ b/src/metabase/async/streaming_response.clj @@ -2,6 +2,7 @@ (:require [cheshire.core :as json] [clojure.core.async :as a] + [clojure.walk :as walk] [compojure.response] [metabase.async.streaming-response.thread-pool :as thread-pool] [metabase.async.util :as async.u] @@ -44,21 +45,30 @@ (defn write-error! "Write an error to the output stream, formatting it nicely. Closes output stream afterwards." - [^OutputStream os obj] + [^OutputStream os obj export-format] (cond (some #(instance? % obj) [InterruptedException EofException]) (log/trace "Error is an InterruptedException or EofException, not writing to output stream") (instance? Throwable obj) - (recur os (format-exception obj)) + (recur os (format-exception obj) export-format) :else (with-open [os os] (log/trace (u/pprint-to-str (list 'write-error! obj))) (try - (with-open [writer (BufferedWriter. (OutputStreamWriter. os StandardCharsets/UTF_8))] - (json/generate-stream obj writer)) + (let [obj (-> (if (not= :api export-format) + (walk/prewalk + (fn [x] + (if (map? x) + (apply dissoc x [:json_query :preprocessed]) + x)) + obj) + obj) + (dissoc :export-format))] + (with-open [writer (BufferedWriter. (OutputStreamWriter. os StandardCharsets/UTF_8))] + (json/generate-stream obj writer))) (catch EofException _) (catch Throwable e (log/error e "Error writing error to output stream" obj)))))) @@ -74,7 +84,7 @@ nil) (catch Throwable e (log/error e "Caught unexpected Exception in streaming response body") - (write-error! os e) + (write-error! os e nil) nil))) (defn- do-f-async diff --git a/src/metabase/query_processor/streaming.clj b/src/metabase/query_processor/streaming.clj index 7af805e949042bde8b13389b29911bdd74ea77a0..5da0b115f5492e873391e11c3bb3a62345785df4 100644 --- a/src/metabase/query_processor/streaming.clj +++ b/src/metabase/query_processor/streaming.clj @@ -181,7 +181,7 @@ (assert (not (instance? ManyToManyChannel result)) "QP should not return a core.async channel.") (when (or (instance? Throwable result) (= (:status result) :failed)) - (streaming-response/write-error! os result))))))) + (streaming-response/write-error! os result export-format))))))) (defmacro streaming-response "Return results of processing a query as a streaming response. This response implements the appropriate Ring/Compojure diff --git a/test/metabase/api/downloads_exports_test.clj b/test/metabase/api/downloads_exports_test.clj index 28feac1ddb698ff1dcd5b46b0e92b7c956f332e8..6529d9e3eb1d4dff0ad6c3665cb8d9bc227ca024 100644 --- a/test/metabase/api/downloads_exports_test.clj +++ b/test/metabase/api/downloads_exports_test.clj @@ -13,8 +13,10 @@ [clojure.data.csv :as csv] [clojure.java.io :as io] [clojure.set :as set] + [clojure.string :as str] [clojure.test :refer :all] [dk.ative.docjure.spreadsheet :as spreadsheet] + [metabase.formatter :as formatter] [metabase.public-settings :as public-settings] [metabase.pulse :as pulse] [metabase.pulse.test-util :as pulse.test-util] @@ -742,3 +744,42 @@ ;; the [$$] part will appear as $ when you open the Excel file in a spreadsheet app (is (= [["Discount"] ["[$$]6.42"]] (-> (card-download card :xlsx true))))))))) + +(deftest clean-errors-test + (testing "Queries that error should not include visualization settings (metabase-private #233)" + (with-redefs [formatter/number-formatter (fn [& _args] (fn [_] (throw (Exception. "Test Exception"))))] + (mt/with-temp [:model/Card {card-id :id} {:display :table + :type :model + :dataset_query {:database (mt/id) + :type :query + :query {:source-table (mt/id :orders) + :filter [:not-null [:field (mt/id :orders :discount) {:base-type :type/Float}]] + :limit 1}} + :visualization_settings {:table.columns + [{:name "ID" :enabled false} + {:name "USER_ID" :enabled false} + {:name "PRODUCT_ID" :enabled false} + {:name "SUBTOTAL" :enabled false} + {:name "TAX" :enabled false} + {:name "TOTAL" :enabled false} + {:name "DISCOUNT" :enabled true} + {:name "CREATED_AT" :enabled false} + {:name "QUANTITY" :enabled false}] + :table.cell_column "SUBTOTAL" + :column_settings {(format "[\"ref\",[\"field\",%s,null]]" (mt/id :orders :discount)) + {:currency_in_header false}}}}] + (let [illegal-strings ["visualization-settings" ":viz-settings" "visualization_settings"]] + (doseq [export-format ["csv" "json" #_"xlsx"]] + ;; for now, don't try to read xlsx back in, it will not be correct since we end up writing + ;; a json blob to the output stream, it creates an invalid xlsx anyway. + ;; This is not new behaviour, we'll just fix it when a better solution to 'errors in downloaded files' comes along + (let [results (mt/user-http-request :rasta :post 200 (format "card/%d/query/%s?format_rows=true" card-id export-format)) + results-string (if (= "xlsx" export-format) + (read-xlsx results) + (str results))] + (testing (format "Testing export format: %s" export-format) + (doseq [illegal illegal-strings] + (is (false? (str/blank? results-string))) + (is (true? (str/includes? results-string "Test Exception"))) + (testing (format "String \"%s\" is not in the error message." illegal) + (is (false? (str/includes? results-string illegal))))))))))))) diff --git a/test/metabase/query_processor/streaming/csv_test.clj b/test/metabase/query_processor/streaming/csv_test.clj index aca1ee757519b6dd9bc9b4a61b5d0c3e1e642735..155dd29ffaca9eede6a8476db110693fd4d7fba6 100644 --- a/test/metabase/query_processor/streaming/csv_test.clj +++ b/test/metabase/query_processor/streaming/csv_test.clj @@ -2,12 +2,16 @@ (:require [cheshire.core :as json] [clojure.data.csv :as csv] + [clojure.string :as str] [clojure.test :refer :all] [metabase.driver :as driver] + [metabase.models.data-permissions :as data-perms] + [metabase.models.permissions-group :as perms-group] [metabase.query-processor :as qp] [metabase.query-processor.streaming.interface :as qp.si] [metabase.test :as mt] - [metabase.test.data.dataset-definitions :as defs]) + [metabase.test.data.dataset-definitions :as defs] + [metabase.util :as u]) (:import (java.io BufferedOutputStream ByteArrayOutputStream))) @@ -46,6 +50,23 @@ (json/generate-string (mt/mbql-query checkins {:order-by [[:asc $id]], :limit 5})))] (take 5 (parse-and-sort-csv result))))))) +(deftest errors-not-include-visualization-settings + (testing "Queries that error should not include visualization settings" + (mt/with-temp [:model/Card {card-id :id} {:dataset_query (mt/mbql-query orders + {:order-by [[:asc $id]], :limit 5}) + :visualization_settings {:column_settings {} + :notvisiblekey :notvisiblevalue}}] + (mt/with-no-data-perms-for-all-users! + (data-perms/set-database-permission! (perms-group/all-users) + (u/the-id (mt/db)) + :perms/create-queries :query-builder) + (let [results (mt/user-http-request :rasta :post 200 (format "card/%d/query/csv" card-id)) + results-string (str results) + illegal-strings ["notvisiblekey" "notvisiblevalue" "column_settings" + "visualization-settings" "viz-settings"]] + (doseq [illegal illegal-strings] + (is (false? (str/includes? results-string illegal))))))))) + (deftest check-an-empty-date-column (testing "NULL values should be written correctly" (mt/dataset defs/test-data-null-date