2020-10-02 16:54:07 +00:00
|
|
|
#!/usr/bin/env python3
|
2019-09-07 14:22:22 +00:00
|
|
|
# encoding: utf-8
|
|
|
|
|
|
|
|
import re
|
|
|
|
import itertools
|
2019-10-04 13:49:34 +00:00
|
|
|
import sys
|
2019-10-24 04:46:52 +00:00
|
|
|
import argparse
|
2019-09-07 14:22:22 +00:00
|
|
|
|
|
|
|
# Create SQL statement to verify dateTime64 is accepted as argument to functions taking DateTime.
|
2023-01-13 14:00:16 +00:00
|
|
|
FUNCTIONS = """
|
2019-10-24 04:46:52 +00:00
|
|
|
toTimeZone(N, 'UTC')
|
2022-03-11 23:45:26 +00:00
|
|
|
toYear(N, 'Asia/Istanbul')
|
|
|
|
toQuarter(N, 'Asia/Istanbul')
|
|
|
|
toMonth(N, 'Asia/Istanbul')
|
|
|
|
toDayOfYear(N, 'Asia/Istanbul')
|
|
|
|
toDayOfMonth(N, 'Asia/Istanbul')
|
2023-01-13 14:00:16 +00:00
|
|
|
toDayOfWeek(N, 0, 'Asia/Istanbul')
|
2022-03-11 23:45:26 +00:00
|
|
|
toHour(N, 'Asia/Istanbul')
|
|
|
|
toMinute(N, 'Asia/Istanbul')
|
|
|
|
toSecond(N, 'Asia/Istanbul')
|
2019-10-24 04:46:52 +00:00
|
|
|
toUnixTimestamp(N)
|
2022-03-11 23:45:26 +00:00
|
|
|
toStartOfYear(N, 'Asia/Istanbul')
|
|
|
|
toStartOfISOYear(N, 'Asia/Istanbul')
|
|
|
|
toStartOfQuarter(N, 'Asia/Istanbul')
|
|
|
|
toStartOfMonth(N, 'Asia/Istanbul')
|
|
|
|
toMonday(N, 'Asia/Istanbul')
|
|
|
|
toStartOfWeek(N, 'Asia/Istanbul')
|
|
|
|
toStartOfDay(N, 'Asia/Istanbul')
|
|
|
|
toStartOfHour(N, 'Asia/Istanbul')
|
|
|
|
toStartOfMinute(N, 'Asia/Istanbul')
|
2022-05-27 00:04:42 +00:00
|
|
|
toStartOfFiveMinutes(N, 'Asia/Istanbul')
|
2022-03-11 23:45:26 +00:00
|
|
|
toStartOfTenMinutes(N, 'Asia/Istanbul')
|
|
|
|
toStartOfFifteenMinutes(N, 'Asia/Istanbul')
|
|
|
|
toStartOfInterval(N, INTERVAL 1 year, 'Asia/Istanbul')
|
|
|
|
toStartOfInterval(N, INTERVAL 1 month, 'Asia/Istanbul')
|
|
|
|
toStartOfInterval(N, INTERVAL 1 day, 'Asia/Istanbul')
|
|
|
|
toStartOfInterval(N, INTERVAL 15 minute, 'Asia/Istanbul')
|
|
|
|
date_trunc('year', N, 'Asia/Istanbul')
|
|
|
|
date_trunc('month', N, 'Asia/Istanbul')
|
|
|
|
date_trunc('day', N, 'Asia/Istanbul')
|
|
|
|
date_trunc('minute', N, 'Asia/Istanbul')
|
|
|
|
toTime(N, 'Asia/Istanbul')
|
|
|
|
toRelativeYearNum(N, 'Asia/Istanbul')
|
|
|
|
toRelativeQuarterNum(N, 'Asia/Istanbul')
|
|
|
|
toRelativeMonthNum(N, 'Asia/Istanbul')
|
|
|
|
toRelativeWeekNum(N, 'Asia/Istanbul')
|
|
|
|
toRelativeDayNum(N, 'Asia/Istanbul')
|
|
|
|
toRelativeHourNum(N, 'Asia/Istanbul')
|
|
|
|
toRelativeMinuteNum(N, 'Asia/Istanbul')
|
|
|
|
toRelativeSecondNum(N, 'Asia/Istanbul')
|
|
|
|
toISOYear(N, 'Asia/Istanbul')
|
|
|
|
toISOWeek(N, 'Asia/Istanbul')
|
|
|
|
toWeek(N, 'Asia/Istanbul')
|
|
|
|
toYearWeek(N, 'Asia/Istanbul')
|
|
|
|
timeSlot(N, 'Asia/Istanbul')
|
|
|
|
toYYYYMM(N, 'Asia/Istanbul')
|
|
|
|
toYYYYMMDD(N, 'Asia/Istanbul')
|
|
|
|
toYYYYMMDDhhmmss(N, 'Asia/Istanbul')
|
|
|
|
addYears(N, 1, 'Asia/Istanbul')
|
|
|
|
addMonths(N, 1, 'Asia/Istanbul')
|
|
|
|
addWeeks(N, 1, 'Asia/Istanbul')
|
|
|
|
addDays(N, 1, 'Asia/Istanbul')
|
|
|
|
addHours(N, 1, 'Asia/Istanbul')
|
|
|
|
addMinutes(N, 1, 'Asia/Istanbul')
|
|
|
|
addSeconds(N, 1, 'Asia/Istanbul')
|
|
|
|
addQuarters(N, 1, 'Asia/Istanbul')
|
|
|
|
subtractYears(N, 1, 'Asia/Istanbul')
|
|
|
|
subtractMonths(N, 1, 'Asia/Istanbul')
|
|
|
|
subtractWeeks(N, 1, 'Asia/Istanbul')
|
|
|
|
subtractDays(N, 1, 'Asia/Istanbul')
|
|
|
|
subtractHours(N, 1, 'Asia/Istanbul')
|
|
|
|
subtractMinutes(N, 1, 'Asia/Istanbul')
|
|
|
|
subtractSeconds(N, 1, 'Asia/Istanbul')
|
|
|
|
subtractQuarters(N, 1, 'Asia/Istanbul')
|
2020-03-10 08:10:07 +00:00
|
|
|
CAST(N as DateTime('Europe/Minsk'))
|
2019-10-24 04:46:52 +00:00
|
|
|
CAST(N as Date)
|
|
|
|
CAST(N as UInt64)
|
2020-03-10 08:10:07 +00:00
|
|
|
CAST(N as DateTime64(0, 'Europe/Minsk'))
|
|
|
|
CAST(N as DateTime64(3, 'Europe/Minsk'))
|
|
|
|
CAST(N as DateTime64(6, 'Europe/Minsk'))
|
|
|
|
CAST(N as DateTime64(9, 'Europe/Minsk'))
|
2019-11-10 13:23:02 +00:00
|
|
|
# Casting our test values to DateTime(12) will cause an overflow and hence will fail the test under UB sanitizer.
|
2022-03-11 23:45:26 +00:00
|
|
|
# CAST(N as DateTime64(12, 'Asia/Istanbul'))
|
2019-11-10 13:23:02 +00:00
|
|
|
# DateTime64(18) will always fail due to zero precision, but it is Ok to test here:
|
2022-03-11 23:45:26 +00:00
|
|
|
# CAST(N as DateTime64(18, 'Asia/Istanbul'))
|
|
|
|
formatDateTime(N, '%C %d %D %e %F %H %I %j %m %M %p %R %S %T %u %V %w %y %Y %%', 'Asia/Istanbul')
|
2019-09-07 14:22:22 +00:00
|
|
|
""".splitlines()
|
|
|
|
|
2020-04-17 13:26:44 +00:00
|
|
|
# Expanded later to cartesian product of all arguments, using format string.
|
2020-10-02 16:54:07 +00:00
|
|
|
extra_ops = [
|
2019-09-07 14:22:22 +00:00
|
|
|
# With same type:
|
|
|
|
(
|
2023-01-13 14:00:16 +00:00
|
|
|
["N {op} N"],
|
2019-09-07 14:22:22 +00:00
|
|
|
{
|
2023-01-13 14:00:16 +00:00
|
|
|
"op": [
|
|
|
|
"- ", # does not work, but should it?
|
|
|
|
"+ ", # does not work, but should it?
|
|
|
|
"!=",
|
|
|
|
"==", # equality and inequality supposed to take sub-second part in account
|
|
|
|
"< ",
|
|
|
|
"<=",
|
|
|
|
"> ",
|
|
|
|
">=",
|
2019-09-07 14:22:22 +00:00
|
|
|
]
|
2023-01-13 14:00:16 +00:00
|
|
|
},
|
2019-09-07 14:22:22 +00:00
|
|
|
),
|
|
|
|
# With other DateTime types:
|
|
|
|
(
|
2023-01-13 14:00:16 +00:00
|
|
|
["N {op} {arg}", "{arg} {op} N"],
|
2019-09-07 14:22:22 +00:00
|
|
|
{
|
2023-01-13 14:00:16 +00:00
|
|
|
"op": [
|
|
|
|
"-", # does not work, but should it?
|
|
|
|
"!=",
|
|
|
|
"==",
|
2019-09-07 14:22:22 +00:00
|
|
|
# these are naturally expected to work, but they don't:
|
2023-01-13 14:00:16 +00:00
|
|
|
"< ",
|
|
|
|
"<=",
|
|
|
|
"> ",
|
|
|
|
">=",
|
2019-09-07 14:22:22 +00:00
|
|
|
],
|
2023-01-13 14:00:16 +00:00
|
|
|
"arg": ["DT", "D", "DT64"],
|
|
|
|
},
|
2019-09-07 14:22:22 +00:00
|
|
|
),
|
|
|
|
# With arithmetic types
|
|
|
|
(
|
2023-01-13 14:00:16 +00:00
|
|
|
["N {op} {arg}", "{arg} {op} N"],
|
2019-09-07 14:22:22 +00:00
|
|
|
{
|
2023-01-13 14:00:16 +00:00
|
|
|
"op": ["+ ", "- ", "==", "!=", "< ", "<=", "> ", ">="],
|
|
|
|
"arg": [
|
|
|
|
"toUInt8(1)",
|
|
|
|
"toInt8(-1)",
|
|
|
|
"toUInt16(1)",
|
|
|
|
"toInt16(-1)",
|
|
|
|
"toUInt32(1)",
|
|
|
|
"toInt32(-1)",
|
|
|
|
"toUInt64(1)",
|
|
|
|
"toInt64(-1)",
|
2019-09-07 14:22:22 +00:00
|
|
|
],
|
|
|
|
},
|
|
|
|
),
|
|
|
|
]
|
|
|
|
|
|
|
|
# Expand extra_ops here
|
2019-10-04 13:49:34 +00:00
|
|
|
for funcs, args in extra_ops:
|
2020-10-02 16:54:07 +00:00
|
|
|
args_keys = list(args.keys())
|
|
|
|
for args_vals in itertools.product(*list(args.values())):
|
2019-10-08 04:57:31 +00:00
|
|
|
for func in funcs:
|
2020-10-02 16:54:07 +00:00
|
|
|
result_func = func.format(**dict(list(zip(args_keys, args_vals))))
|
2019-10-24 04:46:52 +00:00
|
|
|
FUNCTIONS.append(result_func)
|
2019-09-07 14:22:22 +00:00
|
|
|
|
2019-10-22 07:45:00 +00:00
|
|
|
# filter out empty lines and commented out lines
|
|
|
|
COMMENTED_OUT_LINE_RE = re.compile(r"^\s*#")
|
2023-01-13 14:00:16 +00:00
|
|
|
FUNCTIONS = list(
|
|
|
|
[f for f in FUNCTIONS if len(f) != 0 and COMMENTED_OUT_LINE_RE.match(f) == None]
|
|
|
|
)
|
|
|
|
TYPES = ["D", "DT", "DT64"]
|
|
|
|
|
2019-10-22 07:45:00 +00:00
|
|
|
|
2019-10-04 13:49:34 +00:00
|
|
|
def escape_string(s):
|
2020-10-02 16:54:07 +00:00
|
|
|
if sys.version_info[0] > 2:
|
2023-01-13 14:00:16 +00:00
|
|
|
return s.encode("unicode_escape").decode("utf-8").replace("'", "\\'")
|
2020-10-02 16:54:07 +00:00
|
|
|
else:
|
2023-01-13 14:00:16 +00:00
|
|
|
return s.encode("string-escape").decode("utf-8")
|
2019-10-03 02:07:36 +00:00
|
|
|
|
2019-10-24 04:46:52 +00:00
|
|
|
|
|
|
|
def execute_functions_for_types(functions, types):
|
2020-04-17 13:26:44 +00:00
|
|
|
# NOTE: use string.Template here to allow lines with missing keys, like type, e.g. SELECT CAST(toDateTime64(1234567890), 'DateTime64')
|
2019-10-24 04:46:52 +00:00
|
|
|
for func in functions:
|
2020-10-02 16:54:07 +00:00
|
|
|
print(("""SELECT 'SELECT {func}';""".format(func=escape_string(func))))
|
2019-10-24 04:46:52 +00:00
|
|
|
for dt in types:
|
|
|
|
prologue = "\
|
|
|
|
WITH \
|
|
|
|
toDateTime64('2019-09-16 19:20:11.234', 3, 'Europe/Minsk') as DT64, \
|
|
|
|
toDateTime('2019-09-16 19:20:11', 'Europe/Minsk') as DT, \
|
2023-01-13 14:00:16 +00:00
|
|
|
toDate('2019-09-16') as D, {X} as N".format(
|
|
|
|
X=dt
|
|
|
|
)
|
|
|
|
print(
|
|
|
|
(
|
|
|
|
"""{prologue} SELECT toTypeName(r), {func} as r FORMAT CSV;""".format(
|
|
|
|
prologue=prologue, func=func
|
|
|
|
)
|
|
|
|
)
|
|
|
|
)
|
2019-11-07 10:58:16 +00:00
|
|
|
print("""SELECT '------------------------------------------';""")
|
2019-10-24 04:46:52 +00:00
|
|
|
|
2023-01-13 14:00:16 +00:00
|
|
|
|
2019-10-24 04:46:52 +00:00
|
|
|
def main():
|
|
|
|
def parse_args():
|
|
|
|
parser = argparse.ArgumentParser()
|
2023-01-13 14:00:16 +00:00
|
|
|
parser.add_argument(
|
|
|
|
"--functions_re",
|
|
|
|
type=re.compile,
|
|
|
|
help="RE to enable functions",
|
|
|
|
default=None,
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"--types_re",
|
|
|
|
type=lambda s: re.compile("^(" + s + ")$"),
|
|
|
|
help="RE to enable types, supported types: " + ",".join(TYPES),
|
|
|
|
default=None,
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"--list_functions",
|
|
|
|
action="store_true",
|
|
|
|
help="List all functions to be tested and exit",
|
|
|
|
)
|
2019-10-24 04:46:52 +00:00
|
|
|
return parser.parse_args()
|
|
|
|
|
|
|
|
args = parse_args()
|
|
|
|
|
|
|
|
functions = FUNCTIONS
|
|
|
|
types = TYPES
|
|
|
|
|
|
|
|
if args.functions_re:
|
2020-10-02 16:54:07 +00:00
|
|
|
functions = list([f for f in functions if args.functions_re.search(f)])
|
2019-10-24 04:46:52 +00:00
|
|
|
if len(functions) == 0:
|
|
|
|
print("functions list is empty")
|
|
|
|
return -1
|
|
|
|
|
|
|
|
if args.types_re:
|
2020-10-02 16:54:07 +00:00
|
|
|
types = list([t for t in types if args.types_re.match(t)])
|
2019-10-24 04:46:52 +00:00
|
|
|
if len(types) == 0:
|
|
|
|
print("types list is empty")
|
|
|
|
return -1
|
|
|
|
|
|
|
|
if args.list_functions:
|
2020-10-02 16:54:07 +00:00
|
|
|
print(("\n".join(functions)))
|
2019-10-24 04:46:52 +00:00
|
|
|
return 0
|
|
|
|
|
|
|
|
execute_functions_for_types(functions, types)
|
|
|
|
|
2023-01-13 14:00:16 +00:00
|
|
|
|
|
|
|
if __name__ == "__main__":
|
2020-10-02 16:54:07 +00:00
|
|
|
exit(main())
|