mirror of
https://github.com/stedolan/jq.git
synced 2024-05-11 05:55:39 +00:00
Add more date builtins
This commit is contained in:
186
builtin.c
186
builtin.c
@@ -1,4 +1,7 @@
|
||||
#define _BSD_SOURCE
|
||||
#define _XOPEN_SOURCE
|
||||
#include <sys/time.h>
|
||||
#include <alloca.h>
|
||||
#include <assert.h>
|
||||
#include <ctype.h>
|
||||
#include <limits.h>
|
||||
@@ -919,12 +922,24 @@ static jv f_stderr(jq_state *jq, jv input) {
|
||||
return input;
|
||||
}
|
||||
|
||||
static jv tm2jv(struct tm *tm) {
|
||||
return JV_ARRAY(jv_number(tm->tm_year + 1900),
|
||||
jv_number(tm->tm_mon),
|
||||
jv_number(tm->tm_mday),
|
||||
jv_number(tm->tm_hour),
|
||||
jv_number(tm->tm_min),
|
||||
jv_number(tm->tm_sec),
|
||||
jv_number(tm->tm_wday),
|
||||
jv_number(tm->tm_yday));
|
||||
}
|
||||
|
||||
#ifdef HAVE_STRPTIME
|
||||
static jv f_strptime(jq_state *jq, jv a, jv b) {
|
||||
if (jv_get_kind(a) != JV_KIND_STRING || jv_get_kind(b) != JV_KIND_STRING)
|
||||
return jv_invalid_with_msg(jv_string("strptime/2 requires string inputs and arguments"));
|
||||
return jv_invalid_with_msg(jv_string("strptime/1 requires string inputs and arguments"));
|
||||
|
||||
struct tm tm;
|
||||
memset(&tm, 0, sizeof(tm));
|
||||
const char *input = jv_string_value(a);
|
||||
const char *fmt = jv_string_value(b);
|
||||
const char *end = strptime(input, fmt, &tm);
|
||||
@@ -937,56 +952,158 @@ static jv f_strptime(jq_state *jq, jv a, jv b) {
|
||||
}
|
||||
jv_free(a);
|
||||
jv_free(b);
|
||||
jv r = JV_ARRAY(jv_number(tm.tm_year + 1900),
|
||||
jv_number(tm.tm_mon),
|
||||
jv_number(tm.tm_mday),
|
||||
jv_number(tm.tm_hour),
|
||||
jv_number(tm.tm_min),
|
||||
jv_number(tm.tm_sec),
|
||||
jv_number(tm.tm_wday),
|
||||
jv_number(tm.tm_yday));
|
||||
jv r = tm2jv(&tm);
|
||||
if (*end != '\0')
|
||||
r = jv_array_append(r, jv_string(end));
|
||||
return r;
|
||||
}
|
||||
#else
|
||||
static jv f_strptime(jq_state *jq, jv a, jv b) {
|
||||
return jv_invalid_with_msg(jv_string("strptime/2 not implemented on this platform"));
|
||||
jv_free(a);
|
||||
jv_free(b);
|
||||
return jv_invalid_with_msg(jv_string("strptime/1 not implemented on this platform"));
|
||||
}
|
||||
#endif
|
||||
|
||||
#define TO_TM_FIELD(t, j, i, k) \
|
||||
#define TO_TM_FIELD(t, j, i) \
|
||||
do { \
|
||||
jv n = jv_array_get(jv_copy(j), (i)); \
|
||||
if (jv_get_kind(n) != (k)) \
|
||||
return jv_invalid_with_msg(jv_string("mktime() requires a 'gmtime' input (an array inputs of 8 numeric values)")); \
|
||||
if (jv_get_kind(n) != (JV_KIND_NUMBER)) \
|
||||
return 0; \
|
||||
t = jv_number_value(n); \
|
||||
jv_free(n); \
|
||||
} while (0)
|
||||
|
||||
static jv f_mktime(jq_state *jq, jv a) {
|
||||
if (jv_get_kind(a) != JV_KIND_ARRAY)
|
||||
return jv_invalid_with_msg(jv_string("mktime() requires array inputs"));
|
||||
if (jv_array_length(jv_copy(a)) < 6)
|
||||
return jv_invalid_with_msg(jv_string("mktime() requires a 'gmtime' input (an array inputs of 8 numeric values)"));
|
||||
struct tm tm;
|
||||
memset(&tm, 0, sizeof(tm));
|
||||
TO_TM_FIELD(tm.tm_year, a, 0, JV_KIND_NUMBER);
|
||||
TO_TM_FIELD(tm.tm_mon, a, 1, JV_KIND_NUMBER);
|
||||
TO_TM_FIELD(tm.tm_mday, a, 2, JV_KIND_NUMBER);
|
||||
TO_TM_FIELD(tm.tm_hour, a, 3, JV_KIND_NUMBER);
|
||||
TO_TM_FIELD(tm.tm_min, a, 4, JV_KIND_NUMBER);
|
||||
TO_TM_FIELD(tm.tm_sec, a, 5, JV_KIND_NUMBER);
|
||||
tm.tm_year -= 1900;
|
||||
static int jv2tm(jv a, struct tm *tm) {
|
||||
memset(tm, 0, sizeof(*tm));
|
||||
TO_TM_FIELD(tm->tm_year, a, 0);
|
||||
TO_TM_FIELD(tm->tm_mon, a, 1);
|
||||
TO_TM_FIELD(tm->tm_mday, a, 2);
|
||||
TO_TM_FIELD(tm->tm_hour, a, 3);
|
||||
TO_TM_FIELD(tm->tm_min, a, 4);
|
||||
TO_TM_FIELD(tm->tm_sec, a, 5);
|
||||
tm->tm_year -= 1900;
|
||||
jv_free(a);
|
||||
time_t t = mktime(&tm);
|
||||
if (t == (time_t)-1)
|
||||
return jv_invalid_with_msg(jv_string("invalid gmtime representation"));
|
||||
return jv_number(t);
|
||||
return 1;
|
||||
}
|
||||
|
||||
#undef TO_TM_FIELD
|
||||
|
||||
static jv f_mktime(jq_state *jq, jv a) {
|
||||
if (jv_get_kind(a) != JV_KIND_ARRAY)
|
||||
return jv_invalid_with_msg(jv_string("mktime requires array inputs"));
|
||||
if (jv_array_length(jv_copy(a)) < 6)
|
||||
return jv_invalid_with_msg(jv_string("mktime requires parsed datetime inputs")); \
|
||||
struct tm tm;
|
||||
if (!jv2tm(a, &tm))
|
||||
return jv_invalid_with_msg(jv_string("mktime requires parsed datetime inputs")); \
|
||||
/*
|
||||
* mktime() has side-effects and anyways, returns time in the local
|
||||
* timezone, not UTC. We want timegm(), which isn't standard.
|
||||
*
|
||||
* To make things worse, mktime() tells you what the timezone
|
||||
* adjustment is, but you have to #define _BSD_SOURCE to get this
|
||||
* field of struct tm on some systems.
|
||||
*
|
||||
* This is all to blame on POSIX, of course.
|
||||
*/
|
||||
#ifdef HAVE_TIMEGM
|
||||
time_t t = timegm(&tm);
|
||||
if (t == (time_t)-1)
|
||||
return jv_invalid_with_msg(jv_string("invalid gmtime representation"));
|
||||
return jv_number(t);
|
||||
#else /* HAVE_TIMEGM */
|
||||
time_t t = mktime(&tm);
|
||||
if (t == (time_t)-1)
|
||||
return jv_invalid_with_msg(jv_string("invalid gmtime representation"));
|
||||
#ifdef HAVE_TM_TM_GMT_OFF
|
||||
return jv_number(t + tm.tm_gmtoff);
|
||||
#elif defined(HAVE_TM_TM_GMT_OFF)
|
||||
return jv_number(t + tm.__tm_gmtoff);
|
||||
#endif
|
||||
#endif /* HAVE_TIMEGM */
|
||||
}
|
||||
|
||||
#ifdef HAVE_GMTIME_R
|
||||
static jv f_gmtime(jq_state *jq, jv a) {
|
||||
if (jv_get_kind(a) != JV_KIND_NUMBER)
|
||||
return jv_invalid_with_msg(jv_string("gmtime() requires numeric inputs"));
|
||||
struct tm tm, *tmp;
|
||||
memset(&tm, 0, sizeof(tm));
|
||||
double fsecs = jv_number_value(a);
|
||||
time_t secs = fsecs;
|
||||
jv_free(a);
|
||||
tmp = gmtime_r(&secs, &tm);
|
||||
if (tmp == NULL)
|
||||
return jv_invalid_with_msg(jv_string("errror converting number of seconds since epoch to datetime"));
|
||||
a = tm2jv(tmp);
|
||||
return jv_array_set(a, 5, jv_number(jv_number_value(jv_array_get(jv_copy(a), 5)) + (fsecs - floor(fsecs))));
|
||||
}
|
||||
#elif defined HAVE_GMTIME
|
||||
static jv f_gmtime(jq_state *jq, jv a) {
|
||||
if (jv_get_kind(a) != JV_KIND_NUMBER)
|
||||
return jv_invalid_with_msg(jv_string("gmtime requires numeric inputs"));
|
||||
struct tm *tmp;
|
||||
memset(&tm, 0, sizeof(tm));
|
||||
double fsecs = jv_number_value(a);
|
||||
time_t secs = fsecs;
|
||||
jv_free(a);
|
||||
tmp = gmtime(&secs);
|
||||
if (tmp == NULL)
|
||||
return jv_invalid_with_msg(jv_string("errror converting number of seconds since epoch to datetime"));
|
||||
a = tm2jv(tmp);
|
||||
return jv_array_set(a, 5, jv_number(jv_number_value(jv_array_get(jv_copy(a), 5)) + (fsecs - floor(fsecs))));
|
||||
}
|
||||
#else
|
||||
static jv f_gmtime(jq_state *jq, jv a) {
|
||||
jv_free(a);
|
||||
return jv_invalid_with_msg(jv_string("gmtime not implemented on this platform"));
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef HAVE_STRFTIME
|
||||
static jv f_strftime(jq_state *jq, jv a, jv b) {
|
||||
if (jv_get_kind(a) == JV_KIND_NUMBER) {
|
||||
a = f_gmtime(jq, a);
|
||||
} else if (jv_get_kind(a) != JV_KIND_ARRAY) {
|
||||
return jv_invalid_with_msg(jv_string("strftime/1 requires parsed datetime inputs"));
|
||||
}
|
||||
struct tm tm;
|
||||
if (!jv2tm(a, &tm))
|
||||
return jv_invalid_with_msg(jv_string("strftime/1 requires parsed datetime inputs")); \
|
||||
const char *fmt = jv_string_value(b);
|
||||
size_t alloced = strlen(fmt) + 100;
|
||||
char *buf = alloca(alloced);
|
||||
size_t n = strftime(buf, alloced, fmt, &tm);
|
||||
jv_free(b);
|
||||
/* POSIX doesn't provide errno values for strftime() failures; weird */
|
||||
if (n == 0 || n > alloced)
|
||||
return jv_invalid_with_msg(jv_string("strftime/1: unknown system failure"));
|
||||
return jv_string(buf);
|
||||
}
|
||||
#else
|
||||
static jv f_strftime(jq_state *jq, jv a) {
|
||||
jv_free(a);
|
||||
jv_free(b);
|
||||
return jv_invalid_with_msg(jv_string("strftime/1 not implemented on this platform"));
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef HAVE_GETTIMEOFDAY
|
||||
static jv f_now(jq_state *jq, jv a) {
|
||||
jv_free(a);
|
||||
struct timeval tv;
|
||||
if (gettimeofday(&tv, NULL) == -1)
|
||||
return jv_number(time(NULL));
|
||||
return jv_number(tv.tv_sec + tv.tv_usec / 1000000.0);
|
||||
}
|
||||
#else
|
||||
static jv f_now(jq_state *jq, jv a) {
|
||||
jv_free(a);
|
||||
return jv_number(time(NULL));
|
||||
}
|
||||
#endif
|
||||
|
||||
|
||||
#define LIBM_DD(name) \
|
||||
{(cfunction_ptr)f_ ## name, "_" #name, 1},
|
||||
@@ -1047,7 +1164,10 @@ static const struct cfunction function_list[] = {
|
||||
{(cfunction_ptr)f_debug, "debug", 1},
|
||||
{(cfunction_ptr)f_stderr, "stderr", 1},
|
||||
{(cfunction_ptr)f_strptime, "strptime", 2},
|
||||
{(cfunction_ptr)f_strftime, "strftime", 2},
|
||||
{(cfunction_ptr)f_mktime, "mktime", 1},
|
||||
{(cfunction_ptr)f_gmtime, "gmtime", 1},
|
||||
{(cfunction_ptr)f_now, "now", 1},
|
||||
};
|
||||
#undef LIBM_DD
|
||||
|
||||
@@ -1158,6 +1278,10 @@ static const char* const jq_builtins[] = {
|
||||
"def flatten: reduce .[] as $i ([]; if $i | type == \"array\" then . + ($i | flatten) else . + [$i] end);",
|
||||
"def flatten($x): reduce .[] as $i ([]; if $i | type == \"array\" and $x > 0 then . + ($i | flatten($x-1)) else . + [$i] end);",
|
||||
"def range($x): range(0;$x);",
|
||||
"def fromdateiso8601: strptime(\"%Y-%m-%dT%H:%M:%SZ\")|mktime;",
|
||||
"def todateiso8601: strftime(\"%Y-%m-%dT%H:%M:%SZ\");",
|
||||
"def fromdate: fromdateiso8601;",
|
||||
"def todate: todateiso8601;",
|
||||
#ifdef HAVE_ONIGURUMA
|
||||
"def match(re; mode): _match_impl(re; mode; false)|.[];",
|
||||
"def match($val): ($val|type) as $vt | if $vt == \"string\" then match($val; null)"
|
||||
|
11
configure.ac
11
configure.ac
@@ -115,7 +115,16 @@ AM_CONDITIONAL([ENABLE_DOCS], [test "x$enable_docs" != xno])
|
||||
|
||||
AC_FIND_FUNC([isatty], [c], [#include <unistd.h>], [0])
|
||||
AC_FIND_FUNC([_isatty], [c], [#include <io.h>], [0])
|
||||
AC_FIND_FUNC([strptime], [c], [#include <time.h>], [0])
|
||||
AC_FIND_FUNC([strptime], [c], [#include <time.h>], [0, 0, 0])
|
||||
AC_FIND_FUNC([strftime], [c], [#include <time.h>], [0, 0, 0, 0])
|
||||
AC_FIND_FUNC([timegm], [c], [#include <time.h>], [0])
|
||||
AC_FIND_FUNC([gmtime_r], [c], [#include <time.h>], [0, 0])
|
||||
AC_FIND_FUNC([gmtime], [c], [#include <time.h>], [0])
|
||||
AC_FIND_FUNC([gettimeofday], [c], [#include <time.h>], [0, 0])
|
||||
AC_CHECK_MEMBER([struct tm.tm_gmtoff], [AC_DEFINE([HAVE_TM_TM_GMT_OFF],1,[Define to 1 if the system has the tm_gmt_off field in struct tm])],
|
||||
[], [[#include <time.h>]])
|
||||
AC_CHECK_MEMBER([struct tm.__tm_gmtoff], [AC_DEFINE([HAVE_TM___TM_GMT_OFF],1,[Define to 1 if the system has the __tm_gmt_off field in struct tm])],
|
||||
[], [[#include <time.h>]])
|
||||
|
||||
AC_ARG_ENABLE([pthread-tls],
|
||||
[AC_HELP_STRING([--enable-pthread-tls],
|
||||
|
@@ -1594,24 +1594,69 @@ sections:
|
||||
- title: "Dates"
|
||||
body: |
|
||||
|
||||
The `strptime(fmt)` function parses input strings matching the
|
||||
`fmt` argument. The output is an array of eight numbers: the
|
||||
year, the month, the day of the month, the hour of the day,
|
||||
the minute of the hour, the second of the minute, the day of
|
||||
the week, and the day of the year, all zero-based except for
|
||||
the year and the month, which are one-based.
|
||||
jq provides some basic date handling functionality, with some
|
||||
high-level and low-level builtins. In all cases these
|
||||
builtins deal exclusively with time in UTC.
|
||||
|
||||
The `mktime` function consumes outputs from `strptime/1` and
|
||||
produces the time in seconds since the Unix epoch.
|
||||
The `fromdateiso8601` builtin parses datetimes in the ISO 8601
|
||||
format to a number of seconds since the Unix epoch
|
||||
(1970-01-01T00:00:00Z). The `todateiso8601` builtin does the
|
||||
inverse.
|
||||
|
||||
The `fromdate` builtin parses datetime strings. Currently
|
||||
`fromdate` only supports ISO 8601 datetime strings, but in the
|
||||
future it will attempt to parse datetime strings in more
|
||||
formats.
|
||||
|
||||
The `todate` builtin is an alias for `todateiso8601`.
|
||||
|
||||
The `now` builtin outputs the current time, in seconds since
|
||||
the Unix epoch.
|
||||
|
||||
Low-level jq interfaces to the C-library time functions are
|
||||
also provided: `strptime`, `strftime`, `mktime`, and `gmtime`.
|
||||
Refer to your host operating system's documentation for the
|
||||
format strings used by `strptime` and `strftime`. Note: these
|
||||
are not necessarily stable interfaces in jq, particularly as
|
||||
to their localization functionality.
|
||||
|
||||
The `gmtime` builtin consumes a number of seconds since the
|
||||
Unix epoch and outputs a "broken down time" representation of
|
||||
time as an array of numbers representing (in this order): the
|
||||
year, the month (zero-based), the day of the month, the hour
|
||||
of the day, the minute of the hour, the second of the minute,
|
||||
the day of the week, and the day of the year -- all one-based
|
||||
unless otherwise stated.
|
||||
|
||||
The `mktime` builtin consumes "broken down time"
|
||||
representations of time output by `gmtime` and `strptime`.
|
||||
|
||||
The `strptime(fmt)` builtin parses input strings matching the
|
||||
`fmt` argument. The output is in the "broken down time"
|
||||
representation consumed by `gmtime` and output by `mktime`.
|
||||
|
||||
The `strftime(fmt)` builtin formats a time with the given
|
||||
format.
|
||||
|
||||
The format strings for `strptime` and `strftime` are described
|
||||
in typical C library documentation. The format string for ISO
|
||||
8601 datetime is `"%Y-%m-%dT%H:%M:%SZ"`.
|
||||
|
||||
jq may not support some or all of this date functionality on
|
||||
some systems.
|
||||
|
||||
examples:
|
||||
- program: 'strptime'
|
||||
input: '"2015-03-05 23:51:47Z"'
|
||||
- program: 'fromdate'
|
||||
input: '"2015-03-05T23:51:47Z"'
|
||||
output: ['1425599507']
|
||||
|
||||
- program: 'strptime("%Y-%m-%dT%H:%M:%SZ")'
|
||||
input: '"2015-03-05T23:51:47Z"'
|
||||
output: ['[2015,2,5,23,51,47,4,63]']
|
||||
|
||||
- program: 'strptime|mktime'
|
||||
input: '"2015-03-05 23:51:47Z"'
|
||||
output: ['1425621107']
|
||||
- program: 'strptime("%Y-%m-%dT%H:%M:%SZ")|mktime'
|
||||
input: '"2015-03-05T23:51:47Z"'
|
||||
output: ['1425599507']
|
||||
|
||||
- title: Conditionals and Comparisons
|
||||
entries:
|
||||
|
@@ -1140,9 +1140,17 @@ bsearch(4)
|
||||
[1,2,3]
|
||||
-4
|
||||
|
||||
[strptime("%Y-%m-%d %H:%M:%SZ")|(.,mktime)]
|
||||
"2015-03-05 23:51:47Z"
|
||||
[[2015,2,5,23,51,47,4,63],1425621107]
|
||||
[strptime("%Y-%m-%dT%H:%M:%SZ")|(.,mktime)]
|
||||
"2015-03-05T23:51:47Z"
|
||||
[[2015,2,5,23,51,47,4,63],1425599507]
|
||||
|
||||
strftime("%Y-%m-%dT%H:%M:%SZ")
|
||||
[2015,2,5,23,51,47,4,63]
|
||||
"2015-03-05T23:51:47Z"
|
||||
|
||||
gmtime
|
||||
1425599507
|
||||
[2015,2,5,23,51,47,4,63]
|
||||
|
||||
# module system
|
||||
import "a" as foo; import "b" as bar; def fooa: foo::a; [fooa, bar::a, bar::b, foo::a]
|
||||
|
Reference in New Issue
Block a user