WHERE timestamp_col = DATE_SUB('2023-09-01 17:17:15',INTERVAL 1 HOUR)
goes through an inefficient execution path. It involves a gmt_sec_to_TIME() call per every row, which in case of the @@time_zone=SYSTEM involves a slow localtime_r() call.
This happens because TIMESTAMP and DATETIME are compared as DATETIME historically.
This could be optimized to use TIMESTAMP comparison when it's possible to avoid TIMESTAMP->DATETIME per-row conversion. This should be faster for big tables.
What can be done
It's generally not possible to compare always TIMESTAMP and DATETIME as TIMESTAMP without behavior changes, because:
DATETIME has a wider range.
Two different TIMESTAMP values can have the same DATETIME value near the "fall back" DST change, as well as for leap seconds.
There are DATETIME gaps during the "spring forward" DST switch, which are adjusted to the start of the gap.
However, if the DATETIME side is a constant inside monotone continuous periods (without DST changes and leap seconds), then we can compare it to TIMESTAMP as TIMESTAMP. The DATETIME argument can be converted once to TIMESTAMP, so no data type conversion will happen per row.
The new method was needed to override the DATETIME-based code in Type_handler_temporal_result::Item_const_eq(), which compares two MYSQL_TIME values, and which is not relevant for TIMESTAMP any more because Item_timestamp_literal does not have a MYSQL_TIME value inside. Without the change, Type_handler_temporal_result::Item_const_eq() would crash on elimination of equal conditions having Item_timestamp_literal inside.
But it appeared that trivial condition elimination haven't been covered in MTR at all so far, this is why you did not get crashes.
Now instead of "return false", it implements detecting equal Item_timestamp_literal's inside this method, and adds MTR tests.
Alexander Barkov
added a comment - - edited psergei , thanks, good catch about MTR.
The new method was needed to override the DATETIME-based code in Type_handler_temporal_result::Item_const_eq(), which compares two MYSQL_TIME values, and which is not relevant for TIMESTAMP any more because Item_timestamp_literal does not have a MYSQL_TIME value inside. Without the change, Type_handler_temporal_result::Item_const_eq() would crash on elimination of equal conditions having Item_timestamp_literal inside.
But it appeared that trivial condition elimination haven't been covered in MTR at all so far, this is why you did not get crashes.
Please find a new patch:
https://github.com/MariaDB/server/commit/c44b546c0067bdf7aa66706af438b4b024a4578b
Now instead of "return false", it implements detecting equal Item_timestamp_literal's inside this method, and adds MTR tests.
Not sure if it can/should be done as part of this patch, yet, continuing with review.
BETWEEN is not handled, either:
explain format=json select * from t1 where a >= '2021-01-01 00:00:00' and a<='2021-02-02 00:00:00';
"attached_condition": "t1.a >= TIMESTAMP/*WITH LOCAL TIME ZONE*/'2021-01-01 00:00:00' and t1.a <= TIMESTAMP/*WITH LOCAL TIME ZONE*/'2021-02-02 00:00:00'"
explain format=json select * from t1 where a between '2021-01-01 00:00:00' and '2021-02-02 00:00:00';
"attached_condition": "t1.a between <cache>('2021-01-01 00:00:00') and <cache>('2021-02-02 00:00:00')"
The commit comment for the patch:
MDEV-32148 Inefficient WHERE timestamp_column=datetime_const_expr
Changing the way how the following condition is evaluated:
WHERE timestamp_column=datetime_const_expr
Before the change it was always performed as DATETIME.
...
this should be changed to indicate that non-equality comparisons are also handled.
For
--let $replace_regex_tsltz6= ...
...
--replace_regex $replace_regex_tsltz6
I would ask to introduce something with less cryptic name and use it in only one location:
Sergei Petrunia
added a comment - - edited An observation: IN is not covered:
MariaDB [test]> EXPLAIN EXTENDED SELECT * FROM t1 WHERE a in ('2020-03-03 01:01:01', '2020-03-04 01:01:01');
+------+-------------+-------+-------+---------------+------+---------+------+------+----------+--------------------------+
| id | select_type | table | type | possible_keys | key | key_len | ref | rows | filtered | Extra |
+------+-------------+-------+-------+---------------+------+---------+------+------+----------+--------------------------+
| 1 | SIMPLE | t1 | range | a | a | 4 | NULL | 2 | 100.00 | Using where; Using index |
+------+-------------+-------+-------+---------------+------+---------+------+------+----------+--------------------------+
1 row in set, 1 warning (0.002 sec)
MariaDB [test]> show warnings;
+-------+------+-----------------------------------------------------------------------------------------------------------------------+
| Level | Code | Message |
+-------+------+-----------------------------------------------------------------------------------------------------------------------+
| Note | 1003 | select `test`.`t1`.`a` AS `a` from `test`.`t1` where `test`.`t1`.`a` in ('2020-03-03 01:01:01','2020-03-04 01:01:01') |
+-------+------+-----------------------------------------------------------------------------------------------------------------------+
Not sure if it can/should be done as part of this patch, yet, continuing with review.
BETWEEN is not handled, either:
explain format=json select * from t1 where a >= '2021-01-01 00:00:00' and a<='2021-02-02 00:00:00';
"attached_condition": "t1.a >= TIMESTAMP/*WITH LOCAL TIME ZONE*/'2021-01-01 00:00:00' and t1.a <= TIMESTAMP/*WITH LOCAL TIME ZONE*/'2021-02-02 00:00:00'"
explain format=json select * from t1 where a between '2021-01-01 00:00:00' and '2021-02-02 00:00:00';
"attached_condition": "t1.a between <cache>('2021-01-01 00:00:00') and <cache>('2021-02-02 00:00:00')"
The commit comment for the patch:
MDEV-32148 Inefficient WHERE timestamp_column=datetime_const_expr
Changing the way how the following condition is evaluated:
WHERE timestamp_column=datetime_const_expr
Before the change it was always performed as DATETIME.
...
this should be changed to indicate that non-equality comparisons are also handled.
For
--let $replace_regex_tsltz6= ...
...
--replace_regex $replace_regex_tsltz6
I would ask to introduce something with less cryptic name and use it in only one location:
--source include/replace_timestamp_w_local_tz_value.inc
return NULL; // SQL NULL DATETIME, or a DATETIME with zeros in YYYYMMDD
// '0000-00-00 00:00:00' is a special valid MariaDB TIMESTAMP value
Sergei Petrunia
added a comment - A suggestion how to get rid of the extra warnings. At the cost of not providing the optimization for the queries that produce such warnings:
--- item_cmpfunc.cc
+++ item_cmpfunc.cc
@@ -460,9 +460,28 @@ get_timestamp_item_for_comparison
!expr1->type_handler()->can_return_date())
return NULL;
+ struct Count_handler : public Internal_error_handler
+ {
+ bool hit=false;
+ bool handle_condition(THD *thd,
+ uint sql_errno,
+ const char *sqlstate,
+ Sql_condition::enum_warning_level *level,
+ const char *msg,
+ Sql_condition **cond_hdl)
+ {
+ hit=true;
+ return *level == Sql_condition::WARN_LEVEL_WARN;
+ }
+ } cnt_handler;
+
+ //Suppress_warnings_error_handler warning_handler;
+ thd->push_internal_handler(&cnt_handler);
+
Datetime dt(thd, expr1, Timestamp::DatetimeOptions(thd));
- if (!dt.is_valid_datetime())
+ thd->pop_internal_handler();
+ if (cnt_handler.hit || !dt.is_valid_datetime())
return NULL; // SQL NULL DATETIME, or a DATETIME with zeros in YYYYMMDD
// '0000-00-00 00:00:00' is a special valid MariaDB TIMESTAMP value
and it will evaluate ##expr1## if that call returns TRUE. This will happen at Name Resolution (fix_fields()) phase. However, this will not attempt to evaluate "cheap" subqueries: subqueries do not yet have query plans at Name Resolution stage, so they are not considered "cheap".
Sergei Petrunia
added a comment - Note to self: The patch has this call:
expr1->can_eval_in_optimize()
and it will evaluate ##expr1## if that call returns TRUE. This will happen at Name Resolution (fix_fields()) phase. However, this will not attempt to evaluate "cheap" subqueries: subqueries do not yet have query plans at Name Resolution stage, so they are not considered "cheap".
Change the target version to be 11.3 (Discussed this with Monty)
After that, it's ok to push into 11.3.
Sergei Petrunia
added a comment - Ok, the patch needs the following:
Address the input in https://jira.mariadb.org/browse/MDEV-32148?focusedCommentId=277190&page=com.atlassian.jira.plugin.system.issuetabpanels:comment-tabpanel#comment-277190
Consider applying comment/formatting fixes from here: https://gist.github.com/spetrunia/89531b74dfb7ba8aeaf6c293e31aa909
Change the target version to be 11.3 (Discussed this with Monty)
After that, it's ok to push into 11.3.
People
Alexander Barkov
Alexander Barkov
Votes:
1Vote for this issue
Watchers:
8Start watching this issue
Dates
Created:
Updated:
Resolved:
Git Integration
Error rendering 'com.xiplink.jira.git.jira_git_plugin:git-issue-webpanel'. Please contact your Jira administrators.
{"report":{"fcp":1287.9000000953674,"ttfb":358.7000000476837,"pageVisibility":"visible","entityId":124962,"key":"jira.project.issue.view-issue","isInitial":true,"threshold":1000,"elementTimings":{},"userDeviceMemory":8,"userDeviceProcessors":64,"apdex":0.5,"journeyId":"9d924c19-3626-4dd1-8879-9e4084530b74","navigationType":0,"readyForUser":1374.4000000953674,"redirectCount":0,"resourceLoadedEnd":2263.7000000476837,"resourceLoadedStart":386.7999999523163,"resourceTiming":[{"duration":398.60000014305115,"initiatorType":"link","name":"https://jira.mariadb.org/s/2c21342762a6a02add1c328bed317ffd-CDN/lu2cib/820016/12ta74/0a8bac35585be7fc6c9cc5a0464cd4cf/_/download/contextbatch/css/_super/batch.css","startTime":386.7999999523163,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":386.7999999523163,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":785.4000000953674,"responseStart":0,"secureConnectionStart":0},{"duration":398.60000014305115,"initiatorType":"link","name":"https://jira.mariadb.org/s/7ebd35e77e471bc30ff0eba799ebc151-CDN/lu2cib/820016/12ta74/494e4c556ecbb29f90a3d3b4f09cb99c/_/download/contextbatch/css/jira.browse.project,project.issue.navigator,jira.view.issue,jira.general,jira.global,atl.general,-_super/batch.css?agile_global_admin_condition=true&jag=true&jira.create.linked.issue=true&slack-enabled=true&whisper-enabled=true","startTime":387.09999990463257,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":387.09999990463257,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":785.7000000476837,"responseStart":0,"secureConnectionStart":0},{"duration":407.89999985694885,"initiatorType":"script","name":"https://jira.mariadb.org/s/0917945aaa57108d00c5076fea35e069-CDN/lu2cib/820016/12ta74/0a8bac35585be7fc6c9cc5a0464cd4cf/_/download/contextbatch/js/_super/batch.js?locale=en","startTime":387.40000009536743,"connectEnd":387.40000009536743,"connectStart":387.40000009536743,"domainLookupEnd":387.40000009536743,"domainLookupStart":387.40000009536743,"fetchStart":387.40000009536743,"redirectEnd":0,"redirectStart":0,"requestStart":387.40000009536743,"responseEnd":795.2999999523163,"responseStart":795.2999999523163,"secureConnectionStart":387.40000009536743},{"duration":442.2000000476837,"initiatorType":"script","name":"https://jira.mariadb.org/s/2d8175ec2fa4c816e8023260bd8c1786-CDN/lu2cib/820016/12ta74/494e4c556ecbb29f90a3d3b4f09cb99c/_/download/contextbatch/js/jira.browse.project,project.issue.navigator,jira.view.issue,jira.general,jira.global,atl.general,-_super/batch.js?agile_global_admin_condition=true&jag=true&jira.create.linked.issue=true&locale=en&slack-enabled=true&whisper-enabled=true","startTime":387.59999990463257,"connectEnd":387.59999990463257,"connectStart":387.59999990463257,"domainLookupEnd":387.59999990463257,"domainLookupStart":387.59999990463257,"fetchStart":387.59999990463257,"redirectEnd":0,"redirectStart":0,"requestStart":387.59999990463257,"responseEnd":829.7999999523163,"responseStart":829.7999999523163,"secureConnectionStart":387.59999990463257},{"duration":445.69999980926514,"initiatorType":"script","name":"https://jira.mariadb.org/s/a9324d6758d385eb45c462685ad88f1d-CDN/lu2cib/820016/12ta74/c92c0caa9a024ae85b0ebdbed7fb4bd7/_/download/contextbatch/js/atl.global,-_super/batch.js?locale=en","startTime":387.90000009536743,"connectEnd":387.90000009536743,"connectStart":387.90000009536743,"domainLookupEnd":387.90000009536743,"domainLookupStart":387.90000009536743,"fetchStart":387.90000009536743,"redirectEnd":0,"redirectStart":0,"requestStart":387.90000009536743,"responseEnd":833.5999999046326,"responseStart":833.5999999046326,"secureConnectionStart":387.90000009536743},{"duration":446.09999990463257,"initiatorType":"script","name":"https://jira.mariadb.org/s/d41d8cd98f00b204e9800998ecf8427e-CDN/lu2cib/820016/12ta74/1.0/_/download/batch/jira.webresources:calendar-en/jira.webresources:calendar-en.js","startTime":388,"connectEnd":388,"connectStart":388,"domainLookupEnd":388,"domainLookupStart":388,"fetchStart":388,"redirectEnd":0,"redirectStart":0,"requestStart":388,"responseEnd":834.0999999046326,"responseStart":834.0999999046326,"secureConnectionStart":388},{"duration":446.2999999523163,"initiatorType":"script","name":"https://jira.mariadb.org/s/d41d8cd98f00b204e9800998ecf8427e-CDN/lu2cib/820016/12ta74/1.0/_/download/batch/jira.webresources:calendar-localisation-moment/jira.webresources:calendar-localisation-moment.js","startTime":388.2000000476837,"connectEnd":388.2000000476837,"connectStart":388.2000000476837,"domainLookupEnd":388.2000000476837,"domainLookupStart":388.2000000476837,"fetchStart":388.2000000476837,"redirectEnd":0,"redirectStart":0,"requestStart":388.2000000476837,"responseEnd":834.5,"responseStart":834.5,"secureConnectionStart":388.2000000476837},{"duration":552.8999998569489,"initiatorType":"link","name":"https://jira.mariadb.org/s/b04b06a02d1959df322d9cded3aeecc1-CDN/lu2cib/820016/12ta74/a2ff6aa845ffc9a1d22fe23d9ee791fc/_/download/contextbatch/css/jira.global.look-and-feel,-_super/batch.css","startTime":388.40000009536743,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":388.40000009536743,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":941.2999999523163,"responseStart":0,"secureConnectionStart":0},{"duration":446.40000009536743,"initiatorType":"script","name":"https://jira.mariadb.org/rest/api/1.0/shortcuts/820016/47140b6e0a9bc2e4913da06536125810/shortcuts.js?context=issuenavigation&context=issueaction","startTime":388.59999990463257,"connectEnd":388.59999990463257,"connectStart":388.59999990463257,"domainLookupEnd":388.59999990463257,"domainLookupStart":388.59999990463257,"fetchStart":388.59999990463257,"redirectEnd":0,"redirectStart":0,"requestStart":388.59999990463257,"responseEnd":835,"responseStart":834.9000000953674,"secureConnectionStart":388.59999990463257},{"duration":552.7000000476837,"initiatorType":"link","name":"https://jira.mariadb.org/s/3ac36323ba5e4eb0af2aa7ac7211b4bb-CDN/lu2cib/820016/12ta74/d176f0986478cc64f24226b3d20c140d/_/download/contextbatch/css/com.atlassian.jira.projects.sidebar.init,-_super,-project.issue.navigator,-jira.view.issue/batch.css?jira.create.linked.issue=true","startTime":388.7000000476837,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":388.7000000476837,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":941.4000000953674,"responseStart":0,"secureConnectionStart":0},{"duration":446.59999990463257,"initiatorType":"script","name":"https://jira.mariadb.org/s/5d5e8fe91fbc506585e83ea3b62ccc4b-CDN/lu2cib/820016/12ta74/d176f0986478cc64f24226b3d20c140d/_/download/contextbatch/js/com.atlassian.jira.projects.sidebar.init,-_super,-project.issue.navigator,-jira.view.issue/batch.js?jira.create.linked.issue=true&locale=en","startTime":388.90000009536743,"connectEnd":388.90000009536743,"connectStart":388.90000009536743,"domainLookupEnd":388.90000009536743,"domainLookupStart":388.90000009536743,"fetchStart":388.90000009536743,"redirectEnd":0,"redirectStart":0,"requestStart":388.90000009536743,"responseEnd":835.5,"responseStart":835.5,"secureConnectionStart":388.90000009536743},{"duration":747.9000000953674,"initiatorType":"script","name":"https://jira.mariadb.org/s/d41d8cd98f00b204e9800998ecf8427e-CDN/lu2cib/820016/12ta74/1.0/_/download/batch/jira.webresources:bigpipe-js/jira.webresources:bigpipe-js.js","startTime":390.5,"connectEnd":390.5,"connectStart":390.5,"domainLookupEnd":390.5,"domainLookupStart":390.5,"fetchStart":390.5,"redirectEnd":0,"redirectStart":0,"requestStart":390.5,"responseEnd":1138.4000000953674,"responseStart":1138.4000000953674,"secureConnectionStart":390.5},{"duration":1873.2000000476837,"initiatorType":"script","name":"https://jira.mariadb.org/s/d41d8cd98f00b204e9800998ecf8427e-CDN/lu2cib/820016/12ta74/1.0/_/download/batch/jira.webresources:bigpipe-init/jira.webresources:bigpipe-init.js","startTime":390.5,"connectEnd":390.5,"connectStart":390.5,"domainLookupEnd":390.5,"domainLookupStart":390.5,"fetchStart":390.5,"redirectEnd":0,"redirectStart":0,"requestStart":390.5,"responseEnd":2263.7000000476837,"responseStart":2263.7000000476837,"secureConnectionStart":390.5},{"duration":185.5,"initiatorType":"xmlhttprequest","name":"https://jira.mariadb.org/rest/webResources/1.0/resources","startTime":953.4000000953674,"connectEnd":953.4000000953674,"connectStart":953.4000000953674,"domainLookupEnd":953.4000000953674,"domainLookupStart":953.4000000953674,"fetchStart":953.4000000953674,"redirectEnd":0,"redirectStart":0,"requestStart":953.4000000953674,"responseEnd":1138.9000000953674,"responseStart":1138.9000000953674,"secureConnectionStart":953.4000000953674},{"duration":1110.2999999523163,"initiatorType":"link","name":"https://jira.mariadb.org/s/d5715adaadd168a9002b108b2b039b50-CDN/lu2cib/820016/12ta74/be4b45e9cec53099498fa61c8b7acba4/_/download/contextbatch/css/jira.project.sidebar,-_super,-project.issue.navigator,-jira.general,-jira.browse.project,-jira.view.issue,-jira.global,-atl.general,-com.atlassian.jira.projects.sidebar.init/batch.css?agile_global_admin_condition=true&jag=true&jira.create.linked.issue=true&slack-enabled=true&whisper-enabled=true","startTime":1236,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":1236,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":2346.2999999523163,"responseStart":0,"secureConnectionStart":0},{"duration":1105.1999998092651,"initiatorType":"script","name":"https://jira.mariadb.org/s/d41d8cd98f00b204e9800998ecf8427e-CDN/lu2cib/820016/12ta74/e65b778d185daf5aee24936755b43da6/_/download/contextbatch/js/browser-metrics-plugin.contrib,-_super,-project.issue.navigator,-jira.view.issue,-atl.general/batch.js?agile_global_admin_condition=true&jag=true&jira.create.linked.issue=true&slack-enabled=true&whisper-enabled=true","startTime":1236.9000000953674,"connectEnd":1236.9000000953674,"connectStart":1236.9000000953674,"domainLookupEnd":1236.9000000953674,"domainLookupStart":1236.9000000953674,"fetchStart":1236.9000000953674,"redirectEnd":0,"redirectStart":0,"requestStart":1236.9000000953674,"responseEnd":2342.0999999046326,"responseStart":2342.0999999046326,"secureConnectionStart":1236.9000000953674},{"duration":1112.2999999523163,"initiatorType":"script","name":"https://jira.mariadb.org/s/097ae97cb8fbec7d6ea4bbb1f26955b9-CDN/lu2cib/820016/12ta74/be4b45e9cec53099498fa61c8b7acba4/_/download/contextbatch/js/jira.project.sidebar,-_super,-project.issue.navigator,-jira.general,-jira.browse.project,-jira.view.issue,-jira.global,-atl.general,-com.atlassian.jira.projects.sidebar.init/batch.js?agile_global_admin_condition=true&jag=true&jira.create.linked.issue=true&locale=en&slack-enabled=true&whisper-enabled=true","startTime":1237.4000000953674,"connectEnd":1237.4000000953674,"connectStart":1237.4000000953674,"domainLookupEnd":1237.4000000953674,"domainLookupStart":1237.4000000953674,"fetchStart":1237.4000000953674,"redirectEnd":0,"redirectStart":0,"requestStart":1237.4000000953674,"responseEnd":2349.7000000476837,"responseStart":2349.7000000476837,"secureConnectionStart":1237.4000000953674}],"fetchStart":0,"domainLookupStart":0,"domainLookupEnd":0,"connectStart":0,"connectEnd":0,"requestStart":152,"responseStart":359,"responseEnd":365,"domLoading":384,"domInteractive":2295,"domContentLoadedEventStart":2295,"domContentLoadedEventEnd":2346,"domComplete":2744,"loadEventStart":2745,"loadEventEnd":2745,"userAgent":"Mozilla/5.0 AppleWebKit/537.36 (KHTML, like Gecko; compatible; ClaudeBot/1.0; +claudebot@anthropic.com)","marks":[{"name":"bigPipe.sidebar-id.start","time":2265.5},{"name":"bigPipe.sidebar-id.end","time":2266.4000000953674},{"name":"bigPipe.activity-panel-pipe-id.start","time":2266.5999999046326},{"name":"bigPipe.activity-panel-pipe-id.end","time":2270.2000000476837},{"name":"activityTabFullyLoaded","time":2368.7999999523163}],"measures":[],"correlationId":"3486b1e96d4b89","effectiveType":"4g","downlink":9,"rtt":0,"serverDuration":121,"dbReadsTimeInMs":19,"dbConnsTimeInMs":29,"applicationHash":"9d11dbea5f4be3d4cc21f03a88dd11d8c8687422","experiments":[]}}
psergei, thanks, good catch about MTR.
The new method was needed to override the DATETIME-based code in Type_handler_temporal_result::Item_const_eq(), which compares two MYSQL_TIME values, and which is not relevant for TIMESTAMP any more because Item_timestamp_literal does not have a MYSQL_TIME value inside. Without the change, Type_handler_temporal_result::Item_const_eq() would crash on elimination of equal conditions having Item_timestamp_literal inside.
But it appeared that trivial condition elimination haven't been covered in MTR at all so far, this is why you did not get crashes.
Please find a new patch:
https://github.com/MariaDB/server/commit/c44b546c0067bdf7aa66706af438b4b024a4578b
Now instead of "return false", it implements detecting equal Item_timestamp_literal's inside this method, and adds MTR tests.