From 6858dcefac340054a069cc49673f75d237816e03 Mon Sep 17 00:00:00 2001 From: metamben <103100869+metamben@users.noreply.github.com> Date: Fri, 15 Jul 2022 22:26:27 +0300 Subject: [PATCH] Retain source alias for aggregates (#23771) --- .../query_processor_test.clj | 19 ++++++ .../query_processor/util/add_alias_info.clj | 19 +++--- .../util/add_alias_info_test.clj | 62 ++++++++++++++++++- 3 files changed, 92 insertions(+), 8 deletions(-) diff --git a/modules/drivers/bigquery-cloud-sdk/test/metabase/driver/bigquery_cloud_sdk/query_processor_test.clj b/modules/drivers/bigquery-cloud-sdk/test/metabase/driver/bigquery_cloud_sdk/query_processor_test.clj index e0718acd5c1..9bbc6b6e47d 100644 --- a/modules/drivers/bigquery-cloud-sdk/test/metabase/driver/bigquery_cloud_sdk/query_processor_test.clj +++ b/modules/drivers/bigquery-cloud-sdk/test/metabase/driver/bigquery_cloud_sdk/query_processor_test.clj @@ -854,3 +854,22 @@ (let [t [:absolute-datetime #t "2022-04-22T18:27:00-08:00"] hsql-form (sql.qp/add-interval-honeysql-form :bigquery-cloud-sdk t 3 :month)] (sql.qp/format-honeysql :bigquery-cloud-sdk hsql-form)))))) + +(deftest custom-expression-with-space-in-having + (mt/test-driver :bigquery-cloud-sdk + (mt/dataset avian-singles + (testing "Custom expressions with spaces are matched properly (#22310)" + (let [name-with-spaces "sum id diff" + sql-query (-> (mt/mbql-query messages + {:filter [:> [:field name-with-spaces {:base-type :type/Float}] 5] + :source-query {:source-table $$messages + :aggregation [[:aggregation-options + [:sum [:- $sender_id $receiver_id]] + {:name name-with-spaces + :display-name name-with-spaces}]] + :breakout [$text]} + :limit 1}) + qp/compile + :query)] + (is (not (str/includes? sql-query name-with-spaces)) + (format "Query `%s' should not contain `%s'" sql-query name-with-spaces))))))) diff --git a/src/metabase/query_processor/util/add_alias_info.clj b/src/metabase/query_processor/util/add_alias_info.clj index 95a623c0d6a..d1beba4b9c3 100644 --- a/src/metabase/query_processor/util/add_alias_info.clj +++ b/src/metabase/query_processor/util/add_alias_info.clj @@ -186,7 +186,7 @@ (defn- exports [query] (into #{} (mbql.u/match (dissoc query :source-query :source-metadata :joins) - [(_ :guard #{:field :expression :aggregation}) _ (_ :guard (every-pred map? ::position))]))) + [(_ :guard #{:field :expression :aggregation-options}) _ (_ :guard (every-pred map? ::position))]))) (defn- join-with-alias [{:keys [joins]} join-alias] (some (fn [join] @@ -216,12 +216,16 @@ (when-let [field-name (let [[_ id-or-name] field-clause] (when (string? id-or-name) id-or-name))] - (when-let [expression-exports (not-empty (filter (partial mbql.u/is-clause? :expression) all-exports))] - (some - (fn [[_ expression-name :as expression-clause]] - (when (= expression-name field-name) - expression-clause)) - expression-exports)))))) + (or (some + (fn [[_ expression-name :as expression-clause]] + (when (= expression-name field-name) + expression-clause)) + (filter (partial mbql.u/is-clause? :expression) all-exports)) + (some + (fn [[_ _ opts :as aggregation-options-clause]] + (when (= (::source-alias opts) field-name) + aggregation-options-clause)) + (filter (partial mbql.u/is-clause? :aggregation-options) all-exports))))))) (defn- matching-field-in-join-at-this-level "If `field-clause` is the result of a join *at this level* with a `:source-query`, return the 'source' `:field` clause @@ -347,6 +351,7 @@ unique-alias (unique-alias-fn position original-ag-name)] [:aggregation-options wrapped-ag-clause (assoc opts :name unique-alias + ::source-alias original-ag-name ::position position ::desired-alias unique-alias)])) diff --git a/test/metabase/query_processor/util/add_alias_info_test.clj b/test/metabase/query_processor/util/add_alias_info_test.clj index 335a5c82e85..21f65346ff9 100644 --- a/test/metabase/query_processor/util/add_alias_info_test.clj +++ b/test/metabase/query_processor/util/add_alias_info_test.clj @@ -187,6 +187,7 @@ {:aggregation [[:aggregation-options [:count] {:name "count" + ::add/source-alias "count" ::add/desired-alias "count" ::add/position 0}]] :filter [:!= @@ -211,16 +212,19 @@ :aggregation [[:aggregation-options [:count] {:name "count" + ::add/source-alias "count" ::add/desired-alias "count" ::add/position 0}] [:aggregation-options [:count] {:name "count_2" + ::add/source-alias "count_2" ::add/desired-alias "count_2" ::add/position 1}] [:aggregation-options [:count] {:name "count_3" + ::add/source-alias "count_3" ::add/desired-alias "count_3" ::add/position 2}]]} :fields [[:field "count" {:base-type :type/BigInteger @@ -305,6 +309,7 @@ [:avg [:field %reviews.rating {::add/source-table $$reviews ::add/source-alias "RATING"}]] {:name "avg" + ::add/source-alias "avg" ::add/desired-alias "avg" ::add/position 1}]] :breakout [[:field %products.category {:join-alias "P2" @@ -370,9 +375,12 @@ (driver/register! ::custom-escape :parent :h2) +(defn- prefix-alias [alias] + (str "COOL." alias)) + (defmethod driver/escape-alias ::custom-escape [_driver field-alias] - (str "COOL." field-alias)) + (prefix-alias field-alias)) (deftest custom-escape-alias-test (let [db (mt/db)] @@ -399,6 +407,7 @@ ::add/position 0}]] {:aggregation [[:aggregation-options [:count] {:name "COOL.count" ::add/position 1 + ::add/source-alias "count" ::add/desired-alias "COOL.count"}]] :breakout [double-price] :order-by [[:asc double-price]]}))) @@ -413,6 +422,55 @@ add-alias-info :query))))))) +(deftest custom-escape-alias-filtering-aggregation-test + (let [db (mt/db)] + (driver/with-driver ::custom-escape + (mt/with-db db + (is (query= (mt/$ids venues + (let [price [:field %price {::add/source-table $$venues + ::add/source-alias "PRICE" + ::add/desired-alias "COOL.PRICE" + ::add/position 0}] + outer-price (-> price + (assoc-in [2 ::add/source-table] ::add/source) + (update-in [2 ::add/source-alias] prefix-alias) + (update-in [2 ::add/desired-alias] prefix-alias)) + count-opts {:name "COOL.strange count" + ::add/source-alias "strange count" + ::add/desired-alias "COOL.strange count" + ::add/position 1} + outer-count-opts (-> count-opts + (dissoc :name) + (assoc :base-type :type/BigInteger + ::add/source-table ::add/source) + (update ::add/source-alias prefix-alias) + (update ::add/desired-alias prefix-alias))] + {:source-query + {:source-table $$venues + :breakout [price] + :aggregation [[:aggregation-options [:count] count-opts]] + :order-by [[:asc price]]} + :fields [outer-price + [:field + "strange count" + outer-count-opts]] + :filter [:< + [:field + "strange count" + outer-count-opts] + [:value 10 {:base_type :type/BigInteger}]] + :limit 1})) + (-> (mt/mbql-query venues + {:source-query {:source-table $$venues + :aggregation [[:aggregation-options + [:count] + {:name "strange count"}]] + :breakout [$price]} + :filter [:< [:field "strange count" {:base-type :type/BigInteger}] 10] + :limit 1}) + add-alias-info + :query))))))) + (driver/register! ::custom-escape-spaces-to-underscores :parent :h2) (defmethod driver/escape-alias ::custom-escape-spaces-to-underscores @@ -555,6 +613,7 @@ ::add/source-alias "USER_ID"}]] {:name "sum" ::add/position 0 + ::add/source-alias "sum" ::add/desired-alias "sum"}]] :order-by [[:asc [:aggregation 0 {::add/desired-alias "sum" ::add/position 0}]]]}) @@ -569,6 +628,7 @@ :breakout [[:expression "count" {::add/desired-alias "count" ::add/position 0}]] :aggregation [[:aggregation-options [:count] {:name "count_2" + ::add/source-alias "count" ::add/desired-alias "count_2" ::add/position 1}]] :order-by [[:asc [:expression "count" {::add/desired-alias "count" -- GitLab