Details
-
Bug
-
Status: Closed (View Workflow)
-
Major
-
Resolution: Fixed
-
10.0(EOL), 10.1(EOL), 10.2(EOL), 10.3(EOL), 10.4(EOL)
-
None
Description
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.
The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol:
bool Type_handler:: |
Item_send_time(Item *item, Protocol *protocol, st_value *buf) const |
{
|
item->get_time(protocol->thd, &buf->value.m_time);
|
if (!item->null_value) |
return protocol->store_time(&buf->value.m_time, item->decimals); -- UNEXPECTED DIGITS GET CUT HERE |
return protocol->store_null(); |
}
|
However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself.
Looking into this query:
SELECT GREATEST(8395959, TIME'00:00:00'); |
+-----------------------------------+
|
| GREATEST(8395959, TIME'00:00:00') |
|
+-----------------------------------+
|
| 838:59:59 |
|
+-----------------------------------+
|
one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'.
These queries demonstrate confusing effects of hidden extra digits:
SELECT
|
CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, |
CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, |
CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, |
CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, |
CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, |
CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, |
CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, |
CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, |
CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 |
\G
|
ci: 8385959
|
c0: 8385960
|
c1: 8385960.0
|
c2: 8385960.00
|
c3: 8385960.000
|
c4: 8385960.0000
|
c5: 8385960.00000
|
c6: 8385959.999999
|
c7: 8385959.9999990
|
Notice:
- conversion to INT truncated the extra digits (column ci)
- conversion to DECIMAL(30, 0..5) rounded the extra digits (columns c0..c5)
- conversion to DECIMAL(30, 6..7) fully revealed the extra digits (columns c6..c7)
The problem is also repeatable with additive expressions:
SELECT
|
GREATEST(8395959, TIME'00:00:00') AS ci, |
GREATEST(8395959, TIME'00:00:00')+0 AS c0, |
GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, |
GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, |
GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, |
GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, |
GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, |
GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, |
GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 |
\G
|
ci: 838:59:59
|
c0: 8385959
|
c1: 8385960.0
|
c2: 8385960.00
|
c3: 8385960.000
|
c4: 8385960.0000
|
c5: 8385960.00000
|
c6: 8385959.999999
|
c7: 8385959.9999990
|
SELECT
|
ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, |
ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, |
ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, |
ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, |
ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, |
ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, |
ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, |
ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, |
ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 |
\G
|
ci: 2001-02-04 22:59:59
|
c0: 20010204225959
|
c1: 20010204225960.0
|
c2: 20010204225960.00
|
c3: 20010204225960.000
|
c4: 20010204225960.0000
|
c5: 20010204225960.00000
|
c6: 20010204225959.999999
|
c7: 20010204225959.9999990
|
SELECT
|
TIMESTAMP('2001-01-01', 8395959) AS ci, |
TIMESTAMP('2001-01-01', 8395959)+0 AS c0, |
TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, |
TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, |
TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, |
TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, |
TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, |
TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, |
TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 |
\G
|
ci: 2001-02-04 22:59:59
|
c0: 20010204225959
|
c1: 20010204225960.0
|
c2: 20010204225960.00
|
c3: 20010204225960.000
|
c4: 20010204225960.0000
|
c5: 20010204225960.00000
|
c6: 20010204225959.999999
|
c7: 20010204225959.9999990
|
The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only
In this script:
SET sql_mode=''; |
DROP TABLE IF EXISTS t1; |
CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); |
SHOW CREATE TABLE t1; |
+-------+--------------------------------------------------------------------------------------------------------------------+
|
| Table | Create Table |
|
+-------+--------------------------------------------------------------------------------------------------------------------+
|
| t1 | CREATE TABLE `t1` (
|
`GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL
|
) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
|
+-------+--------------------------------------------------------------------------------------------------------------------+
|
the result of GREATEST() has data type of TIME(0).
However, MICROSECOND() for the same expression returns 999999:
SELECT GREATEST(8395959, TIME'00:00:00') AS c1, |
MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; |
+-----------+--------+
|
| c1 | c2 |
|
+-----------+--------+
|
| 838:59:59 | 999999 |
|
+-----------+--------+
|
Looks wrong.
The problem is repeatable with MICROSECOND(ADDTIME(..)):
DROP TABLE IF EXISTS t1; |
CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; |
SHOW CREATE TABLE t1; |
+-------+-----------------------------------------------------------------------------------------+
|
| Table | Create Table |
|
+-------+-----------------------------------------------------------------------------------------+
|
| t1 | CREATE TABLE `t1` (
|
`c1` datetime DEFAULT NULL
|
) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
|
+-------+-----------------------------------------------------------------------------------------+
|
SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, |
MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; |
+---------------------+--------+
|
| c1 | c2 |
|
+---------------------+--------+
|
| 2001-02-04 22:59:59 | 999999 |
|
+---------------------+--------+
|
The problem is repeatable with MICROSECOND(TIMESTAMP(..))
SET sql_mode=''; |
DROP TABLE IF EXISTS t1; |
CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; |
SHOW CREATE TABLE t1; |
+-------+-----------------------------------------------------------------------------------------+
|
| Table | Create Table |
|
+-------+-----------------------------------------------------------------------------------------+
|
| t1 | CREATE TABLE `t1` (
|
`c1` datetime DEFAULT NULL
|
) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
|
+-------+-----------------------------------------------------------------------------------------+
|
SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, |
MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; |
+---------------------+--------+
|
| c1 | c2 |
|
+---------------------+--------+
|
| 2001-02-04 22:59:59 | 999999 |
|
+---------------------+--------+
|
Attachments
Issue Links
- blocks
-
MDEV-16991 Rounding vs truncation for TIME, DATETIME, TIMESTAMP
-
- Closed
-
- relates to
-
MDEV-17385 MICROSECOND() returns confusing results with an out-of-range TIME-alike argument
-
- Open
-
-
MDEV-17776 CAST(x AS INTERVAL DAY_SECOND(N))
-
- Closed
-
-
MDEV-23388 Assertion `args[0]->decimals == 0' failed in Item_func_round::fix_arg_int
-
- Closed
-
Activity
Field | Original Value | New Value |
---|---|---|
Link |
This issue blocks |
Description |
In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT). h3. The problem is repeatable with MICROSECOND(GREATEST(..)) In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} |
In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).
h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} |
Description |
In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).
h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} |
In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).
h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} |
Description |
In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).
h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} |
In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).
h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} |
Description |
In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).
h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} |
In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).
h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More examples {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('839:59:59') AS c1, MICROSECOND('839:59:59') AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} |
Description |
In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).
h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More examples {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('839:59:59') AS c1, MICROSECOND('839:59:59') AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} |
In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).
h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More examples {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('839:59:59') AS c1, MICROSECOND('839:59:59') AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME('916:40:00', '416:40:00') AS c3, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4; {code} {noformat} +-----------+-----------+------------------+-----------+ | c1 | c2 | c3 | c4 | +-----------+-----------+------------------+-----------+ | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 | +-----------+-----------+------------------+-----------+ {noformat} |
Description |
In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).
h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More examples {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('839:59:59') AS c1, MICROSECOND('839:59:59') AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME('916:40:00', '416:40:00') AS c3, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4; {code} {noformat} +-----------+-----------+------------------+-----------+ | c1 | c2 | c3 | c4 | +-----------+-----------+------------------+-----------+ | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 | +-----------+-----------+------------------+-----------+ {noformat} |
In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).
h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More examples {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('839:59:59') AS c1, MICROSECOND('839:59:59') AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME('916:40:00', '416:40:00') AS c3, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4; {code} {noformat} +-----------+-----------+------------------+-----------+ | c1 | c2 | c3 | c4 | +-----------+-----------+------------------+-----------+ | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 | +-----------+-----------+------------------+-----------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS c1, ADDTIME('00:00:00', 8395959) AS c1; {code} {noformat} +-----------+------------------+ | c1 | c1 | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Description |
In some cases MICROSECOND() returns 999999 for a TIME(0) argument which went outside of the supported TIME range during conversion from other data types (e.g. from INT).
h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More examples {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('839:59:59') AS c1, MICROSECOND('839:59:59') AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME('916:40:00', '416:40:00') AS c3, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4; {code} {noformat} +-----------+-----------+------------------+-----------+ | c1 | c2 | c3 | c4 | +-----------+-----------+------------------+-----------+ | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 | +-----------+-----------+------------------+-----------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS c1, ADDTIME('00:00:00', 8395959) AS c1; {code} {noformat} +-----------+------------------+ | c1 | c1 | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'. The extra digits are not seen when you do just a "SELECT" using the expression, because they are cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits - conversion to DECIMAL(30, 0..5) rounded the extra digits - conversion to DECIMAL(30, 6..7) fully revealed the extra digits h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More examples {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('839:59:59') AS c1, MICROSECOND('839:59:59') AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME('916:40:00', '416:40:00') AS c3, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4; {code} {noformat} +-----------+-----------+------------------+-----------+ | c1 | c2 | c3 | c4 | +-----------+-----------+------------------+-----------+ | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 | +-----------+-----------+------------------+-----------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS c1, ADDTIME('00:00:00', 8395959) AS c1; {code} {noformat} +-----------+------------------+ | c1 | c1 | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Description |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'. The extra digits are not seen when you do just a "SELECT" using the expression, because they are cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits - conversion to DECIMAL(30, 0..5) rounded the extra digits - conversion to DECIMAL(30, 6..7) fully revealed the extra digits h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More examples {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('839:59:59') AS c1, MICROSECOND('839:59:59') AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME('916:40:00', '416:40:00') AS c3, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4; {code} {noformat} +-----------+-----------+------------------+-----------+ | c1 | c2 | c3 | c4 | +-----------+-----------+------------------+-----------+ | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 | +-----------+-----------+------------------+-----------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS c1, ADDTIME('00:00:00', 8395959) AS c1; {code} {noformat} +-----------+------------------+ | c1 | c1 | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.
The extra digits are not seen when you just do a SELECT for such expressions, because they are cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits - conversion to DECIMAL(30, 0..5) rounded the extra digits - conversion to DECIMAL(30, 6..7) fully revealed the extra digits h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More examples {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('839:59:59') AS c1, MICROSECOND('839:59:59') AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME('916:40:00', '416:40:00') AS c3, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4; {code} {noformat} +-----------+-----------+------------------+-----------+ | c1 | c2 | c3 | c4 | +-----------+-----------+------------------+-----------+ | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 | +-----------+-----------+------------------+-----------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS c1, ADDTIME('00:00:00', 8395959) AS c1; {code} {noformat} +-----------+------------------+ | c1 | c1 | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Description |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.
The extra digits are not seen when you just do a SELECT for such expressions, because they are cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits - conversion to DECIMAL(30, 0..5) rounded the extra digits - conversion to DECIMAL(30, 6..7) fully revealed the extra digits h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More examples {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('839:59:59') AS c1, MICROSECOND('839:59:59') AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME('916:40:00', '416:40:00') AS c3, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4; {code} {noformat} +-----------+-----------+------------------+-----------+ | c1 | c2 | c3 | c4 | +-----------+-----------+------------------+-----------+ | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 | +-----------+-----------+------------------+-----------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS c1, ADDTIME('00:00:00', 8395959) AS c1; {code} {noformat} +-----------+------------------+ | c1 | c1 | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.
The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits - conversion to DECIMAL(30, 0..5) rounded the extra digits - conversion to DECIMAL(30, 6..7) fully revealed the extra digits h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More examples {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('839:59:59') AS c1, MICROSECOND('839:59:59') AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME('916:40:00', '416:40:00') AS c3, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4; {code} {noformat} +-----------+-----------+------------------+-----------+ | c1 | c2 | c3 | c4 | +-----------+-----------+------------------+-----------+ | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 | +-----------+-----------+------------------+-----------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS c1, ADDTIME('00:00:00', 8395959) AS c1; {code} {noformat} +-----------+------------------+ | c1 | c1 | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Description |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.
The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits - conversion to DECIMAL(30, 0..5) rounded the extra digits - conversion to DECIMAL(30, 6..7) fully revealed the extra digits h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More examples {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('839:59:59') AS c1, MICROSECOND('839:59:59') AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME('916:40:00', '416:40:00') AS c3, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4; {code} {noformat} +-----------+-----------+------------------+-----------+ | c1 | c2 | c3 | c4 | +-----------+-----------+------------------+-----------+ | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 | +-----------+-----------+------------------+-----------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS c1, ADDTIME('00:00:00', 8395959) AS c1; {code} {noformat} +-----------+------------------+ | c1 | c1 | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.
The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits - conversion to DECIMAL(30, 0..5) rounded the extra digits - conversion to DECIMAL(30, 6..7) fully revealed the extra digits h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More confusing examples when a number gets converted to '838:59:59.999999' {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} h3. More confusing examples when a string gets converted to '838:59:59.999999' Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. {code:sql} SELECT TIME('839:59:59') AS c1, MICROSECOND('839:59:59') AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME('916:40:00', '416:40:00') AS c3, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4; {code} {noformat} +-----------+-----------+------------------+-----------+ | c1 | c2 | c3 | c4 | +-----------+-----------+------------------+-----------+ | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 | +-----------+-----------+------------------+-----------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS c1, ADDTIME('00:00:00', 8395959) AS c1; {code} {noformat} +-----------+------------------+ | c1 | c1 | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Description |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.
The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits - conversion to DECIMAL(30, 0..5) rounded the extra digits - conversion to DECIMAL(30, 6..7) fully revealed the extra digits h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More confusing examples when a number gets converted to '838:59:59.999999' {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} h3. More confusing examples when a string gets converted to '838:59:59.999999' Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. {code:sql} SELECT TIME('839:59:59') AS c1, MICROSECOND('839:59:59') AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME('916:40:00', '416:40:00') AS c3, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4; {code} {noformat} +-----------+-----------+------------------+-----------+ | c1 | c2 | c3 | c4 | +-----------+-----------+------------------+-----------+ | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 | +-----------+-----------+------------------+-----------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS c1, ADDTIME('00:00:00', 8395959) AS c1; {code} {noformat} +-----------+------------------+ | c1 | c1 | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.
The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits - conversion to DECIMAL(30, 0..5) rounded the extra digits - conversion to DECIMAL(30, 6..7) fully revealed the extra digits h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More confusing examples when a number gets converted to '838:59:59.999999' {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} h3. More confusing examples when a string gets converted to '838:59:59.999999' Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. However, intuitively, an explicit conversion from string to time, e.g. using the TIME() function, and an implicit conversion should return the same result. But they do not: {code:sql} SELECT TIME('839:59:59') AS c1, MICROSECOND('839:59:59') AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME('916:40:00', '416:40:00') AS c3, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4; {code} {noformat} +-----------+-----------+------------------+-----------+ | c1 | c2 | c3 | c4 | +-----------+-----------+------------------+-----------+ | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 | +-----------+-----------+------------------+-----------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS c1, ADDTIME('00:00:00', 8395959) AS c1; {code} {noformat} +-----------+------------------+ | c1 | c1 | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Description |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.
The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits - conversion to DECIMAL(30, 0..5) rounded the extra digits - conversion to DECIMAL(30, 6..7) fully revealed the extra digits h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More confusing examples when a number gets converted to '838:59:59.999999' {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} h3. More confusing examples when a string gets converted to '838:59:59.999999' Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. However, intuitively, an explicit conversion from string to time, e.g. using the TIME() function, and an implicit conversion should return the same result. But they do not: {code:sql} SELECT TIME('839:59:59') AS c1, MICROSECOND('839:59:59') AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME('916:40:00', '416:40:00') AS c3, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS c4; {code} {noformat} +-----------+-----------+------------------+-----------+ | c1 | c2 | c3 | c4 | +-----------+-----------+------------------+-----------+ | 838:59:59 | 416:40:00 | 422:19:59.999999 | 422:19:59 | +-----------+-----------+------------------+-----------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS c1, ADDTIME('00:00:00', 8395959) AS c1; {code} {noformat} +-----------+------------------+ | c1 | c1 | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.
The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits - conversion to DECIMAL(30, 0..5) rounded the extra digits - conversion to DECIMAL(30, 6..7) fully revealed the extra digits h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More confusing examples when a number gets converted to '838:59:59.999999' {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} h3. More confusing examples when a string gets converted to '838:59:59.999999' Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments. Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion should return the same result. But they do not, because of different ways of precision detection (mentioned in the previous paragraph): {code:sql} SELECT TIME('839:59:59') AS c0, MICROSECOND(TIME('839:59:59')) AS explicit, MICROSECOND('839:59:59') AS implicit; {code} {noformat} +-----------+----------+----------+ | c0 | explicit | implicit | +-----------+----------+----------+ | 838:59:59 | 0 | 999999 | +-----------+----------+----------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit, SUBTIME('916:40:00', '416:40:00') AS implicit; {code} {noformat} +-----------+-----------+-----------+------------------+ | c1 | c2 | explicit | implicit | +-----------+-----------+-----------+------------------+ | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 | +-----------+-----------+-----------+------------------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS explicit, ADDTIME('00:00:00', 8395959) AS implicit; {code} {noformat} +-----------+------------------+ | explicit | implicit | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Description |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.
The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits - conversion to DECIMAL(30, 0..5) rounded the extra digits - conversion to DECIMAL(30, 6..7) fully revealed the extra digits h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More confusing examples when a number gets converted to '838:59:59.999999' {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} h3. More confusing examples when a string gets converted to '838:59:59.999999' Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments. Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion should return the same result. But they do not, because of different ways of precision detection (mentioned in the previous paragraph): {code:sql} SELECT TIME('839:59:59') AS c0, MICROSECOND(TIME('839:59:59')) AS explicit, MICROSECOND('839:59:59') AS implicit; {code} {noformat} +-----------+----------+----------+ | c0 | explicit | implicit | +-----------+----------+----------+ | 838:59:59 | 0 | 999999 | +-----------+----------+----------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit, SUBTIME('916:40:00', '416:40:00') AS implicit; {code} {noformat} +-----------+-----------+-----------+------------------+ | c1 | c2 | explicit | implicit | +-----------+-----------+-----------+------------------+ | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 | +-----------+-----------+-----------+------------------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS explicit, ADDTIME('00:00:00', 8395959) AS implicit; {code} {noformat} +-----------+------------------+ | explicit | implicit | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.
The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits - conversion to DECIMAL(30, 0..5) rounded the extra digits - conversion to DECIMAL(30, 6..7) fully revealed the extra digits h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More confusing examples when a number gets converted to '838:59:59.999999' {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} h3. More confusing examples when a string gets converted to '838:59:59.999999' Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments. Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion should return the same result. But they do not, because of different ways of precision detection for string arguments (mentioned in the previous paragraph): {code:sql} SELECT TIME('839:59:59') AS c0, MICROSECOND(TIME('839:59:59')) AS explicit, MICROSECOND('839:59:59') AS implicit; {code} {noformat} +-----------+----------+----------+ | c0 | explicit | implicit | +-----------+----------+----------+ | 838:59:59 | 0 | 999999 | +-----------+----------+----------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit, SUBTIME('916:40:00', '416:40:00') AS implicit; {code} {noformat} +-----------+-----------+-----------+------------------+ | c1 | c2 | explicit | implicit | +-----------+-----------+-----------+------------------+ | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 | +-----------+-----------+-----------+------------------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS explicit, ADDTIME('00:00:00', 8395959) AS implicit; {code} {noformat} +-----------+------------------+ | explicit | implicit | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Description |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.
The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits - conversion to DECIMAL(30, 0..5) rounded the extra digits - conversion to DECIMAL(30, 6..7) fully revealed the extra digits h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More confusing examples when a number gets converted to '838:59:59.999999' {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} h3. More confusing examples when a string gets converted to '838:59:59.999999' Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments. Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion should return the same result. But they do not, because of different ways of precision detection for string arguments (mentioned in the previous paragraph): {code:sql} SELECT TIME('839:59:59') AS c0, MICROSECOND(TIME('839:59:59')) AS explicit, MICROSECOND('839:59:59') AS implicit; {code} {noformat} +-----------+----------+----------+ | c0 | explicit | implicit | +-----------+----------+----------+ | 838:59:59 | 0 | 999999 | +-----------+----------+----------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit, SUBTIME('916:40:00', '416:40:00') AS implicit; {code} {noformat} +-----------+-----------+-----------+------------------+ | c1 | c2 | explicit | implicit | +-----------+-----------+-----------+------------------+ | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 | +-----------+-----------+-----------+------------------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS explicit, ADDTIME('00:00:00', 8395959) AS implicit; {code} {noformat} +-----------+------------------+ | explicit | implicit | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.
The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); -- UNEXPECTED DIGITS GET CUT HERE return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits - conversion to DECIMAL(30, 0..5) rounded the extra digits - conversion to DECIMAL(30, 6..7) fully revealed the extra digits h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More confusing examples when a number gets converted to '838:59:59.999999' {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} h3. More confusing examples when a string gets converted to '838:59:59.999999' Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments. Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion should return the same result. But they do not, because of different ways of precision detection for string arguments (mentioned in the previous paragraph): {code:sql} SELECT TIME('839:59:59') AS c0, MICROSECOND(TIME('839:59:59')) AS explicit, MICROSECOND('839:59:59') AS implicit; {code} {noformat} +-----------+----------+----------+ | c0 | explicit | implicit | +-----------+----------+----------+ | 838:59:59 | 0 | 999999 | +-----------+----------+----------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit, SUBTIME('916:40:00', '416:40:00') AS implicit; {code} {noformat} +-----------+-----------+-----------+------------------+ | c1 | c2 | explicit | implicit | +-----------+-----------+-----------+------------------+ | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 | +-----------+-----------+-----------+------------------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS explicit, ADDTIME('00:00:00', 8395959) AS implicit; {code} {noformat} +-----------+------------------+ | explicit | implicit | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Description |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.
The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); -- UNEXPECTED DIGITS GET CUT HERE return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits - conversion to DECIMAL(30, 0..5) rounded the extra digits - conversion to DECIMAL(30, 6..7) fully revealed the extra digits h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More confusing examples when a number gets converted to '838:59:59.999999' {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} h3. More confusing examples when a string gets converted to '838:59:59.999999' Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments. Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion should return the same result. But they do not, because of different ways of precision detection for string arguments (mentioned in the previous paragraph): {code:sql} SELECT TIME('839:59:59') AS c0, MICROSECOND(TIME('839:59:59')) AS explicit, MICROSECOND('839:59:59') AS implicit; {code} {noformat} +-----------+----------+----------+ | c0 | explicit | implicit | +-----------+----------+----------+ | 838:59:59 | 0 | 999999 | +-----------+----------+----------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit, SUBTIME('916:40:00', '416:40:00') AS implicit; {code} {noformat} +-----------+-----------+-----------+------------------+ | c1 | c2 | explicit | implicit | +-----------+-----------+-----------+------------------+ | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 | +-----------+-----------+-----------+------------------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS explicit, ADDTIME('00:00:00', 8395959) AS implicit; {code} {noformat} +-----------+------------------+ | explicit | implicit | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.
The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); -- UNEXPECTED DIGITS GET CUT HERE return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits (column ci) - conversion to DECIMAL(30, 0..5) rounded the extra digits (columns c0..c5) - conversion to DECIMAL(30, 6..7) fully revealed the extra digits (columns c6..c7) h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More confusing examples when a number gets converted to '838:59:59.999999' {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} h3. More confusing examples when a string gets converted to '838:59:59.999999' Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments. Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion should return the same result. But they do not, because of different ways of precision detection for string arguments (mentioned in the previous paragraph): {code:sql} SELECT TIME('839:59:59') AS c0, MICROSECOND(TIME('839:59:59')) AS explicit, MICROSECOND('839:59:59') AS implicit; {code} {noformat} +-----------+----------+----------+ | c0 | explicit | implicit | +-----------+----------+----------+ | 838:59:59 | 0 | 999999 | +-----------+----------+----------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit, SUBTIME('916:40:00', '416:40:00') AS implicit; {code} {noformat} +-----------+-----------+-----------+------------------+ | c1 | c2 | explicit | implicit | +-----------+-----------+-----------+------------------+ | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 | +-----------+-----------+-----------+------------------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS explicit, ADDTIME('00:00:00', 8395959) AS implicit; {code} {noformat} +-----------+------------------+ | explicit | implicit | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Description |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.
The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); -- UNEXPECTED DIGITS GET CUT HERE return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits (column ci) - conversion to DECIMAL(30, 0..5) rounded the extra digits (columns c0..c5) - conversion to DECIMAL(30, 6..7) fully revealed the extra digits (columns c6..c7) h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More confusing examples when a number gets converted to '838:59:59.999999' {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} h3. More confusing examples when a string gets converted to '838:59:59.999999' Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments. Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion should return the same result. But they do not, because of different ways of precision detection for string arguments (mentioned in the previous paragraph): {code:sql} SELECT TIME('839:59:59') AS c0, MICROSECOND(TIME('839:59:59')) AS explicit, MICROSECOND('839:59:59') AS implicit; {code} {noformat} +-----------+----------+----------+ | c0 | explicit | implicit | +-----------+----------+----------+ | 838:59:59 | 0 | 999999 | +-----------+----------+----------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit, SUBTIME('916:40:00', '416:40:00') AS implicit; {code} {noformat} +-----------+-----------+-----------+------------------+ | c1 | c2 | explicit | implicit | +-----------+-----------+-----------+------------------+ | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 | +-----------+-----------+-----------+------------------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS explicit, ADDTIME('00:00:00', 8395959) AS implicit; {code} {noformat} +-----------+------------------+ | explicit | implicit | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.
The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); -- UNEXPECTED DIGITS GET CUT HERE return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits (column ci) - conversion to DECIMAL(30, 0..5) rounded the extra digits (columns c0..c5) - conversion to DECIMAL(30, 6..7) fully revealed the extra digits (columns c6..c7) h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More confusing examples when a number gets converted to '838:59:59.999999' {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} h3. More confusing examples when a string gets converted to '838:59:59.999999' Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments. Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion (from a non-TIME actual parameter to TIME formal parameter) should return the same result. But they do not, because of different ways of precision detection for string arguments (mentioned in the previous paragraph): {code:sql} SELECT TIME('839:59:59') AS c0, MICROSECOND(TIME('839:59:59')) AS explicit, MICROSECOND('839:59:59') AS implicit; {code} {noformat} +-----------+----------+----------+ | c0 | explicit | implicit | +-----------+----------+----------+ | 838:59:59 | 0 | 999999 | +-----------+----------+----------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit, SUBTIME('916:40:00', '416:40:00') AS implicit; {code} {noformat} +-----------+-----------+-----------+------------------+ | c1 | c2 | explicit | implicit | +-----------+-----------+-----------+------------------+ | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 | +-----------+-----------+-----------+------------------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS explicit, ADDTIME('00:00:00', 8395959) AS implicit; {code} {noformat} +-----------+------------------+ | explicit | implicit | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Description |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.
The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); -- UNEXPECTED DIGITS GET CUT HERE return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits (column ci) - conversion to DECIMAL(30, 0..5) rounded the extra digits (columns c0..c5) - conversion to DECIMAL(30, 6..7) fully revealed the extra digits (columns c6..c7) h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More confusing examples when a number gets converted to '838:59:59.999999' {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} h3. More confusing examples when a string gets converted to '838:59:59.999999' Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments. Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion (from a non-TIME actual parameter to TIME formal parameter) should return the same result. But they do not, because of different ways of precision detection for string arguments (mentioned in the previous paragraph): {code:sql} SELECT TIME('839:59:59') AS c0, MICROSECOND(TIME('839:59:59')) AS explicit, MICROSECOND('839:59:59') AS implicit; {code} {noformat} +-----------+----------+----------+ | c0 | explicit | implicit | +-----------+----------+----------+ | 838:59:59 | 0 | 999999 | +-----------+----------+----------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit, SUBTIME('916:40:00', '416:40:00') AS implicit; {code} {noformat} +-----------+-----------+-----------+------------------+ | c1 | c2 | explicit | implicit | +-----------+-----------+-----------+------------------+ | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 | +-----------+-----------+-----------+------------------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS explicit, ADDTIME('00:00:00', 8395959) AS implicit; {code} {noformat} +-----------+------------------+ | explicit | implicit | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.
The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); -- UNEXPECTED DIGITS GET CUT HERE return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits (column ci) - conversion to DECIMAL(30, 0..5) rounded the extra digits (columns c0..c5) - conversion to DECIMAL(30, 6..7) fully revealed the extra digits (columns c6..c7) h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More confusing examples when a number gets converted to '838:59:59.999999' {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} h3. More confusing examples when a string gets converted to '838:59:59.999999' Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments. Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion of a function argument (from a non-TIME actual parameter to TIME formal parameter) should return the same result. But they do not, because of different ways of precision detection for string arguments (mentioned in the previous paragraph): {code:sql} SELECT TIME('839:59:59') AS c0, MICROSECOND(TIME('839:59:59')) AS explicit, MICROSECOND('839:59:59') AS implicit; {code} {noformat} +-----------+----------+----------+ | c0 | explicit | implicit | +-----------+----------+----------+ | 838:59:59 | 0 | 999999 | +-----------+----------+----------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit, SUBTIME('916:40:00', '416:40:00') AS implicit; {code} {noformat} +-----------+-----------+-----------+------------------+ | c1 | c2 | explicit | implicit | +-----------+-----------+-----------+------------------+ | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 | +-----------+-----------+-----------+------------------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS explicit, ADDTIME('00:00:00', 8395959) AS implicit; {code} {noformat} +-----------+------------------+ | explicit | implicit | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Description |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.
The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); -- UNEXPECTED DIGITS GET CUT HERE return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits (column ci) - conversion to DECIMAL(30, 0..5) rounded the extra digits (columns c0..c5) - conversion to DECIMAL(30, 6..7) fully revealed the extra digits (columns c6..c7) h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. More confusing examples when a number gets converted to '838:59:59.999999' {code:sql} SELECT TIME(8395959) AS c1, MICROSECOND(8395959) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(8395959.0) AS c1, MICROSECOND(8395959.0) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS SIGNED)) AS c1, MICROSECOND(CAST(8395959 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} {code:sql} SELECT TIME(CAST(8395959 AS DECIMAL(30,1))) AS c1, MICROSECOND(CAST(8395959 AS DECIMAL(10,1))) AS c2; {code} {noformat} +-------------+--------+ | c1 | c2 | +-------------+--------+ | 838:59:59.9 | 999999 | +-------------+--------+ {noformat} {code:sql} SELECT TIME(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c1, MICROSECOND(COLUMN_GET(COLUMN_CREATE(0, 8395959), 0 AS SIGNED)) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} h3. More confusing examples when a string gets converted to '838:59:59.999999' Note, this problem with string-to-time conversion might be harder to fix than the problem with number-to-time, because in case of strings it's harder to detect the proper amount of fractional digits in the result TIME expression. The explicit conversion function TIME() detects the amount of fractional digits by looking inside constant string arguments. Other functions, like MICROSECOND(), GREATEST(), ADDTIME(), do not look inside constant arguments. Intuitively, an explicit conversion from string to time using the TIME() function, and an implicit conversion of a function argument (from a non-TIME actual parameter to TIME formal parameter) should return the same result. But they do not, because of different ways of precision detection for string arguments (mentioned in the previous paragraph): {code:sql} SELECT TIME('839:59:59') AS c0, MICROSECOND(TIME('839:59:59')) AS explicit, MICROSECOND('839:59:59') AS implicit; {code} {noformat} +-----------+----------+----------+ | c0 | explicit | implicit | +-----------+----------+----------+ | 838:59:59 | 0 | 999999 | +-----------+----------+----------+ {noformat} {code:sql} SELECT TIME('916:40:00') AS c1, TIME('416:40:00') AS c2, SUBTIME(TIME('916:40:00'), TIME('416:40:00')) AS explicit, SUBTIME('916:40:00', '416:40:00') AS implicit; {code} {noformat} +-----------+-----------+-----------+------------------+ | c1 | c2 | explicit | implicit | +-----------+-----------+-----------+------------------+ | 838:59:59 | 416:40:00 | 422:19:59 | 422:19:59.999999 | +-----------+-----------+-----------+------------------+ {noformat} {code:sql} SELECT ADDTIME(time'00:00:00', 8395959) AS explicit, ADDTIME('00:00:00', 8395959) AS implicit; {code} {noformat} +-----------+------------------+ | explicit | implicit | +-----------+------------------+ | 838:59:59 | 838:59:59.999999 | +-----------+------------------+ {noformat} |
Some time expressions can return more hidden fractional digits than the effective data type of this expression supposes to. This happens when an out-of-range non-TIME value gets converted to the maximum possible value of '838:59:59.999999'.
The extra digits are not seen when you just do a SELECT for such expressions, because the extra digits get cut by the protocol: {code:cpp} bool Type_handler:: Item_send_time(Item *item, Protocol *protocol, st_value *buf) const { item->get_time(protocol->thd, &buf->value.m_time); if (!item->null_value) return protocol->store_time(&buf->value.m_time, item->decimals); -- UNEXPECTED DIGITS GET CUT HERE return protocol->store_null(); } {code} However, if you pass this expression to certain functions, e.g. MICROSECOND(), or arithmetic operators, or cast to DECIMAL, the presence of extra fractional digits reveals itself. Looking into this query: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00'); {code} {noformat} +-----------------------------------+ | GREATEST(8395959, TIME'00:00:00') | +-----------------------------------+ | 838:59:59 | +-----------------------------------+ {noformat} one might think that the returned value is '838:59:59'. But in fact it is not. The real returned value is '838:59:59.999999'. These queries demonstrate confusing effects of hidden extra digits: {code:sql} SELECT CAST(GREATEST(8395959, TIME'00:00:00') AS SIGNED) AS ci, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,0)) AS c0, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,1)) AS c1, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,2)) AS c2, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,3)) AS c3, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,4)) AS c4, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,5)) AS c5, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,6)) AS c6, CAST(GREATEST(8395959, TIME'00:00:00') AS DECIMAL(30,7)) AS c7 \G {code} {noformat} ci: 8385959 c0: 8385960 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} Notice: - conversion to INT truncated the extra digits (column ci) - conversion to DECIMAL(30, 0..5) rounded the extra digits (columns c0..c5) - conversion to DECIMAL(30, 6..7) fully revealed the extra digits (columns c6..c7) h3. The problem is also repeatable with additive expressions: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS ci, GREATEST(8395959, TIME'00:00:00')+0 AS c0, GREATEST(8395959, TIME'00:00:00')+0.0 AS c1, GREATEST(8395959, TIME'00:00:00')+0.00 AS c2, GREATEST(8395959, TIME'00:00:00')+0.000 AS c3, GREATEST(8395959, TIME'00:00:00')+0.0000 AS c4, GREATEST(8395959, TIME'00:00:00')+0.00000 AS c5, GREATEST(8395959, TIME'00:00:00')+0.000000 AS c6, GREATEST(8395959, TIME'00:00:00')+0.0000000 AS c7 \G {code} {noformat} ci: 838:59:59 c0: 8385959 c1: 8385960.0 c2: 8385960.00 c3: 8385960.000 c4: 8385960.0000 c5: 8385960.00000 c6: 8385959.999999 c7: 8385959.9999990 {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS ci, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0 AS c0, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0 AS c1, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00 AS c2, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000 AS c3, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000 AS c4, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.00000 AS c5, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.000000 AS c6, ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01', 8395959) AS ci, TIMESTAMP('2001-01-01', 8395959)+0 AS c0, TIMESTAMP('2001-01-01', 8395959)+0.0 AS c1, TIMESTAMP('2001-01-01', 8395959)+0.00 AS c2, TIMESTAMP('2001-01-01', 8395959)+0.000 AS c3, TIMESTAMP('2001-01-01', 8395959)+0.0000 AS c4, TIMESTAMP('2001-01-01', 8395959)+0.00000 AS c5, TIMESTAMP('2001-01-01', 8395959)+0.000000 AS c6, TIMESTAMP('2001-01-01', 8395959)+0.0000000 AS c7 \G {code} {noformat} ci: 2001-02-04 22:59:59 c0: 20010204225959 c1: 20010204225960.0 c2: 20010204225960.00 c3: 20010204225960.000 c4: 20010204225960.0000 c5: 20010204225960.00000 c6: 20010204225959.999999 c7: 20010204225959.9999990 {noformat} h3. The problem is repeatable with MICROSECOND(GREATEST(..)), but starting from 10.3 only In this script: {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT GREATEST(8395959, TIME'00:00:00'); SHOW CREATE TABLE t1; {code} {noformat} +-------+--------------------------------------------------------------------------------------------------------------------+ | Table | Create Table | +-------+--------------------------------------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `GREATEST(8395959, TIME'00:00:00')` time DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+--------------------------------------------------------------------------------------------------------------------+ {noformat} the result of GREATEST() has data type of TIME(0). However, MICROSECOND() for the same expression returns 999999: {code:sql} SELECT GREATEST(8395959, TIME'00:00:00') AS c1, MICROSECOND(GREATEST(8395959, TIME'00:00:00')) AS c2; {code} {noformat} +-----------+--------+ | c1 | c2 | +-----------+--------+ | 838:59:59 | 999999 | +-----------+--------+ {noformat} Looks wrong. h3. The problem is repeatable with MICROSECOND(ADDTIME(..)): {code:sql} DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(ADDTIME(TIMESTAMP'2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} h3. The problem is repeatable with MICROSECOND(TIMESTAMP(..)) {code:sql} SET sql_mode=''; DROP TABLE IF EXISTS t1; CREATE TABLE t1 AS SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1; SHOW CREATE TABLE t1; {code} {noformat} +-------+-----------------------------------------------------------------------------------------+ | Table | Create Table | +-------+-----------------------------------------------------------------------------------------+ | t1 | CREATE TABLE `t1` ( `c1` datetime DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 | +-------+-----------------------------------------------------------------------------------------+ {noformat} {code:sql} SELECT TIMESTAMP('2001-01-01 00:00:00', 8395959) AS c1, MICROSECOND(TIMESTAMP('2001-01-01 00:00:00', 8395959)) AS c2; {code} {noformat} +---------------------+--------+ | c1 | c2 | +---------------------+--------+ | 2001-02-04 22:59:59 | 999999 | +---------------------+--------+ {noformat} |
Summary | MICROSECOND(XXX(int_number_out_of_range)) erroneously returns 999999 | MICROSECOND() erroneously returns 999999 for GREATEST,TIMESTAMP,ADDTIME with an out-of-range TIME argument |
Summary | MICROSECOND() erroneously returns 999999 for GREATEST,TIMESTAMP,ADDTIME with an out-of-range TIME argument | Wrong results for GREATEST,TIMESTAMP,ADDTIME with an out-of-range TIME-alike argument |
Link | This issue relates to MDEV-17385 [ MDEV-17385 ] |
issue.field.resolutiondate | 2018-10-08 09:39:08.0 | 2018-10-08 09:39:08.718 |
Fix Version/s | 10.4.0 [ 23115 ] | |
Fix Version/s | 10.4 [ 22408 ] | |
Resolution | Fixed [ 1 ] | |
Status | Open [ 1 ] | Closed [ 6 ] |
Link |
This issue relates to |
Link |
This issue relates to |
Workflow | MariaDB v3 [ 89863 ] | MariaDB v4 [ 155003 ] |