Skip to content
Snippets Groups Projects
Unverified Commit 25c9addf authored by Mark Bastian's avatar Mark Bastian Committed by GitHub
Browse files

Ensure Metadata is Propagated for Standalone Pulse Cards (#36357)

Pulse cards based on models with user-curated metadata were not showing proper formatting for percentages. This is because the singe `execute-card` function in `metabase.pulse.util` was not adding in the card `:result_metadata` when the card was a model.

The metadata is now properly added.

Additionally, `metabase.pulse.pulse-integration-test` has been refactored for reuse. The `with-metadata-data-cards` macro provides 3 examples of cards that should be tested (a base card, a model card, and a derived question) and a function that executeds the pulse and returns the resulting data in the last column (The one we are asserting on).

Note that as we develop the test scenario in this ns there may be further refactoring opportunities, such as returning the attachments from the pulse email as well.
parent a6cba791
No related branches found
No related tags found
No related merge requests found
......@@ -18,16 +18,21 @@
{:pre [(integer? pulse-creator-id)]}
(let [card-id (u/the-id card-or-id)]
(try
(when-let [{query :dataset_query, :as card} (t2/select-one :model/Card :id card-id, :archived false)]
(when-let [{query :dataset_query
:keys [dataset result_metadata]
:as card} (t2/select-one :model/Card :id card-id, :archived false)]
(let [query (assoc query :async? false)
process-query (fn []
(binding [qp.perms/*card-id* card-id]
(qp/process-query-and-save-with-max-results-constraints!
(assoc query :middleware {:process-viz-settings? true
:js-int-to-string? false})
(merge {:executed-by pulse-creator-id
:context :pulse
:card-id card-id}
(merge (cond->
{:executed-by pulse-creator-id
:context :pulse
:card-id card-id}
dataset
(assoc :metadata/dataset-metadata result_metadata))
options))))
result (if pulse-creator-id
(mw.session/with-current-user pulse-creator-id
......
......@@ -4,19 +4,23 @@
These tests should build content then mock out distrubution by usual channels (e.g. email) and check the results of
the distributed content for correctness."
(:require
[clojure.string :as str]
[clojure.test :refer :all]
[hickory.core :as hik]
[hickory.select :as hik.s]
[metabase.email :as email]
[metabase.models :refer [Card Dashboard DashboardCard Pulse PulseCard PulseChannel PulseChannelRecipient]]
[metabase.pulse]
[metabase.test :as mt]))
[clojure.string :as str]
[clojure.test :refer :all]
[hickory.core :as hik]
[hickory.select :as hik.s]
[metabase.email :as email]
[metabase.models :refer [Card Dashboard DashboardCard Pulse PulseCard PulseChannel PulseChannelRecipient]]
[metabase.pulse]
[metabase.test :as mt]))
(deftest result-metadata-preservation-in-html-static-viz-test
(testing "Results metadata applied to a model or query based on a model should be used in the HTML rendering of the pulse email."
(mt/dataset sample-dataset
(mt/with-temp [Card {base-card-id :id} {:name "Base question - no special metadata"
(defmacro with-metadata-data-cards
"Provide a fixture that includes:
- A card with a custom column
- A model with curated metadata (override on the Tax Rate type)
- A question based on the above model"
[[base-card-id model-card-id question-card-id] & body]
`(mt/dataset ~'sample-dataset
(mt/with-temp [Card {~base-card-id :id} {:name "Base question - no special metadata"
:dataset_query {:database (mt/id)
:type :query
:query {:source-table (mt/id :orders)
......@@ -27,11 +31,11 @@
[:field (mt/id :orders :total) {:base-type :type/Float}]
[:expression "Tax Rate"]]
:limit 10}}}
Card {model-card-id :id} {:name "Model with percent semantic type"
Card {~model-card-id :id} {:name "Model with percent semantic type"
:dataset true
:dataset_query {:type :query
:database (mt/id)
:query {:source-table (format "card__%s" base-card-id)}}
:query {:source-table (format "card__%s" ~'base-card-id)}}
:result_metadata [{:name "TAX"
:display_name "Tax"
:base_type :type/Float}
......@@ -43,11 +47,41 @@
:base_type :type/Float
:semantic_type :type/Percentage
:field_ref [:field "Tax Rate" {:base-type :type/Float}]}]}
Card {question-card-id :id} {:name "Query based on model"
Card {~question-card-id :id} {:name "Query based on model"
:dataset_query {:type :query
:database (mt/id)
:query {:source-table (format "card__%s" model-card-id)}}}
Dashboard {dash-id :id} {:name "just dash"}
:query {:source-table (format "card__%s" ~'model-card-id)}}}]
~@body)))
(defn- run-pulse-and-return-last-data-columns
"Simulate sending the pulse email, get the html body of the response, then return the last columns of each pulse body
element. In our test cases that's the Tax Rate column."
[pulse]
(mt/with-fake-inbox
(with-redefs [email/bcc-enabled? (constantly false)]
(mt/with-test-user nil
(metabase.pulse/send-pulse! pulse)))
(let [html-body (get-in @mt/inbox ["rasta@metabase.com" 0 :body 0 :content])
doc (-> html-body hik/parse hik/as-hickory)
data-tables (hik.s/select
(hik.s/class "pulse-body")
doc)]
(map
(fn [data-table]
(->> (hik.s/select
(hik.s/child
(hik.s/tag :tbody)
(hik.s/tag :tr)
hik.s/last-child)
data-table)
(map (comp first :content))))
data-tables))))
(deftest result-metadata-preservation-in-html-static-viz-for-dashboard-test
(testing "In a dashboard, results metadata applied to a model or query based on a model should be used in the HTML rendering of the pulse email."
#_{:clj-kondo/ignore [:unresolved-symbol]}
(with-metadata-data-cards [base-card-id model-card-id question-card-id]
(mt/with-temp [Dashboard {dash-id :id} {:name "just dash"}
DashboardCard {base-dash-card-id :id} {:dashboard_id dash-id
:card_id base-card-id}
DashboardCard {model-dash-card-id :id} {:dashboard_id dash-id
......@@ -71,34 +105,60 @@
:enabled true}
PulseChannelRecipient _ {:pulse_channel_id pulse-channel-id
:user_id (mt/user->id :rasta)}]
(mt/with-fake-inbox
(with-redefs [email/bcc-enabled? (constantly false)]
(mt/with-test-user nil
(metabase.pulse/send-pulse! pulse)))
;; NOTE -- (get-in @mt/inbox ["rasta@metabase.com"]) produces the full email to rasta.
;; This test checks the HTML output for formatting only. A TODO is to add tests to check the attached csv and
;; any other result form for consistent formatting as well. As of 2023-12-01, all 2 csvs in this attachment
;; are not formatted as percentages. We need to add an issue and fix this.
(let [html-body (get-in @mt/inbox ["rasta@metabase.com" 0 :body 0 :content])
doc (-> html-body hik/parse hik/as-hickory)
data-tables (hik.s/select
(hik.s/class "pulse-body")
doc)
[base-data-row
model-data-row
question-data-row] (map
(fn [data-table]
(->> (hik.s/select
(hik.s/child
(hik.s/tag :tbody)
(hik.s/tag :tr)
hik.s/last-child)
data-table)
(map (comp first :content))))
data-tables)]
(testing "The data from the first question is just numbers."
(is (every? (partial re-matches #"\d+\.\d+") base-data-row)))
(testing "The data from the second question (a model) is percent formatted"
(is (every? #(str/ends-with? % "%") model-data-row)))
(testing "The data from the last question (based on the above model) is percent formatted"
(is (every? #(str/ends-with? % "%") question-data-row)))))))))
;; NOTE -- (get-in @mt/inbox ["rasta@metabase.com"]) produces the full email to rasta.
;; This test checks the HTML output for formatting only. A TODO is to add tests to check the attached csv and
;; any other result form for consistent formatting as well. As of 2023-12-01, all 2 csvs in this attachment
;; are not formatted as percentages. We need to add an issue and fix this.
(let [[base-data-row
model-data-row
question-data-row] (run-pulse-and-return-last-data-columns pulse)]
(testing "The data from the first question is just numbers."
(is (every? (partial re-matches #"\d+\.\d+") base-data-row)))
(testing "The data from the second question (a model) is percent formatted"
(is (every? #(str/ends-with? % "%") model-data-row)))
(testing "The data from the last question (based on the above model) is percent formatted"
(is (every? #(str/ends-with? % "%") question-data-row))))))))
(deftest simple-model-with-metadata-no-dashboard-in-html-static-viz-test
(testing "Cards with metadata sent as pulses should preserve metadata in html formatting (#36323)"
;; NOTE: We run this as three tests vs. creating a single pulse with 3 images due to ordering issues. The reason for
;; this is that it appears that the image ordering when running without a dashboard is indeterminate. In general,
;; you subscribe to either a dashboard or a card anyway, so it probably isn't a current real world use case to
;; combine all 3 cards into a single pulse with no dashboard.
(with-metadata-data-cards [base-card-id model-card-id question-card-id]
(testing "The data from the first question is just numbers."
(mt/with-temp [Pulse {pulse-id :id
:as pulse} {:name "Test Pulse"}
PulseCard _ {:pulse_id pulse-id
:card_id base-card-id}
PulseChannel {pulse-channel-id :id} {:channel_type :email
:pulse_id pulse-id
:enabled true}
PulseChannelRecipient _ {:pulse_channel_id pulse-channel-id
:user_id (mt/user->id :rasta)}]
(is (every? (partial re-matches #"\d+\.\d+")
(first (run-pulse-and-return-last-data-columns pulse))))))
(testing "The data from the second question (a model) is percent formatted"
(mt/with-temp [Pulse {pulse-id :id
:as pulse} {:name "Test Pulse"}
PulseCard _ {:pulse_id pulse-id
:card_id model-card-id}
PulseChannel {pulse-channel-id :id} {:channel_type :email
:pulse_id pulse-id
:enabled true}
PulseChannelRecipient _ {:pulse_channel_id pulse-channel-id
:user_id (mt/user->id :rasta)}]
(is (every? #(str/ends-with? % "%")
(first (run-pulse-and-return-last-data-columns pulse))))))
(testing "The data from the last question (based ona a model) is percent formatted"
(mt/with-temp [Pulse {pulse-id :id
:as pulse} {:name "Test Pulse"}
PulseCard _ {:pulse_id pulse-id
:card_id question-card-id}
PulseChannel {pulse-channel-id :id} {:channel_type :email
:pulse_id pulse-id
:enabled true}
PulseChannelRecipient _ {:pulse_channel_id pulse-channel-id
:user_id (mt/user->id :rasta)}]
(is (every? #(str/ends-with? % "%")
(first (run-pulse-and-return-last-data-columns pulse)))))))))
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment