From 3530241c94233395cdbff8256b03934b0ea726bc Mon Sep 17 00:00:00 2001 From: dpsutton <dan@dpsutton.com> Date: Thu, 16 Jun 2022 16:28:08 -0500 Subject: [PATCH] Handle quarters in native queries (#23368) MBQL goes through a different path. This is only for native queries. Also the diff is huge. Ignoring whitespace shows a very modest diff: ```diff "month" {:unit-range month-range :to-period t/months} + "quarter" {:unit-range relative-quarter-range + :to-period (comp t/months (partial * 3))} "year" {:unit-range year-range :to-period t/years}}) ... "past2months~" {:end "2016-06-30", :start "2016-04-01"} "past13months" {:end "2016-05-31", :start "2015-05-01"} + "past2quarters" {:end "2016-03-31", :start "2015-10-01"} + "past2quarters~" {:end "2016-06-30", :start "2015-10-01"} "past1years" {:end "2015-12-31", :start "2015-01-01"} "past1years~" {:end "2016-12-31", :start "2015-01-01"} ``` Helpful Testing Strategies -------------------------- Sample DB ========= `select id, created_at from orders where {{created_at}}` and set a field filter of type "Date Filter" Custom Table ============ Create a table that has entries in the middle of each quarter. ```sql esting=# create table quarters (id serial primary key not null, description text, t timestamptz); CREATE TABLE testing=# insert into quarters (description, t) values ('Q1 2022', '2022-02-01'), ('Q2 2022', '2022-05-01'), ('Q3 2022', '2022-08-01'), ('Q4 2022', '2022-11-01'); INSERT 0 4 testing=# select * from quarters; id | description | t ----+-------------+------------------------ 1 | Q1 2022 | 2022-02-01 00:00:00-06 2 | Q2 2022 | 2022-05-01 00:00:00-05 3 | Q3 2022 | 2022-08-01 00:00:00-05 4 | Q4 2022 | 2022-11-01 00:00:00-05 (4 rows) ``` Before this change ------------------ > Cannot invoke "clojure.lang.IFn.invoke(Object)" because "to_period" is null (note, if you cannot reproduce this its because you haven't gotten https://github.com/metabase/metabase/pull/23346 in your local version which ensured errors always appear as errors on the frontend. java 11 has no message for NPE so the Frontend missed it was an error and displayed no results as if the query had succeeded) This was the case for the following scenarios: - "last1quarters" - "last1quarters~" - "thisquarter" - "next1quarters" - "next1quarters~" where the ~ means to include the current quarter. After this change ----------------- Running the queries against the custom table I made (current time is Jun 15, Q2) - "last1quarters": "Q1 2022" "February 1, 2022, 6:00 AM" - "last1quarters~": "Q1 2022" "February 1, 2022, 6:00 AM" | "Q2 2022" "May 1, 2022, 5:00 AM" - "thisquarter": "Q2 2022" "May 1, 2022, 5:00 AM" - "next1quarters" "Q3 2022" "August 1, 2022, 5:00 AM" - "next1quarters~": "Q2 2022" "May 1, 2022, 5:00 AM" | "Q3 2022" "August 1, 2022, 5:00 AM" And of course added tests into the matrix for the date parsing. --- .../driver/common/parameters/dates.clj | 40 +++++---- .../driver/common/parameters/dates_test.clj | 84 ++++++++++--------- 2 files changed, 68 insertions(+), 56 deletions(-) diff --git a/src/metabase/driver/common/parameters/dates.clj b/src/metabase/driver/common/parameters/dates.clj index 16f0c440b9e..de6da786418 100644 --- a/src/metabase/driver/common/parameters/dates.clj +++ b/src/metabase/driver/common/parameters/dates.clj @@ -69,7 +69,11 @@ (defn- year-range [start end] (comparison-range start end :year)) -(defn- quarter-range +(defn- relative-quarter-range + [start end] + (comparison-range start end :quarter)) + +(defn- absolute-quarter-range [quarter year] (let [year-quarter (t/year-quarter year (case quarter "Q1" 1 @@ -80,20 +84,22 @@ :end (.atEndOfQuarter year-quarter)})) (def ^:private operations-by-date-unit - {"second" {:unit-range second-range - :to-period t/seconds} - "minute" {:unit-range minute-range - :to-period t/minutes} - "hour" {:unit-range hour-range - :to-period t/hours} - "day" {:unit-range day-range - :to-period t/days} - "week" {:unit-range week-range - :to-period t/weeks} - "month" {:unit-range month-range - :to-period t/months} - "year" {:unit-range year-range - :to-period t/years}}) + {"second" {:unit-range second-range + :to-period t/seconds} + "minute" {:unit-range minute-range + :to-period t/minutes} + "hour" {:unit-range hour-range + :to-period t/hours} + "day" {:unit-range day-range + :to-period t/days} + "week" {:unit-range week-range + :to-period t/weeks} + "month" {:unit-range month-range + :to-period t/months} + "quarter" {:unit-range relative-quarter-range + :to-period (comp t/months (partial * 3))} + "year" {:unit-range year-range + :to-period t/years}}) (defn- maybe-reduce-resolution [unit dt] (if @@ -269,9 +275,9 @@ ;; quarter year {:parser (regex->parser #"(Q[1-4]{1})-([0-9]{4})" [:quarter :year]) :range (fn [{:keys [quarter year]} _] - (quarter-range quarter (Integer/parseInt year))) + (absolute-quarter-range quarter (Integer/parseInt year))) :filter (fn [{:keys [quarter year]} field-clause] - (range->filter (quarter-range quarter (Integer/parseInt year)) + (range->filter (absolute-quarter-range quarter (Integer/parseInt year)) field-clause))} ;; single day {:parser (regex->parser #"([0-9-T:]+)" [:date]) diff --git a/test/metabase/driver/common/parameters/dates_test.clj b/test/metabase/driver/common/parameters/dates_test.clj index d63defb18b5..ab3b6412cb3 100644 --- a/test/metabase/driver/common/parameters/dates_test.clj +++ b/test/metabase/driver/common/parameters/dates_test.clj @@ -133,45 +133,51 @@ "2016-04-18~2016-04-23" {:end "2016-04-23", :start "2016-04-18"} "2016-04-18~" {:start "2016-04-18"} "~2016-04-18" {:end "2016-04-18"}} - "relative (past)" {"past30seconds" {:end "2016-06-07T12:13:54", :start "2016-06-07T12:13:25"} - "past5minutes~" {:end "2016-06-07T12:13:00", :start "2016-06-07T12:08:00"} - "past3hours" {:end "2016-06-07T11:00:00", :start "2016-06-07T09:00:00"} - "past3days" {:end "2016-06-06", :start "2016-06-04"} - "past3days~" {:end "2016-06-07", :start "2016-06-04"} - "past7days" {:end "2016-06-06", :start "2016-05-31"} - "past30days" {:end "2016-06-06", :start "2016-05-08"} - "past2months" {:end "2016-05-31", :start "2016-04-01"} - "past2months~" {:end "2016-06-30", :start "2016-04-01"} - "past13months" {:end "2016-05-31", :start "2015-05-01"} - "past1years" {:end "2015-12-31", :start "2015-01-01"} - "past1years~" {:end "2016-12-31", :start "2015-01-01"} - "past16years" {:end "2015-12-31", :start "2000-01-01"}} - "relative (next)" {"next45seconds" {:end "2016-06-07T12:14:40", :start "2016-06-07T12:13:56"} - "next20minutes" {:end "2016-06-07T12:33:00", :start "2016-06-07T12:14:00"} - "next6hours" {:end "2016-06-07T18:00:00", :start "2016-06-07T13:00:00"} - "next3days" {:end "2016-06-10", :start "2016-06-08"} - "next3days~" {:end "2016-06-10", :start "2016-06-07"} - "next7days" {:end "2016-06-14", :start "2016-06-08"} - "next30days" {:end "2016-07-07", :start "2016-06-08"} - "next2months" {:end "2016-08-31", :start "2016-07-01"} - "next2months~" {:end "2016-08-31", :start "2016-06-01"} - "next13months" {:end "2017-07-31", :start "2016-07-01"} - "next1years" {:end "2017-12-31", :start "2017-01-01"} - "next1years~" {:end "2017-12-31", :start "2016-01-01"} - "next16years" {:end "2032-12-31", :start "2017-01-01"}} - "relative (this)" {"thissecond" {:end "2016-06-07T12:13:55", :start "2016-06-07T12:13:55"} - "thisminute" {:end "2016-06-07T12:13:00", :start "2016-06-07T12:13:00"} - "thishour" {:end "2016-06-07T12:00:00", :start "2016-06-07T12:00:00"} - "thisday" {:end "2016-06-07", :start "2016-06-07"} - "thisweek" {:end "2016-06-11", :start "2016-06-05"} - "thismonth" {:end "2016-06-30", :start "2016-06-01"} - "thisyear" {:end "2016-12-31", :start "2016-01-01"}} - "relative (last)" {"lastsecond" {:end "2016-06-07T12:13:54", :start "2016-06-07T12:13:54"} - "lastminute" {:end "2016-06-07T12:12:00", :start "2016-06-07T12:12:00"} - "lasthour" {:end "2016-06-07T11:00:00", :start "2016-06-07T11:00:00"} - "lastweek" {:end "2016-06-04", :start "2016-05-29"} - "lastmonth" {:end "2016-05-31", :start "2016-05-01"} - "lastyear" {:end "2015-12-31", :start "2015-01-01"}} + "relative (past)" {"past30seconds" {:end "2016-06-07T12:13:54", :start "2016-06-07T12:13:25"} + "past5minutes~" {:end "2016-06-07T12:13:00", :start "2016-06-07T12:08:00"} + "past3hours" {:end "2016-06-07T11:00:00", :start "2016-06-07T09:00:00"} + "past3days" {:end "2016-06-06", :start "2016-06-04"} + "past3days~" {:end "2016-06-07", :start "2016-06-04"} + "past7days" {:end "2016-06-06", :start "2016-05-31"} + "past30days" {:end "2016-06-06", :start "2016-05-08"} + "past2months" {:end "2016-05-31", :start "2016-04-01"} + "past2months~" {:end "2016-06-30", :start "2016-04-01"} + "past13months" {:end "2016-05-31", :start "2015-05-01"} + "past2quarters" {:end "2016-03-31", :start "2015-10-01"} + "past2quarters~" {:end "2016-06-30", :start "2015-10-01"} + "past1years" {:end "2015-12-31", :start "2015-01-01"} + "past1years~" {:end "2016-12-31", :start "2015-01-01"} + "past16years" {:end "2015-12-31", :start "2000-01-01"}} + "relative (next)" {"next45seconds" {:end "2016-06-07T12:14:40", :start "2016-06-07T12:13:56"} + "next20minutes" {:end "2016-06-07T12:33:00", :start "2016-06-07T12:14:00"} + "next6hours" {:end "2016-06-07T18:00:00", :start "2016-06-07T13:00:00"} + "next3days" {:end "2016-06-10", :start "2016-06-08"} + "next3days~" {:end "2016-06-10", :start "2016-06-07"} + "next7days" {:end "2016-06-14", :start "2016-06-08"} + "next30days" {:end "2016-07-07", :start "2016-06-08"} + "next2months" {:end "2016-08-31", :start "2016-07-01"} + "next2months~" {:end "2016-08-31", :start "2016-06-01"} + "next2quarters" {:end "2016-12-31", :start "2016-07-01"} + "next2quarters~" {:end "2016-12-31", :start "2016-04-01"} + "next13months" {:end "2017-07-31", :start "2016-07-01"} + "next1years" {:end "2017-12-31", :start "2017-01-01"} + "next1years~" {:end "2017-12-31", :start "2016-01-01"} + "next16years" {:end "2032-12-31", :start "2017-01-01"}} + "relative (this)" {"thissecond" {:end "2016-06-07T12:13:55", :start "2016-06-07T12:13:55"} + "thisminute" {:end "2016-06-07T12:13:00", :start "2016-06-07T12:13:00"} + "thishour" {:end "2016-06-07T12:00:00", :start "2016-06-07T12:00:00"} + "thisday" {:end "2016-06-07", :start "2016-06-07"} + "thisweek" {:end "2016-06-11", :start "2016-06-05"} + "thismonth" {:end "2016-06-30", :start "2016-06-01"} + "thisquarter" {:end "2016-06-30", :start "2016-04-01"} + "thisyear" {:end "2016-12-31", :start "2016-01-01"}} + "relative (last)" {"lastsecond" {:end "2016-06-07T12:13:54", :start "2016-06-07T12:13:54"} + "lastminute" {:end "2016-06-07T12:12:00", :start "2016-06-07T12:12:00"} + "lasthour" {:end "2016-06-07T11:00:00", :start "2016-06-07T11:00:00"} + "lastweek" {:end "2016-06-04", :start "2016-05-29"} + "lastmonth" {:end "2016-05-31", :start "2016-05-01"} + "lastquarter" {:end "2016-03-31", :start "2016-01-01"} + "lastyear" {:end "2015-12-31", :start "2015-01-01"}} "relative (today/yesterday)" {"yesterday" {:end "2016-06-06", :start "2016-06-06"} "today" {:end "2016-06-07", :start "2016-06-07"}} "relative (past) with starting from" {"past3days-from-3years" {:end "2013-06-07", :start "2013-06-04"}} -- GitLab