Extensions
Documentation
- CHANGELOG.preamble
- pg_extra_time changelog / release notes
- CHANGELOG
- pg_extra_time changelog / release notes
- LICENCE
- LICENCE
README
Contents
pg_extension_name: pg_extra_time pg_extension_version: 2.0.0 pg_readme_generated_at: 2025-01-18 13:58:09.104687+00 pg_readme_version: 0.7.0
pg_extra_time
PostgreSQL extension
The pg_extra_time
PostgreSQL extension contains some date time functions and
operators that, according to the extension author, ought to be part of the
PostgreSQL standard distribution.
From tstzrange
/tsrange
, interval
and timestamp
/timestamptz
to seconds or days
pg_extra_time
has functions to convert various PostgreSQL datetime types
(tstzrange
/tsrange
, interval
, and timestamp
/timestamptz
) to the
number of seconds or days (in the range, interval or since the Unix epoch,
respectively).
| Function | Example |
| ———————————————————————– | –––––––––––––––––––––––––––––––––––––– |
| to_float(timestamptz)
| to_float('1970-01-01 00:00:00+0'::timestamptz) --> 0.0
|
| to_float(timestamp)
| to_float('1970-01-01 00:00:00+0'::timestamp) --> 0.0
|
| to_float(tstzrange)
| to_float('[2024-06-06 05:58:00,2024-06-06 06:00:10]'::tstzrange) --> 130.0
|
| to_float(tsrange)
| to_float('[2024-06-06 05:58:00,2024-06-06 06:00:10]'::tsrange) --> 130.0
|
| to_float(interval)
| to_float(interval '10 seconds 100 milliseconds') --> 10.100
|
| days(tstzrange)
| days('[2024-06-06,2024-06-08 06:00]'::tstzrange) --> 3.25
|
| whole_days(tstzrange)
| whole_days('[2024-06-06,2024-06-08 18:00]'::tstzrange) --> 2.5
|
| days(interval)
| days('10 days 12 hours'::interval) --> 10.5
|
| whole_days(interval)
| whole_days('10 days 20 hours 20 minutes'::interval) --> 10
|
Converting from these datetime types to seconds can be done by casting to
a double precision
/float
value as well.
| Cast | WITH FUNCTION
| Example |
| –––––––––– | ——————————————————— | —————————————————————– |
| timestamptz::float
| to_float(timestamptz)
| '1970-01-01 01:03:01+00'::timestamptz::float --> 3181.00
|
| timestamp::float
| to_float(timestamp)
| '1970-01-02 00:00:01+00'::timestamp::float --> 86401.0
|
| tstzrange::float
| to_float(tstzrange)
| '[epoch,1970-01-01T01:03:01+00]'::tstzrange::float --> 3181.00
|
| tsrange::float
| to_float(tsrange)
| '[epoch,1970-01-01T01:03:01+00]'::tsrange::float --> 3181.00
|
| interval::float
| to_float(interval)
| '1 day 1 sec 200 ms 200 us'::interval::float --> 86401.2002
|
Note that the
to_float(tstzrange)
/tstzrange::float
and
to_float(tsrange)
/tsrange::float
functions
will:
- return positive or negative infinity if there’s no upper bound and/or lower bound in the range, respectively;
- return 0 if the given range is altogether empty.
Casts to integer
are not provided, because the extension author doesn’t want
to impose an opinion on rounding to the users of pg_extra_time
. Note that
this distinctly differs from pg_extra_time
< 2.0.0, when casting a
tstzrange
or interval
to an integer meant the number of whole days (rounded
down) in that range or interval; this functionality is now available via the
whole_days(tstzrange)
and
whole_days(interval)
functions,
respectively.
Why cast to seconds and not days (or something else)?
PostgreSQL (as of version 16) doesn’t come with built-in casts of date-timey
types to float
s and/or integer
s. So the extension author had to make a
choice what unit to cast to.
Counting the number of seconds since the Unix epoch has become a prominent
means of representing timestamps on modern platforms. PostgreSQL itself
stores timestamp
values internally as the number of micro-seconds and not
counting from the Unix epoch (January 1 1970), but from January 1 2000. But,
as a testimony to the ubiquitousness of the Unix epoch, Postgres allows users
to easily convert timestamp
(tz
) values to and from a Unix timestamp:
extract('epoch' from timestamp)
,extract('epoch' from timestamptz)
,extract('epoch' from interval)
; andto_timestamp(double precision)
, respectively.
select extract('epoch' from timestamptz '2010-09-13 04:32:03+00'); --> 1284352323
select extract('epoch' from interval '10 seconds 100 milliseconds'); --> 10.100000
select to_timestamp(1284352323); --> '2010-09-13 04:32:03+00'
PostgreSQL even allows its users to simply input epoch
in a timestamp
string literal:
select 'epoch'::timestamptz; --> '1970-01-01 01:00:00+01'
So, indeed, although PostgreSQL internally counts from 2000-01-01, it considers as the epoch the One True Epoch: The Unix One.
In pg_extra_time
< 2.0, the extension author made the mistake of following
MSSQL’s lead in making tstzrange
and interval
values cast to days rather
than seconds when converting to integer
. This was mistaken:
- Even though MSSQL has forever come with the ability to convert
DATETIME
values toFLOAT
values (and, until recently, the avility to convertDATETIME
values toINTEGER
as well), MSSQL, as of its 2016 release, has no concept of intervals, nor of ranges. Thus, there were really no sensible semantics to copy there. CAST(DATETIME AS FLOAT)
in MSSQL returns the number of days since the MSSQL’s epoch (see themssql_epoch()
function), which is completely senseless out of the context of MSSQL. (Not thatpg_extra_time
supported castingtimestamp(tz)
to float at all, just to show that furthering consistency with MSSQL would make things even worse for modern users not used to Microsoft’s decrepit SQL Server.)
Why to_float()
and not float()
?
Indeed, if this wouldn’t cause a syntax error, the extension author would
have called these functions float(datetimeytype)
rather then
to_float(datetimeytype)
, because “it is
recommended that
you continue to follow this old convention of naming cast implementation
functions after the target data type.”
Backwards (in)compatibility of casting to integer
days
pg_extra_time
2.0.0 dropped the pre-2.0 semantics that casting a tstzrange
or interval
to an integer
would cast to a number of days (MSSQL-style)
rather than the number of seconds. (As discussed above, pg_extra_time
2.0
Installing & using this extension
When using this extension, please feel free to not actually use it as an extension and instead just copy-paste the precise function(s) and/or cast(s) that you need. To make copy-pasting bits and pieces easier, the extension author has tried his best not to succumb to the DRY disease and thus not reuse functions in other functions, only to save a few characters and introduce extra indirection (and sloth, because Postgres, as of version 16, supports inlining of SQL functions, but only one level deep). And DRY to save on bugs? Come on now, there’s a test procedure for each function.
Object reference
Routines
Function: current_timezone()
Returns a pg_timezone_names
record with the currently active timezone.
See the official Postgres documentation for the
pg_timezone_names
view for the precise record structure returned by this function.
Function return type: pg_timezone_names
Function attributes: STABLE
, PARALLEL SAFE
Function-local settings:
SET pg_readme.include_this_routine_definition TO true
CREATE OR REPLACE FUNCTION current_timezone()
RETURNS pg_timezone_names
LANGUAGE sql
STABLE PARALLEL SAFE
SET "pg_readme.include_this_routine_definition" TO 'true'
RETURN (SELECT ROW(pg_timezone_names.name, pg_timezone_names.abbrev, pg_timezone_names.utc_offset, pg_timezone_names.is_dst)::pg_timezone_names AS "row" FROM pg_timezone_names WHERE (pg_timezone_names.name = current_setting('timezone'::text)))
Function: date_part_parts (text, text, timestamp with time zone, text)
Extract the number of date parts that exist in the other given date part for the given date.
Use this function:
- if you want to know the number of days in the month for the month that the given date falls in;
- if you want to know the number of days in the year for the year that the given date falls in;
- if you need to be reminded that really every year has 12 months;
- if you want to know the number of hours in a day (which is not always 24);
- etc.
This function is primarily useful to avoid conditional nightmares in other date-time-related calculations.
The names of the date parts are those supported by the standard PostgreSQL
date_part()
and date_trunc()
functions.
See the test__date_part_parts()
routine
for example use of this function.
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | text
| |
| $2
| IN
| | text
| |
| $3
| IN
| | timestamp with time zone
| |
| $4
| IN
| | text
| current_setting('timezone'::text)
|
Function return type: integer
Function attributes: IMMUTABLE
, LEAKPROOF
, RETURNS NULL ON NULL INPUT
, PARALLEL SAFE
Function-local settings:
SET pg_readme.include_this_routine_definition TO true
CREATE OR REPLACE FUNCTION date_part_parts(text, text, timestamp with time zone, text DEFAULT current_setting('timezone'::text))
RETURNS integer
LANGUAGE sql
IMMUTABLE PARALLEL SAFE STRICT LEAKPROOF
SET "pg_readme.include_this_routine_definition" TO 'true'
RETURN (SELECT (count(*) - 1) FROM generate_series(date_trunc($1, $3), (date_trunc($1, $3) + (format('1 %s'::text, $1))::interval), (format('1 %s'::text, $2))::interval, $4) generate_series(generate_series))
Function: days (daterange)
Get the number of days in the given daterange
.
Given the simplicity of this function, you may (and probably should) just as
well use the upper($1) - lower($1) - 1
expression in your code directly.
Still, it is useful to have the function here in pg_extra_time
from a
librarian perspective, to remind us that, as stated under Discrete Range
Types
in the Postgres docs:
The built-in range type[…]
daterange
[…] use[s] a canonical form that includes the lower bound and excludes the upper bound; that is,[)
.
This means that [2022-06-01,2022-06-02]
::daterange is canonicalized into
[2022-06-01,2022-06-03)
. As a consequence:
- we never have to check
lower_inc(daterange)
orupper_inc(daterange)
(the former always and the latter never being true); and - we always have to subtract 1 from the difference between
upper(daterange)
andlower(daterange)
, because of that.
See the test__daterange_to_days()
procedure for usage examples.
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | daterange
| |
Function return type: integer
Function attributes: IMMUTABLE
, LEAKPROOF
, RETURNS NULL ON NULL INPUT
, PARALLEL SAFE
Function-local settings:
SET pg_readme.include_this_routine_definition TO true
CREATE OR REPLACE FUNCTION days(daterange)
RETURNS integer
LANGUAGE sql
IMMUTABLE PARALLEL SAFE STRICT LEAKPROOF
SET "pg_readme.include_this_routine_definition" TO 'true'
RETURN ((upper($1) - lower($1)) - 1)
Function: days (interval)
Extract the number of days, including fractions, from a given interval
value.
If you want to convert an interval
to (sub)seconds, use the Postgres-standard:
extract('epoch' from interval)
Or you can use the cast(interval as float)
functionality provided by this
extension (that uses the above technique, via the
to_float(interval)
function.
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | interval
| |
Function return type: double precision
Function attributes: IMMUTABLE
, LEAKPROOF
, RETURNS NULL ON NULL INPUT
, PARALLEL SAFE
Function-local settings:
SET pg_readme.include_this_routine_definition TO true
CREATE OR REPLACE FUNCTION days(interval)
RETURNS double precision
LANGUAGE sql
IMMUTABLE PARALLEL SAFE STRICT LEAKPROOF
SET "pg_readme.include_this_routine_definition" TO 'true'
RETURN (EXTRACT(epoch FROM $1) / (86400)::numeric)
Function: days (tstzrange)
Get the number of days, including the fraction of the remainder day, between the start and end of the given tstzrange
.
Inclusivity or exclusivity of the range is ignored. If you want to treat a
tstzrange
as discrete and have inclusivity/exclusivity interpreted as the
inclusion or not of a (sub)second, instead see the days(tstzrange, integer)
function.
For example usage of days(tstzrange)
, see the test procedure
test__days_from_tstzrange()
.
Note that when casting to float, it’s not this present function, but the
to_float(tstzrange)
` function that is used,
which puts the seconds rather than the days in front of the decimal point.
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | tstzrange
| |
Function return type: double precision
Function attributes: IMMUTABLE
, LEAKPROOF
, RETURNS NULL ON NULL INPUT
, PARALLEL SAFE
Function: days (tstzrange, double precision)
Get the number of days, including the fraction of the remainder day, between the start and end of the given tstzrange
, and treat range bounds as discrete values with a subtraction of the amount of (sub)seconds in the second argument for each exclusive bound.
The first argument is a tstzrange
value that will be treated as discrete
rather than continuous as per the precision (between 0 and 6) given in the
second argument. 1.0/10^$2
seconds are subtracted for each exlusive
tstzrange
bound.
Postgres’ built-in tstzrange
type has microsecond precision; thus you may
wish to pass 10.0^(-6)
to have exclusivity of the lower and upper bounds be
interpreted as the subtraction or not of a microsecond at either end.
If you have your own custom tstzrange*
domains based on timestamptz
subtypes of other precisions, you may want to pass p
as the second argument
instead.
For example usage of days(tstzrange, integer)
, see the test procedure
test__days_from_tstzrange()
.
Note that when casting to float, it’s not this present function, but the
to_float(tstzrange)
` function that is used,
which puts the seconds rather than the days in front of the decimal point.
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | tstzrange
| |
| $2
| IN
| | double precision
| |
Function return type: double precision
Function attributes: IMMUTABLE
, LEAKPROOF
, RETURNS NULL ON NULL INPUT
, PARALLEL SAFE
Function: each_subperiod (tstzrange, interval, integer)
Divide the given dividend$
into divisor$
-sized chunks.
The remainder is rounded:
- up, to a complete
divisor$
, ifround_remainder$ >= 1
; - down, discarding the remainder, if
round_remainder$ <= -1
; or - not at all and kept as the remainder, if
round_remainder = 0
.
See the test__each_subperiod
routine for
examples.
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| dividend$
| tstzrange
| |
| $2
| IN
| divisor$
| interval
| |
| $3
| IN
| round_remainder$
| integer
| 0
|
| $4
| TABLE
| quotient
| tstzrange
| |
Function return type: TABLE(quotient tstzrange)
Function attributes: IMMUTABLE
, LEAKPROOF
, PARALLEL SAFE
, ROWS 1000
Function: extract_days (interval)
Deprecated function, as of pg_extra_time
2.0—then reincarnated as the more correct whole_days(interval)
.
Contrary to this (deprecated) function,
whole_days(interval)
also does the right
thing with negative intervals.
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | interval
| |
Function return type: integer
Function attributes: IMMUTABLE
, LEAKPROOF
, RETURNS NULL ON NULL INPUT
, PARALLEL SAFE
Function-local settings:
SET pg_readme.include_this_routine_definition TO true
CREATE OR REPLACE FUNCTION extract_days(interval)
RETURNS integer
LANGUAGE plpgsql
IMMUTABLE PARALLEL SAFE STRICT LEAKPROOF
SET "pg_readme.include_this_routine_definition" TO 'true'
AS $function$
begin
raise warning using
errcode = '01P01' -- `deprecated_feature`; https://www.postgresql.org/docs/current/errcodes-appendix.html
,message = '`extract_days(interval)` has been deprected since `pg_extra_time` 2.0.'
,hint = 'Please use `whole_days(interval)` directly instead of `extract_days(interval)`.';
return floor(extract(epoch from $1) / 86400);
end;
$function$
Function: extract_days (tstzrange)
Deprecated function to get the number of whole days between the two points in the given range.
This function has a rather peculiar interpretation of inclusivity of the given
tstzrange
bounds: it interprets inclusivity and exclusivity of these bounds
as representing the absence or presence of that day. The sensible thing to do
is to either:
- ignore inclusivity altogether, since
tstzrange
s are continuous and therefore ambiguous as to where they should be cut off, which is what the newdays(tstzrange)
andwhole_days(tstzrange)
functions do; or - treat the range as discrete, and interpret inclusivity at some given level
of subsecond precision, which is what the new
days(tstzrange, float)
andwhole_days(tstzrange, float)
functions do.
Other reasons for deprecating this function (and its name) were:
- Postgres its built-in
extract()
function has different semantics—a good reason to dropextract_
from this function’s name; and - it needed to be distinguishable from the new
days(tstzrange)
function, which will also return fractions of a day in itsdouble precision
return value.
For example usage of extract_days(tstzrange)
, see the
test__extract_days_from_tstzrange()
procedure (but don’t use it).
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | tstzrange
| |
Function return type: integer
Function attributes: IMMUTABLE
, LEAKPROOF
, RETURNS NULL ON NULL INPUT
, PARALLEL SAFE
Function: extract_interval (tstzrange)
Deprecated function alias, as of pg_extra_time
2.0, then reincarnated as to_interval(tstzrange)
.
If still using this function, change your code to use
to_interval(tstzrange])
.
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | tstzrange
| |
Function return type: interval
Function attributes: IMMUTABLE
, LEAKPROOF
, RETURNS NULL ON NULL INPUT
, PARALLEL SAFE
Function-local settings:
SET pg_readme.include_this_routine_definition TO true
CREATE OR REPLACE FUNCTION extract_interval(tstzrange)
RETURNS interval
LANGUAGE plpgsql
IMMUTABLE PARALLEL SAFE STRICT LEAKPROOF
SET "pg_readme.include_this_routine_definition" TO 'true'
AS $function$
begin
raise warning using
errcode = '01P01' -- `deprecated_feature`; https://www.postgresql.org/docs/current/errcodes-appendix.html
,message = '`extract_interval(tstzrange)` has been deprected since `pg_extra_time` 2.0.'
,hint = 'Please use `to_interval(tstzrange)` directly instead of `extract_interval(tstzrange)`.';
return to_interval($1);
end;
$function$
Function: extract_interval (tstzrange, interval[])
Deprecated function alias, as of pg_extra_time
2.0, then reincarnated as to_interval(tstzrange, interval[])
.
If still using this function, change your code to use to_interval(tstzrange, interval[])
.
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | tstzrange
| |
| $2
| IN
| | interval[]
| |
Function return type: interval
Function attributes: IMMUTABLE
, LEAKPROOF
, RETURNS NULL ON NULL INPUT
, PARALLEL SAFE
Function-local settings:
SET pg_readme.include_this_routine_definition TO true
CREATE OR REPLACE FUNCTION extract_interval(tstzrange, interval[])
RETURNS interval
LANGUAGE plpgsql
IMMUTABLE PARALLEL SAFE STRICT LEAKPROOF
SET "pg_readme.include_this_routine_definition" TO 'true'
AS $function$
begin
raise warning using
errcode = '01P01' -- `deprecated_feature`; https://www.postgresql.org/docs/current/errcodes-appendix.html
,message = '`extract_interval(tstzrange, interval[])` has been deprected since `pg_extra_time` 2.0.'
,hint = 'Please use `to_interval(tstzrange, interval[])` directly instead of'
' `extract_interval(tstzrange, interval[])`.';
return to_interval($1, $2);
end;
$function$
Function: make_tsrange (timestamp without time zone, interval, text)
Build a tsrange
from a given timestamp from or until the given interval.
This function will do the right thing when confronted with negative intervals.
The function name is chosen for consistency with (some of) PostgreSQL built-in
date/time functions. I would have preferred to call it plainly tsrange()
,
but that would require users of this extensions to have to become explicit when
calling the existing tsrange(text)
constructor while relying on an explicit
cast of unknown
to text
.
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | timestamp without time zone
| |
| $2
| IN
| | interval
| |
| $3
| IN
| | text
| '[)'::text
|
Function return type: tsrange
Function attributes: IMMUTABLE
, LEAKPROOF
, RETURNS NULL ON NULL INPUT
, PARALLEL SAFE
Function: make_tstzrange (timestamp with time zone, interval, text)
Build a tstzrange
from a given timestamp from or until the given interval.
This function will do the right thing when confronted with negative intervals.
The function name is chosen for consistency with (some of) PostgreSQL built-in
date/time functions. I would have preferred to call it plainly tstzrange()
,
but that would require users of this extensions to have to become explicit when
calling the existing tsrange(text)
constructor while relying on an explicit
cast of unknown
to text
.
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | timestamp with time zone
| |
| $2
| IN
| | interval
| |
| $3
| IN
| | text
| '[)'::text
|
Function return type: tstzrange
Function attributes: IMMUTABLE
, LEAKPROOF
, RETURNS NULL ON NULL INPUT
, PARALLEL SAFE
Function: modulo (interval, interval)
As one would expect from a modulo operator, this function returns the remainder of the first given interval after dividing it into as many of the intervals given in the second argument as possible.
This function ignores the sign of the second argument. The sign of the first
argument is preserved. To take the absolute (intermediate) value of both
arguments, greatest(interval, -interval)
is used. According to some, this
might
be a too simplistic approach, but the extension author (Rowan) is of the
opinion that that’s okay in this context.
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | interval
| |
| $2
| IN
| | interval
| |
Function return type: interval
Function attributes: IMMUTABLE
, LEAKPROOF
, PARALLEL SAFE
Function-local settings:
SET pg_readme.include_this_routine_definition TO true
CREATE OR REPLACE FUNCTION modulo(interval, interval)
RETURNS interval
LANGUAGE sql
IMMUTABLE PARALLEL SAFE LEAKPROOF
SET "pg_readme.include_this_routine_definition" TO 'true'
RETURN ((sign(EXTRACT(epoch FROM $1)))::double precision * ((to_timestamp((0)::double precision) + GREATEST($1, (- $1))) - date_bin(GREATEST($2, (- $2)), (to_timestamp((0)::double precision) + GREATEST($1, (- $1))), to_timestamp((0)::double precision))))
Function: modulo (tstzrange, interval)
As you would expect from a modulo operator, this function returns the remainder of the given datetime range after dividing it in as many of the given whole intervals as possible.
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | tstzrange
| |
| $2
| IN
| | interval
| |
Function return type: interval
Function attributes: IMMUTABLE
, LEAKPROOF
, PARALLEL SAFE
Function-local settings:
SET pg_readme.include_this_routine_definition TO true
CREATE OR REPLACE FUNCTION modulo(tstzrange, interval)
RETURNS interval
LANGUAGE sql
IMMUTABLE PARALLEL SAFE LEAKPROOF
SET "pg_readme.include_this_routine_definition" TO 'true'
AS $function$
select
upper($1) - max(i)
from
generate_series(lower($1), upper($1), $2) as i
;
$function$
Function: mssql_epoch()
Constant function to retrieve Microsoft SQL Server’s epoch as a timestamptz
value.
Of course, you can (and, in most cases, should) also just copy-paste the timestamp literal from this function’s body.
Function return type: timestamp with time zone
Function attributes: IMMUTABLE
, LEAKPROOF
, PARALLEL SAFE
Function-local settings:
SET pg_readme.include_this_routine_definition TO true
CREATE OR REPLACE FUNCTION mssql_epoch()
RETURNS timestamp with time zone
LANGUAGE sql
IMMUTABLE PARALLEL SAFE LEAKPROOF
SET "pg_readme.include_this_routine_definition" TO 'true'
RETURN '1899-12-31 23:23:15-00:36:45'::timestamp with time zone
Function: nt_epoch()
The time epoch used in Windows 32/64, NTFS and COBOL.
Of course, you can (and, in most cases, should) also just copy-paste the timestamp literal from this function’s body.
Function return type: timestamp with time zone
Function attributes: IMMUTABLE
, LEAKPROOF
, PARALLEL SAFE
Function-local settings:
SET pg_readme.include_this_routine_definition TO true
CREATE OR REPLACE FUNCTION nt_epoch()
RETURNS timestamp with time zone
LANGUAGE sql
IMMUTABLE PARALLEL SAFE LEAKPROOF
SET "pg_readme.include_this_routine_definition" TO 'true'
RETURN '1600-12-31 23:23:15-00:36:45'::timestamp with time zone
Function: pg_extra_time_meta_pgxn()
Returns the JSON meta data that has to go into the META.json
file needed for PGXN—PostgreSQL Extension Network—packages.
The Makefile
includes a recipe to allow the developer to: make META.json
to
refresh the meta file with the function’s current output, including the
default_version
.
pg_extra_time
can indeed be found on PGXN: https://pgxn.org/dist/pg_readme/
Function return type: jsonb
Function attributes: STABLE
Function: pg_extra_time_readme()
Fire up the pg_readme
extension to generate a thorough README for this extension, based on the pg_catalog
and the COMMENT
objects found therein.
Function return type: text
Function-local settings:
SET pg_readme.include_view_definitions TO true
SET pg_readme.include_routine_definitions_like TO {test__%}
SET pg_readme.readme_url TO https://github.com/bigsmoke/pg_extra_time/blob/master/README.md
Procedure: test__date_part_parts()
Procedure-local settings:
SET pg_readme.include_this_routine_definition TO true
SET plpgsql.check_asserts TO true
CREATE OR REPLACE PROCEDURE test__date_part_parts()
LANGUAGE plpgsql
SET "pg_readme.include_this_routine_definition" TO 'true'
SET "plpgsql.check_asserts" TO 'true'
AS $procedure$
declare
_tz constant text := 'Europe/Amsterdam';
_dst_day constant timestamptz := ('2024-03-31 '||_tz)::timestamptz;
begin
perform set_config('timezone', _tz, true);
assert date_part_parts('year', 'days', make_date(2022,8,23)) = 365;
assert date_part_parts('year', 'days', make_date(2024,8,23)) = 366;
assert date_part_parts('year', 'months', make_date(1900,1,1)) = 12;
assert date_part_parts('month', 'days', make_date(2024,2,12)) = 29;
assert date_part_parts('day', 'hours', _dst_day - interval '1 day', _tz) = 24;
assert date_part_parts('day', 'minutes', _dst_day - interval '1 day', _tz) = 24 * 60;
assert date_part_parts('day', 'hours', _dst_day, _tz) = 23, format(
'The day that the switch to summertime happens in %s should have only 23 hours, not %s.'
,_tz
,date_part_parts('day', 'hours', _dst_day, _tz)
);
assert date_part_parts('day', 'minutes', _dst_day, _tz) = 23 * 60;
assert date_part_parts('day', 'hours', _dst_day + interval '1 day', _tz) = 24;
reset timezone;
end;
$procedure$
Procedure: test__daterange_to_days()
Procedure-local settings:
SET pg_readme.include_this_routine_definition TO true
SET plpgsql.check_asserts TO true
CREATE OR REPLACE PROCEDURE test__daterange_to_days()
LANGUAGE plpgsql
SET "pg_readme.include_this_routine_definition" TO 'true'
SET "plpgsql.check_asserts" TO 'true'
AS $procedure$
begin
assert days('[2022-06-01,2022-06-01)'::daterange) is null;
assert days('[2022-06-01,2022-06-02)'::daterange) = 0;
assert days('[2022-06-01,2022-06-22)'::daterange) = 20;
assert '[2022-06-01,2022-06-22]'::daterange::text = '[2022-06-01,2022-06-23)'::daterange::text,
'“The built-in range type[…] daterange […] use[s] a canonical form that includes the lower bound and excludes the upper bound; that is, `[)`”.';
assert days('[2022-06-01,2022-06-22]'::daterange) = 21;
-- Now that we tested its supporting function, we now only need to test that the cast is working at all.
assert '[2022-06-01,2022-06-03)'::daterange::int = 1;
assert cast('[2022-06-01,2022-06-03)'::daterange as integer) = 1;
end;
$procedure$
Procedure: test__days_from_interval()
Procedure-local settings:
SET pg_readme.include_this_routine_definition TO true
SET plpgsql.check_asserts TO true
CREATE OR REPLACE PROCEDURE test__days_from_interval()
LANGUAGE plpgsql
SET "pg_readme.include_this_routine_definition" TO 'true'
SET "plpgsql.check_asserts" TO 'true'
AS $procedure$
begin
assert days('0'::interval) = 0;
assert days('1 day'::interval) = 1;
assert days('12 hours'::interval) = 0.5;
assert days('1 day 12 hours'::interval) = 1.5;
assert days('3 days 6 hours'::interval) = 3.25;
assert days('720 minutes'::interval) = 0.5;
assert days('-1 day'::interval) = -1;
assert days('-1 day - 23 seconds'::interval) = -1 - (23.0/86400);
assert days('12 hours - 1 day'::interval) = -0.5;
end;
$procedure$
Procedure: test__each_subperiod()
Procedure-local settings:
SET pg_readme.include_this_routine_definition TO true
CREATE OR REPLACE PROCEDURE test__each_subperiod()
LANGUAGE plpgsql
SET "pg_readme.include_this_routine_definition" TO 'true'
AS $procedure$
begin
assert (
select
array_agg(quotient)
from
each_subperiod('[2023-01-01,2023-04-01)'::tstzrange, '1 month'::interval, 0)
) = '{
"[2023-01-01, 2023-02-01)",
"[2023-02-01, 2023-03-01)",
"[2023-03-01, 2023-04-01)"
}'::tstzrange[];
assert (
select
array_agg(quotient)
from
each_subperiod('[2023-01-01,2023-04-02)'::tstzrange, '1 month'::interval, 0)
) = '{
"[2023-01-01, 2023-02-01)",
"[2023-02-01, 2023-03-01)",
"[2023-03-01, 2023-04-01)",
"[2023-04-01, 2023-04-02)"
}'::tstzrange[];
assert (
select
array_agg(quotient)
from
each_subperiod('[2023-01-01,2023-04-02)'::tstzrange, '1 month'::interval, 1)
) = '{
"[2023-01-01, 2023-02-01)",
"[2023-02-01, 2023-03-01)",
"[2023-03-01, 2023-04-01)",
"[2023-04-01, 2023-05-01)"
}'::tstzrange[];
assert (
select
array_agg(quotient)
from
each_subperiod('[2023-01-01,2023-01-02)'::tstzrange, '1 month'::interval, 1)
) = '{"[2023-01-01, 2023-02-01)"}'::tstzrange[];
assert (
select
array_agg(quotient)
from
each_subperiod('[2023-01-01,2023-04-02)'::tstzrange, '1 month'::interval, -1)
) = '{
"[2023-01-01, 2023-02-01)",
"[2023-02-01, 2023-03-01)",
"[2023-03-01, 2023-04-01)"
}'::tstzrange[];
assert (
select
count(*)
from
each_subperiod('[2023-01-01,2023-01-31)'::tstzrange, '1 month'::interval, -1)
) = 0;
end;
$procedure$
Procedure: test__extract_days_from_interval()
Procedure-local settings:
SET pg_readme.include_this_routine_definition TO true
SET plpgsql.check_asserts TO true
CREATE OR REPLACE PROCEDURE test__extract_days_from_interval()
LANGUAGE plpgsql
SET "pg_readme.include_this_routine_definition" TO 'true'
SET "plpgsql.check_asserts" TO 'true'
AS $procedure$
begin
assert extract_days(interval '1 month') = 30;
assert extract_days(interval '1 week') = 7;
assert extract_days(interval '3 month 1 week 2 days') = 99;
assert extract_days(interval '1 day 23 hours') = 1;
end;
$procedure$
Procedure: test__extract_days_from_tstzrange()
Procedure-local settings:
SET pg_readme.include_this_routine_definition TO true
SET plpgsql.check_asserts TO true
CREATE OR REPLACE PROCEDURE test__extract_days_from_tstzrange()
LANGUAGE plpgsql
SET "pg_readme.include_this_routine_definition" TO 'true'
SET "plpgsql.check_asserts" TO 'true'
AS $procedure$
begin
assert extract_days('[2021-12-01,2022-01-01)'::tstzrange) = 31;
assert extract_days('[2021-12-01,2022-01-01]'::tstzrange) = 32;
assert extract_days('(2021-12-01,2022-01-01)'::tstzrange) = 30;
assert extract_days('(2021-12-01,2021-12-02)'::tstzrange) = 0;
assert extract_days('[2021-12-01,2021-12-02)'::tstzrange) = 1;
assert extract_days('[2022-01-01,2022-01-10 15:00)'::tstzrange) = 9;
assert extract_days('[2022-01-01,2022-01-01 23:00)'::tstzrange) = 0;
assert extract_days('[2022-01-01,2022-01-01 23:00]'::tstzrange) = 1;
end;
$procedure$
Procedure: test__extract_interval()
Procedure-local settings:
SET pg_readme.include_this_routine_definition TO true
SET plpgsql.check_asserts TO true
CREATE OR REPLACE PROCEDURE test__extract_interval()
LANGUAGE plpgsql
SET "pg_readme.include_this_routine_definition" TO 'true'
SET "plpgsql.check_asserts" TO 'true'
AS $procedure$
begin
assert extract_interval(
tstzrange('2022-07-22', '2022-09-23'),
array[interval '1 month', interval '1 hour']
) = interval '2 month 24 hour';
-- Internally in Pg, intervals are stored as months, days and microseconds.
assert interval '2 month 1 week 1 day' = interval '2 month 8 day'; -- See?
assert extract_interval(
tstzrange('1001-07-20', '2242-07-20')
) = interval '1 millennium 2 century 4 decade 1 year';
-- Internally in Pg, intervals are stored as months, days and microseconds.
assert interval '1 millennium 2 century 4 decade 1 year' = interval '1241 year';
-- Summer time started on March 27 in 2022.
assert extract_interval(
tstzrange('2022-03-01', '2022-05-8'),
array[interval '1 month', interval '1 day', interval '1 hour']
) = interval '2 month 1 week';
end;
$procedure$
Procedure: test__make_tsrange()
Procedure-local settings:
SET plpgsql.check_asserts TO true
SET pg_readme.include_this_routine_definition TO true
CREATE OR REPLACE PROCEDURE test__make_tsrange()
LANGUAGE plpgsql
SET "plpgsql.check_asserts" TO 'true'
SET "pg_readme.include_this_routine_definition" TO 'true'
AS $procedure$
begin
assert make_tsrange('2023-02-21 01:02'::timestamp, '1 day'::interval) = tsrange(
'2023-02-21 01:02'::timestamp
,'2023-02-22 01:02'::timestamp
);
assert make_tsrange('2023-02-21 01:02'::timestamp, '-1 month'::interval) = tsrange(
'2023-01-21 01:02'::timestamp
,'2023-02-21 01:02'::timestamp
);
end;
$procedure$
Procedure: test__make_tstzrange()
Procedure-local settings:
SET plpgsql.check_asserts TO true
SET pg_readme.include_this_routine_definition TO true
CREATE OR REPLACE PROCEDURE test__make_tstzrange()
LANGUAGE plpgsql
SET "plpgsql.check_asserts" TO 'true'
SET "pg_readme.include_this_routine_definition" TO 'true'
AS $procedure$
begin
assert make_tstzrange('2023-02-21 01:02'::timestamptz, '1 day'::interval) = tstzrange(
'2023-02-21 01:02'::timestamptz
,'2023-02-22 01:02'::timestamptz
);
assert make_tstzrange('2023-02-21 01:02'::timestamptz, '-1 month'::interval) = tstzrange(
'2023-01-21 01:02'::timestamptz
,'2023-02-21 01:02'::timestamptz
);
end;
$procedure$
Procedure: test__modulo__interval__interval()
Procedure-local settings:
SET pg_readme.include_this_routine_definition TO true
SET plpgsql.check_asserts TO true
CREATE OR REPLACE PROCEDURE test__modulo__interval__interval()
LANGUAGE plpgsql
SET "pg_readme.include_this_routine_definition" TO 'true'
SET "plpgsql.check_asserts" TO 'true'
AS $procedure$
begin
assert ('8 days 3 seconds'::interval % '2 days'::interval) = interval '3 seconds';
assert ('9 days 3 seconds'::interval % '2 days'::interval) = interval '1 day 3 seconds';
assert ('30 days'::interval % '10 days'::interval) = interval '0';
assert ('-10 days'::interval % '1 day'::interval) = interval '0',
format('%L ≠ %L', '-10 days'::interval % '1 day'::interval, '@ 00:00:00');
assert ('-10 days -4 hours'::interval % '1 day'::interval) = interval '-4 hours',
format('%L ≠ %L', '-10 days 4 hours'::interval % '1 day'::interval, '@ -4 hours');
assert ('28 days'::interval % '-7 days'::interval) = interval '0 seconds';
assert ('29 hours'::interval % '-7 hours'::interval) = interval '1 hour';
end;
$procedure$
Procedure: test__modulo__tsttzrange__interval()
Procedure-local settings:
SET pg_readme.include_this_routine_definition TO true
SET plpgsql.check_asserts TO true
CREATE OR REPLACE PROCEDURE test__modulo__tsttzrange__interval()
LANGUAGE plpgsql
SET "pg_readme.include_this_routine_definition" TO 'true'
SET "plpgsql.check_asserts" TO 'true'
AS $procedure$
begin
assert (tstzrange(make_date(2022,7,1), make_date(2022,8,2)) % interval '1 month') = interval '1 day';
end;
$procedure$
Procedure: test__to_interval()
Procedure-local settings:
SET pg_readme.include_this_routine_definition TO true
SET plpgsql.check_asserts TO true
CREATE OR REPLACE PROCEDURE test__to_interval()
LANGUAGE plpgsql
SET "pg_readme.include_this_routine_definition" TO 'true'
SET "plpgsql.check_asserts" TO 'true'
AS $procedure$
begin
assert to_interval(
tstzrange('2022-07-22', '2022-09-23'),
array[interval '1 month', interval '1 hour']
) = interval '2 month 24 hour';
assert tstzrange('2022-07-20', '2022-09-28')::interval = interval '2 month 1 week 1 day';
-- `WEEK` is support as input, but is always outputted as `7 DAYS`
assert interval '2 month 1 week 1 day' = interval '2 month 8 day'; -- See?
assert tstzrange('1001-07-20', '2002-07-20')::interval = interval '1 millennium 1 year';
assert to_interval(
tstzrange('1001-07-20', '2242-07-20')
) = interval '1 millennium 2 century 4 decade 1 year';
assert interval '1 millennium 2 century 4 decade 1 year' = interval '1241 year';
-- Summer time started on March 27 in 2022
assert to_interval(
tstzrange('2022-03-01', '2022-05-8'),
array[interval '1 month', interval '1 day', interval '1 hour']
) = interval '2 month 1 week';
end;
$procedure$
Procedure: test__tstzrange_to_days()
Procedure-local settings:
SET pg_readme.include_this_routine_definition TO true
SET plpgsql.check_asserts TO true
CREATE OR REPLACE PROCEDURE test__tstzrange_to_days()
LANGUAGE plpgsql
SET "pg_readme.include_this_routine_definition" TO 'true'
SET "plpgsql.check_asserts" TO 'true'
AS $procedure$
begin
assert days('[2021-12-01,2022-01-01)'::tstzrange) = 31.0;
assert days('[2021-12-01,)'::tstzrange) = '+infinity'::float;
assert days('[,)'::tstzrange) = '+infinity'::float;
assert days('[,2021-12-31)'::tstzrange) = '-infinity'::float;
-- Without a `range_bound_exclusivity_penalty_secs` argument, `days()` will ignore inclusivity/exclusivity
-- of both the `lower(tstzrange)` and the `upper(tstzrange)` bounds.
assert days('[2021-12-01,2021-12-02)'::tstzrange) = 1.0;
assert days('[2021-12-01,2021-12-02)'::tstzrange, 0) = 1.0;
assert days('[2021-12-01,2021-12-02)'::tstzrange) = days('[2021-12-01,2021-12-02]'::tstzrange);
assert days('[2021-12-01,2021-12-02)'::tstzrange, 0) = days('[2021-12-01,2021-12-02]'::tstzrange, 0);
assert days('[2021-12-01,2021-12-02)'::tstzrange) = days('(2021-12-01,2021-12-02]'::tstzrange);
assert days('[2021-12-01,2021-12-02)'::tstzrange, 0) = days('(2021-12-01,2021-12-02]'::tstzrange, 0);
assert days('[2022-01-01,2022-01-09 12:00)'::tstzrange) = 8.5;
assert days('[2022-01-01,2022-01-09 12:00)'::tstzrange, 0) = 8.5;
assert days('[2022-01-01,2022-02-01 06:00)'::tstzrange) = 31.25;
assert days('[2022-01-01,2022-02-01 06:00)'::tstzrange, 0) = 31.25;
assert days('(2021-12-01,2021-12-02)'::tstzrange, 1.0) = 1.0 - (2.0/86400)/10^0,
'1 second should have been subtracted for both exclusive `tstzrange` bounds.';
assert days('(2021-12-01,2021-12-02)'::tstzrange, 10^(-6)) = 1.0 - (2.0/86400)/10^6,
'1 microsecond should have been subtracted for both exclusive `tstzrange` bounds.';
assert days('[2021-12-01,2021-12-02)'::tstzrange, 10^(-6)) = 1.0 - (1.0/86400)/10^6,
'1 microsecond should be subtracted for either exclusive `tstzrange` bound.';
assert days('(2021-12-01,2021-12-02]'::tstzrange, 10^(-6)) = 1.0 - (1.0/86400)/10^6,
'1 microsecond should be subtracted for either exclusive `tstzrange` bound.';
assert days('[2021-12-01,2021-12-02]'::tstzrange, 10^(-6)) = 1.0,
'Nothing should be subtracted when both bounds are inclusive.';
declare
_p int;
begin
foreach _p in array array[0, 1, 2, 3, 4, 5, 6] loop
assert days('[2021-12-01,2021-12-02 00:00:00)'::tstzrange, 10^(-_p)) = 1.0 - (1.0/86400)/10^_p;
assert days('[2022-01-01,2022-01-09 12:00:00)'::tstzrange, 10^(-_p)) = 8.5 - (1.0/86400)/10^_p;
assert days('[2022-01-01,2022-02-01 06:00:00)'::tstzrange, 10^(-_p)) = 31.25 - (1.0/86400)/10^_p;
end loop;
end;
end;
$procedure$
Procedure: test__whole_days_from_interval()
Procedure-local settings:
SET pg_readme.include_this_routine_definition TO true
SET plpgsql.check_asserts TO true
CREATE OR REPLACE PROCEDURE test__whole_days_from_interval()
LANGUAGE plpgsql
SET "pg_readme.include_this_routine_definition" TO 'true'
SET "plpgsql.check_asserts" TO 'true'
AS $procedure$
begin
assert whole_days(interval '1 month') = 30;
assert whole_days(interval '1 week') = 7;
assert whole_days(interval '3 month 1 week 2 days') = 99;
assert whole_days(interval '1 day 23 hours') = 1;
assert whole_days(interval '1 day 24 hours') = 2;
assert whole_days('-1 day'::interval) = -1;
assert whole_days('-1 day -10 minutes'::interval) = -1, whole_days('-1 day 10 minutes'::interval);
assert whole_days('-1 day -23 hours -59 minutes'::interval) = -1;
assert whole_days('+12 hours - 1 day'::interval) = 0;
end;
$procedure$
Procedure: test__whole_days_from_tstzrange()
Procedure-local settings:
SET pg_readme.include_this_routine_definition TO true
SET plpgsql.check_asserts TO true
CREATE OR REPLACE PROCEDURE test__whole_days_from_tstzrange()
LANGUAGE plpgsql
SET "pg_readme.include_this_routine_definition" TO 'true'
SET "plpgsql.check_asserts" TO 'true'
AS $procedure$
declare
_tstzrange tstzrange;
_2nd_arg float;
_expected_int int;
begin
-- First, we test the truncating without awareness of bound in/exclusivity.
for _tstzrange, _expected_int in
select
tstzrange(lower_bound::timestamptz, upper_bound::timestamptz, bound_type)
,expected_int
from (
values
('2024-06-15 01:10', '2024-06-16 01:10', 1)
,('2021-12-01', '2022-01-01', 31)
,('2022-01-01', '2022-01-10', 9)
,('2022-01-01', '2022-01-01', 0)
,('2022-01-01', '2022-01-01 23:59:59.999999', 0)
) as without_bound_types (lower_bound, upper_bound, expected_int)
cross join (
values ('[]'), ('[)'), ('(]'), ('()')
) as bound_types (bound_type)
loop
assert whole_days(_tstzrange) = _expected_int,
format('whole_days(%L::tstzrange) ≠ %s', _tztzrange, _expected_int);
assert whole_days(_tstzrange, 0.0) = _expected_int,
format('whole_days(%L::tstzrange, %s) ≠ %s', _tztzrange, 0.0, _expected_int);
end loop;
-- Then we test treating the ranges as discrete ranges with a certain precision.
assert whole_days('[2022-01-01, 2022-01-01 23:59:59.999999]'::tstzrange, 1.0) = 0;
assert whole_days('[2022-01-01, 2022-01-02]'::tstzrange, 1.0) = 1;
assert whole_days('[2022-01-01, 2022-01-02)'::tstzrange, 1.0) = 0;
assert whole_days('[2022-01-01, 2022-01-02 00:00:00.000001]'::tstzrange, 0.000001) = 1;
assert whole_days('[2022-01-01, 2022-01-02 00:00:00.000001)'::tstzrange, 0.000001) = 1;
assert whole_days('[2022-01-01, 2022-01-02 00:00:00.000001)'::tstzrange, 0.000010) = 0;
assert whole_days('[2022-01-01, 2022-01-02 00:00:00.000010)'::tstzrange, 0.000010) = 1;
-- PostgreSQL integers do not support positive or negative infinity; `null` will have to do.
assert whole_days('[,)'::tstzrange) is null;
assert whole_days('[2024-06-15,)'::tstzrange) is null;
assert whole_days('[,2024-06-15)'::tstzrange) is null;
end;
$procedure$
Function: to_float (interval)
Convert a given interval
to the number of seconds, including fractions.
You probably shouldn’t use this function and just use extract('epoch' from interval)
instead. This function primarily exists to power cast(interval as float)
(and to remind the reader that extract('epoch' from interval)
is
always already possible in Postgres.
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | interval
| |
Function return type: double precision
Function attributes: IMMUTABLE
, LEAKPROOF
, RETURNS NULL ON NULL INPUT
, PARALLEL SAFE
Function-local settings:
SET pg_readme.include_this_routine_definition TO true
CREATE OR REPLACE FUNCTION to_float(interval)
RETURNS double precision
LANGUAGE sql
IMMUTABLE PARALLEL SAFE STRICT LEAKPROOF
SET "pg_readme.include_this_routine_definition" TO 'true'
RETURN EXTRACT(epoch FROM $1)
Function: to_float (timestamp without time zone)
Get the given timestamp
as the number of seconds (down to microseconds) elapsed since the Unix epoch.
Don’t use this function. Just use extract('epoch' from timestamp)
directly
instead. This function solely serves as a support function for the
timestamp::float
cast.
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | timestamp without time zone
| |
Function return type: double precision
Function attributes: IMMUTABLE
, LEAKPROOF
, RETURNS NULL ON NULL INPUT
, PARALLEL SAFE
Function-local settings:
SET pg_readme.include_this_routine_definition TO true
CREATE OR REPLACE FUNCTION to_float(timestamp without time zone)
RETURNS double precision
LANGUAGE sql
IMMUTABLE PARALLEL SAFE STRICT LEAKPROOF
SET "pg_readme.include_this_routine_definition" TO 'true'
RETURN EXTRACT(epoch FROM $1)
Function: to_float (timestamp with time zone)
Get the given timestamp
as the number of seconds (down to microseconds) elapsed since the Unix epoch.
Don’t use this function. Just use extract('epoch' from timestamptz)
directly
instead. This function solely serves as a support function for the
timestamptz::float
cast.
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | timestamp with time zone
| |
Function return type: double precision
Function attributes: IMMUTABLE
, LEAKPROOF
, RETURNS NULL ON NULL INPUT
, PARALLEL SAFE
Function-local settings:
SET pg_readme.include_this_routine_definition TO true
CREATE OR REPLACE FUNCTION to_float(timestamp with time zone)
RETURNS double precision
LANGUAGE sql
IMMUTABLE PARALLEL SAFE STRICT LEAKPROOF
SET "pg_readme.include_this_routine_definition" TO 'true'
RETURN EXTRACT(epoch FROM $1)
Function: to_float (tsrange)
Get the number of (fractional) seconds between the two points in the given tsrange
.
-
When the given range is empty, that is interpreted as 0 seconds.
-
When the upper point in the given range is empty, the range is interpreted as an infinite number of seconds.
-
When the lower point is absent, the range instead is interpreted as a infinite negative number of seconds.
-
When both points are present, the distance in seconds between the two points is returned.
Because the points in
tsrange
are of typetimestamp without time zone
/timestamp
, which has microsecond precision, a microsecond is subtracted for either or both the lower and upper bound when it is exclusive.
If you have a custom timestamp(p)
range type, you will want to overload this
function and subtract a fractional of a second corresponding to that precision
instead.
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | tsrange
| |
Function return type: double precision
Function attributes: IMMUTABLE
, LEAKPROOF
, RETURNS NULL ON NULL INPUT
, PARALLEL SAFE
Function-local settings:
SET pg_readme.include_this_routine_definition TO true
CREATE OR REPLACE FUNCTION to_float(tsrange)
RETURNS double precision
LANGUAGE sql
IMMUTABLE PARALLEL SAFE STRICT LEAKPROOF
SET "pg_readme.include_this_routine_definition" TO 'true'
RETURN CASE WHEN isempty($1) THEN (0)::numeric WHEN (lower_inf($1) AND (NOT upper_inf($1))) THEN '-Infinity'::numeric WHEN upper_inf($1) THEN 'Infinity'::numeric ELSE ((EXTRACT(epoch FROM (upper($1) - lower($1))) - (0.000001 * (((NOT upper_inc($1)))::integer)::numeric)) - (0.000001 * (((NOT lower_inc($1)))::integer)::numeric)) END
Function: to_float (tstzrange)
Get the number of (fractional) seconds between the two points in the given tstzrange
.
-
When the given range is empty, that is interpreted as 0 seconds.
-
When the upper point in the given range is empty, the range is interpreted as an infinite number of seconds.
-
When the lower point is absent, the range instead is interpreted as a infinite negative number of seconds.
-
When both points are present, the distance in seconds between the two points is returned.
Because the points in
tstzrange
are of typetimestamp with time zone
/timestamptz
, which has microsecond precision, a microsecond is subtracted for either or both the lower and upper bound when it is exclusive.
For example usage of to_float(tstzrange)
, see the test procedure
test__tstzrange_to_seconds()
.
If you have a custom timestamptz(p)
range type, you will want to overload this
function and subtract a fractional of a second corresponding to that precision
instead.
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | tstzrange
| |
Function return type: double precision
Function attributes: IMMUTABLE
, LEAKPROOF
, RETURNS NULL ON NULL INPUT
, PARALLEL SAFE
Function: to_interval (tstzrange)
Extract an interval from a datetime range, starting with the largest interval unit possible, and down to the microsecond.
“Largest interval unit possible” must be taken with a grain of salt, because
this function ignores the units that are never part of the interval text
output.
Those are:
- all units greater than a
year
—decade
,century
andmillennium
, - as well as
week
.
Note that, even if you call to_interval(tstzrange, interval[])
directly instead
and include those excluded intervals in the given interval
array, the result
will be the same as if you did include these units, because “[i]nternally,
interval values are stored as three integral fields: months, days, and
microseconds.”
The extension author would have preferred to call this function simply
interval
, thereby conforming to the recommendationin the CREATE CAST
documentation,
if not for the fact that that produced a “syntax error”:
While not required, it is recommended that you continue to follow this old convention of naming cast implementation functions after the target data type. Many users are used to being able to cast data types using a function-style notation, that is
typename(x)
. This notation is in fact nothing more nor less than a call of the cast implementation function; it is not specially treated as a cast. If your conversion functions are not named to support this convention then you will have surprised users. Since PostgreSQL allows overloading of the same function name with different argument types, there is no difficulty in having multiple conversion functions from different types that all use the target type’s name.
In the author’s opinion, this is another good reason to have this function in
Postgres’ core. Or should it then be called fit_interval()
instead?
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | tstzrange
| |
Function return type: interval
Function attributes: IMMUTABLE
, LEAKPROOF
, RETURNS NULL ON NULL INPUT
, PARALLEL SAFE
Function-local settings:
SET pg_readme.include_this_routine_definition TO true
CREATE OR REPLACE FUNCTION to_interval(tstzrange)
RETURNS interval
LANGUAGE sql
IMMUTABLE PARALLEL SAFE STRICT LEAKPROOF
SET "pg_readme.include_this_routine_definition" TO 'true'
RETURN to_interval($1, ARRAY['1 mon'::interval, '1 day'::interval, '00:00:00.000001'::interval])
Function: to_interval (tstzrange, interval[])
Divide the datetime range given in the first argument over the given interval
(s) in the second argument.
The function starts with as many of the biggest units given as fit in the datetime range, then tries the next-biggest unit with the remainder, etc.
As of version 2.0.0, this function does not (yet) order the interval[]
array
by decreasing interval
size itself. Therefore, the interval[]
array must
be passed greatest-first for this function to work correctly.
This function simply discards the remainder of the range that does not fit in
the smallest given interval
in the array of valid intervals. Thus, rounding
is always down and never up. If you are specifically interested in the
remainder, see the tstzrange % interval
operator and its supporting
modulo(tstzrange, interval)
function.
Note that to_interval(tstzrange, interval[])
has very distinct semantics from
simply subtracting the lower(tstzrange)
from the upper(tstzrange)
and then
truncating using date_trunc('<field>' from interval)
or cast(interval as interval <fields>)
, because when truncating you will simply use any subfields
with units smaller than the included intervals, even if these subfields have
large enough quantities to fit in the bigger, included fields:
select '3 months 12 days 70 minutes'::interval day; --> '3 mons 12 days'
See the test__to_interval()
procedure for examples.
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | tstzrange
| |
| $2
| IN
| | interval[]
| |
Function return type: interval
Function attributes: IMMUTABLE
, LEAKPROOF
, RETURNS NULL ON NULL INPUT
, PARALLEL SAFE
Function: unix_epoch()
Constant function to retrieve the Unix epoch as a timestamptz
value.
Don’t use this function. Directly use 'epoch'::timestamptz
instead. Yeah,
this function just exists to remind the extension author of Postgres’
understanding of the 'epoch'
timestamp
(tz
) input.
Function return type: timestamp with time zone
Function attributes: IMMUTABLE
, LEAKPROOF
, PARALLEL SAFE
Function-local settings:
SET pg_readme.include_this_routine_definition TO true
CREATE OR REPLACE FUNCTION unix_epoch()
RETURNS timestamp with time zone
LANGUAGE sql
IMMUTABLE PARALLEL SAFE LEAKPROOF
SET "pg_readme.include_this_routine_definition" TO 'true'
RETURN '1970-01-01 01:00:00+01'::timestamp with time zone
Function: whole_days (interval)
Get the number of whole days (rounded down) present in a given interval
value.
If you want to convert an interval
to (sub)seconds, use the Postgres-standard:
extract('epoch' from interval)`
For example usage of this function, see the
test__whole_days_from_interval()
procedure.
Please don’t be clever (and overly DRY) and make this function reuse
days(interval)
:
- The functions in this extensions are meant to be copy-pastable apart from each other independently.
- Individual SQL functions can be inlined by the planner, but not nested functions.
- These functions are way to simple to justify reuse.
- And we have test cases.
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | interval
| |
Function return type: integer
Function attributes: IMMUTABLE
, LEAKPROOF
, RETURNS NULL ON NULL INPUT
, PARALLEL SAFE
Function-local settings:
SET pg_readme.include_this_routine_definition TO true
CREATE OR REPLACE FUNCTION whole_days(interval)
RETURNS integer
LANGUAGE sql
IMMUTABLE PARALLEL SAFE STRICT LEAKPROOF
SET "pg_readme.include_this_routine_definition" TO 'true'
RETURN (sign(EXTRACT(epoch FROM $1)) * floor(abs((EXTRACT(epoch FROM $1) / (86400)::numeric))))
Function: whole_days (tstzrange)
Get the number of whole days that fit in the given tstzrange
value.
Zero is returned when the given tstzrange
is empty.
Because Postgres doesn’t support infinity values for integers, null
is
returned when either end of the range is infinity; thus no distinction is made
between positive or negative infinity. In fact, negative infinity would be an
impossibility anyway, because a range’s lower bound must be less than or
equal to its upper bound, and null
is not a too shabby choice to denote
infinity in this case, because null
is similarly used to indicate infinity by
various built-in range functions in Postgres
The days(tstzrange)
function, which returns a
float value rather than an int can be used instead if you want explicit support
for infinity (which Postgres does have for floating point values, contrary to
its lack of support for postive and negative integer infinity.)
This function ignores inclusivity/exclusivity of both lower and upper bounds
of the given range. Practically, this means that when you naively construct
a tstzrange
r
with an upper bound that is d
days later than it lower
bound, whole_days(r)
will return that same number of days d
:
do $$
declare
d int := 5;
r tstzrange := tstzrange(now(), now() + format('%s days', d)::interval);
begin
assert whole_days(r) = d;
end;
$$;
If you want to treat the tstzrange
value as discrete rather than continuous
at a given level of subsecond precision, see the whole_days(tstzrange, int)
function.
For example usage of whole_days(tstzrange)
, see the test procedure
test__whole_days_from_tstzrange()
.
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | tstzrange
| |
Function return type: integer
Function attributes: IMMUTABLE
, LEAKPROOF
, RETURNS NULL ON NULL INPUT
, PARALLEL SAFE
Function: whole_days (tstzrange, double precision)
Get the number of whole days that fit in the given tstzrange
value, subtracting the amount of double precision
seconds for bounds that are exclusive.
When its second argument is zero (0
), this function behaves identical to
whole_days(tstzrange)
, blindly truncating
without regard for bound exclusivity.
Zero is returned when the given tstzrange
is empty.
Because Postgres doesn’t support infinity values for integers, null
is
returned when either end of the range is infinity; thus no distinction is made
between positive or negative infinity. In fact, negative infinity would be an
impossibility anyway, because a range’s lower bound must be less than or
equal to its upper bound, and null
is not a too shabby choice to denote
infinity in this case, because null
is similarly used to indicate infinity by
various built-in range functions in Postgres
The days(tstzrange, float)
function, which returns a float value rather than an int can be used instead if
you want explicit support for infinity (which Postgres does have for floating
point values, contrary to its lack of support for postive and negative
integer infinity.)
For example usage of whole_days(tstzrange, float)
, see the test procedure
test__whole_days_from_tstzrange()
.
Function arguments:
| Arg. # | Arg. mode | Argument name | Argument type | Default expression |
| —— | ––––– | —————————————————————– | –––––––––––––––––––––––––––––––––– | —————–– |
| $1
| IN
| | tstzrange
| |
| $2
| IN
| | double precision
| |
Function return type: integer
Function attributes: IMMUTABLE
, LEAKPROOF
, RETURNS NULL ON NULL INPUT
, PARALLEL SAFE
Extension origins
pg_extra_time
was developed to simplify quite a bit of code in the PostgreSQL
backend of the FlashMQ MQTT hosting platform,
especially for financial calculations regarding subscription durations, etc..
Datetime calculations are notoriously easy to get wrong, and therefore better
to isolate and test well rather than mix into the business logic on an ad hoc
basis.
Extension author(s)
-
Rowan Rodrik van der Molen—the original (and so far only) author of
pg_extra_time
—identifies more as a restorative farmer, permaculture writer and reanimist than as a techologist. Nevertheless, computer technology has remained stubbornly intertwined with his life, the trauma of which he tries to process by writing the book on Why Programming Still Sucks (@ysosuckysoft). As of 2023, he is applying his painfully earned IT wisdom to a robust MQTT SaaS service, and he does so alternatingly:- from within a permaculture project in central Portugal;
- and his beautiful holiday home for rent in the forests of Drenthe, where from his work place he looks out over his lush ecological garden and a private heather field.
His day to day musings on technology he usually slaps onto his blog.
Colophon
This README.md
for the pg_extra_time
extension was automatically generated using the pg_readme
PostgreSQL extension.