diff --git a/src/metabase/lib/js.cljs b/src/metabase/lib/js.cljs index 410b61b699469e91da286a60476830fd8a7660a3..6d52930cc66f5e994bc28cd3292ac234f1f14b16 100644 --- a/src/metabase/lib/js.cljs +++ b/src/metabase/lib/js.cljs @@ -128,7 +128,11 @@ "Return a sequence of Column metadatas about the columns you can add order bys for in a given stage of `a-query.` To add an order by, pass the result to [[order-by]]." [a-query stage-number] - (to-array (lib.order-by/orderable-columns a-query stage-number))) + ;; Attaches the cached columns directly to this query, in case it gets called again. + (lib.cache/side-channel-cache + (keyword "orderable-columns" (str "stage-" stage-number)) a-query + (fn [_] + (to-array (lib.order-by/orderable-columns a-query stage-number))))) ;; Display-info ===================================================================================================== ;; This is a complicated stack of caches and inner functions, so some guidance is in order. @@ -244,7 +248,11 @@ To break out by a given column, the corresponding element of the result has to be added to the query using [[breakout]]." [a-query stage-number] - (to-array (lib.core/breakoutable-columns a-query stage-number))) + ;; Attaches the cached columns directly to this query, in case it gets called again. + (lib.cache/side-channel-cache + (keyword "breakoutable-columns" (str "stage-" stage-number)) a-query + (fn [_] + (to-array (lib.core/breakoutable-columns a-query stage-number))))) (defn ^:export breakouts "Get the breakout clauses (as an array of opaque objects) in `a-query` at a given `stage-number`. @@ -440,7 +448,11 @@ (defn ^:export filterable-columns "Get the available filterable columns for the stage with `stage-number` of the query `a-query`." [a-query stage-number] - (to-array (lib.core/filterable-columns a-query stage-number))) + ;; Attaches the cached columns directly to this query, in case it gets called again. + (lib.cache/side-channel-cache + (keyword "filterable-columns" (str "stage-" stage-number)) a-query + (fn [_] + (to-array (lib.core/filterable-columns a-query stage-number))))) (defn ^:export filterable-column-operators "Returns the operators for which `filterable-column` is applicable." @@ -526,7 +538,11 @@ (defn ^:export fieldable-columns "Return a sequence of column metadatas for columns that you can specify in the `:fields` of a query." [a-query stage-number] - (to-array (lib.core/fieldable-columns a-query stage-number))) + ;; Attaches the cached columns directly to this query, in case it gets called again. + (lib.cache/side-channel-cache + (keyword "fieldable-columns" (str "stage-" stage-number)) a-query + (fn [_] + (to-array (lib.core/fieldable-columns a-query stage-number))))) (defn ^:export add-field "Adds a given field (`ColumnMetadata`, as returned from eg. [[visible-columns]]) to the fields returned by the query. @@ -560,6 +576,14 @@ ;; [[lib.convert/legacy-ref->pMBQL]] will handle JS -> Clj conversion as needed (lib.core/find-column-for-legacy-ref a-query stage-number a-legacy-ref columns)) +(defn- visible-columns* + "Inner implementation for [[visible-columns]], which wraps this with caching." + [a-query stage-number] + (let [stage (lib.util/query-stage a-query stage-number) + vis-columns (lib.metadata.calculation/visible-columns a-query stage-number stage) + ret-columns (lib.metadata.calculation/returned-columns a-query stage-number stage)] + (to-array (lib.equality/mark-selected-columns a-query stage-number vis-columns ret-columns)))) + ;; TODO: Added as an expedient to fix metabase/metabase#32373. Due to the interaction with viz-settings, this issue ;; was difficult to fix entirely within MLv2. Once viz-settings are ported, this function should not be needed, and the ;; FE logic using it should be ported to MLv2 behind more meaningful names. @@ -568,19 +592,29 @@ Does not pass any options to [[visible-columns]], so it uses the defaults." [a-query stage-number] - (let [stage (lib.util/query-stage a-query stage-number) - vis-columns (lib.metadata.calculation/visible-columns a-query stage-number stage) - ret-columns (lib.metadata.calculation/returned-columns a-query stage-number stage)] - (to-array (lib.equality/mark-selected-columns a-query stage-number vis-columns ret-columns)))) + ;; Attaches the cached columns directly to this query, in case it gets called again. + (lib.cache/side-channel-cache + (keyword "visible-columns" (str "stage-" stage-number)) a-query + (fn [_] + (visible-columns* a-query stage-number)))) -(defn ^:export returned-columns - "Return a sequence of column metadatas for columns returned by the query." +(defn- returned-columns* + "Inner implementation for [[returned-columns]], which wraps this with caching." [a-query stage-number] (let [stage (lib.util/query-stage a-query stage-number)] (->> (lib.metadata.calculation/returned-columns a-query stage-number stage) (map #(assoc % :selected? true)) to-array))) +(defn ^:export returned-columns + "Return a sequence of column metadatas for columns returned by the query." + [a-query stage-number] + ;; Attaches the cached columns directly to this query, in case it gets called again. + (lib.cache/side-channel-cache + (keyword "returned-columns" (str "stage-" stage-number)) a-query + (fn [_] + (returned-columns* a-query stage-number)))) + (defn- normalize-legacy-ref [a-ref] (if (#{:metric :segment} (first a-ref)) @@ -721,10 +755,15 @@ (defn ^:export expressionable-columns "Return an array of Column metadatas about the columns that can be used in an expression in a given stage of `a-query`. Pass the current `expression-position` or `null` for new expressions." - ([a-query expression-position] - (expressionable-columns a-query expression-position)) - ([a-query stage-number expression-position] - (to-array (lib.core/expressionable-columns a-query stage-number expression-position)))) + [a-query stage-number expression-position] + (lib.cache/side-channel-cache + ;; Caching is based on both the stage and expression position, since they can return different sets. + ;; TODO: Since these caches are mainly here to avoid expensively recomputing things in rapid succession, it would + ;; probably suffice to cache only the last position, and evict if it's different. But the lib.cache system doesn't + ;; support that currently. + (keyword "expressionable-columns" (str "stage-" stage-number "-" expression-position)) a-query + (fn [_] + (to-array (lib.core/expressionable-columns a-query stage-number expression-position))))) (defn ^:export suggested-join-conditions "Return suggested default join conditions when constructing a join against `joinable`, e.g. a Table, Saved @@ -916,6 +955,8 @@ something [[Joinable]] (i.e., a Table or Card) or manipulating an existing join. When passing in a join, currently selected columns (those in the join's `:fields`) will include `:selected true` information." [a-query stage-number join-or-joinable] + ;; TODO: It's not practical to cache this currently. We need to be able to key off the query and the joinable, which + ;; is not supported by the lib.cache system. (to-array (lib.core/joinable-columns a-query stage-number join-or-joinable))) (defn ^:export table-or-card-metadata