260

This answer and comments mention --rfc-3339 and a "hidden" --iso-8601 option that I have used for a long time and now seems to be undocumented.

When did that option documentation get removed from the --help text?

Will the option go away anytime soon?

4
  • 3
    To get this feature on OSX I finally got the GNU tools: $ brew install coreutils ; gdate --iso-8601 -d "yesterday 12:00" Commented Jul 5, 2018 at 18:51
  • 23
    @AnneTheAgile for macOS, you can manually format the date like so: "date +%Y-%m-%dT%H:%M:%S%z" Commented Apr 23, 2019 at 12:47
  • 3
    I feel like the top answers are pretty complicated considering that date -Iseconds seems to "just work" on all platforms Commented Oct 4, 2022 at 17:46
  • in 2023 date -I is working at UBUNTU Commented Apr 18, 2023 at 21:25

6 Answers 6

184

The option was introduced in the coreutils date (which is probably what you have) in 1999 (Apr. 8).

The documentation was removed in 2005 without much explanation in the commit.

In 2011, the help for --iso-8601 was reintroduced with the following explanation:

We deprecated and undocumented the --iso-8601 (-I) option mostly
because date could not parse that particular format.  Now that
it can, it's time to restore the documentation.
* src/date.c (usage): Document it.
* doc/coreutils.texi (Options for date): Reinstate documentation.
Reported by Hubert Depesz Lubaczewski in http://bugs.gnu.org/7444.

It looks like the help was taken out in version 5.90 and put back in, in version 8.15 (it is not in my 8.13) and the comment above suggests that it is now back to stay and not likely to be disappearing any time soon.

In version 8.31 (as provided by Solus July 2020) the man page has these two options:

   -I[FMT], --iso-8601[=FMT]
          output date/time in ISO 8601 format.  FMT='date' for date only (the default), 'hours', 'minutes', 'sec‐
          onds', or 'ns' for date and time to the indicated precision.  Example: 2006-08-14T02:34:56-06:00

   --rfc-3339=FMT
          output date/time in RFC 3339 format.  FMT='date', 'seconds', or 'ns' for date and time to the indicated
          precision.  Example: 2006-08-14 02:34:56-06:00
0
178

For a platform independent, (almost) fully compliant, ISO 8601 date, use this:

date +"%Y-%m-%dT%H:%M:%S%z"

This will result in a time such as: 2021-01-16T23:09:44-0500

This should work on macOS (BSD) and Linux.

Technically, for full ISO 8601 compliance, there should be a colon in the timezone offset:

date +"%Y-%m-%dT%H:%M:%S%:z"

The %:z format directive contains a colon between the hour and minute of the timezone offset. This will result in a fully compliant ISO 8601 time with a colon in the time offset, such as: 2021-01-16T23:09:44-05:00

But this doesn't work on macOS (as of this writing). That is, the date command on macOS (BSD) does not support the %:z directive. This should work on Linux (with GNU date).

Fully compliant and portable solution:

You can pass the date output through sed for a fully compliant ISO 8601 date. This should be portable on both macOS (BSD) and Linux:

date +"%Y-%m-%dT%H:%M:%S%z" | sed -E 's/([+-][0-9]{2})([0-9]{2})$/\1:\2/'

For UTC time (zero offset, historically known as "Zulu time") you can use:

date -u +"%Y-%m-%dT%H:%M:%SZ"

Note: The -u option sets the output to UTC time. The Z is not preceded by a % (or a colon) – so it is not a format directive; it is a literal 'Z' character. This is also fully compliant and works on macOS (BSD) and Linux.

This will result in a time such as: 2021-01-17T04:16:14Z

5
  • 4
    Mac OSX seems happy now w/zsh & bash: % date +"%Y-%m-%dT%H:%M:%S%:z" yields 2022-03-22T11:05:42:z on Darwin 21.3.0 Darwin Kernel Version 21.3.0. Commented Mar 22, 2022 at 17:11
  • 9
    @jgreve no, you just showed that Mac's date does not understand the %:z format control. Check your output thoroughly, it wrote literal :z instead of your time zone with the : separator like for example date +%:z -> +02:00 --- Also shell normally has nothing to do with this because date is normally an external executable, not the shell's built-in. Commented May 6, 2022 at 15:12
  • 1
    argh. :-/ I stand corrected. Commented May 6, 2022 at 16:36
  • 4
    MacOS 12.6 supports date -Iseconds, can't figure out when this was changed though! Commented Oct 9, 2022 at 18:05
  • 1
    The first example also is not fully portable, as %z is not defined (only %Z is defined) in the POSIX standard for date(1) (see pubs.opengroup.org/onlinepubs/9699919799/utilities/date.html) Commented Jul 16, 2023 at 17:21
65

The --help got an update recently actually, so the option definitely isn't going away:

-I[FMT], --iso-8601[=FMT]  output date/time in ISO 8601 format.
                             FMT='date' for date only (the default),
                             'hours', 'minutes', 'seconds', or 'ns'
                             for date and time to the indicated precision.
                             Example: 2006-08-14T02:34:56-06:00

     -R, --rfc-2822        output date and time in RFC 2822 format.
                             Example: Mon, 14 Aug 2006 02:34:56 -0600

         --rfc-3339=FMT    output date/time in RFC 3339 format.
                             FMT='date', 'seconds', or 'ns'
                             for date and time to the indicated precision.
                             Example: 2006-08-14 02:34:56-06:00

Note since coreutils-8.27 --rfc-2822 is deprecated in favor of the more general --rfc-email

     -R, --rfc-email       output date and time in RFC 5322 format.
                             Example: Mon, 14 Aug 2006 02:34:56 -0600
3
  • 8
    It would be nice if there was a -I style without the timezone. Would be more useful for logging, etc, where the timezone doesn't matter Commented Jan 17, 2019 at 5:56
  • 2
    @naught101 you can get rid of the timezone offset by specifying the component parts: date +"%Y-%m-%dT%H:%M:%S. Commented Jan 23, 2023 at 22:13
  • From my experience, it is always good to have the timezone set - especially when logging. Commented Sep 12, 2024 at 15:20
58

I'm running Linux Mint and the option is available:

$ lsb_release -a
No LSB modules are available.
Distributor ID: LinuxMint
Description:    Linux Mint 17.3 Rosa
Release:    17.3
Codename:   rosa

The execution of the command:

$ date --iso-8601=seconds
2016-12-14T09:53:25-0400
6
  • 29
    or just date -Iseconds Commented Nov 15, 2019 at 1:29
  • 14
    but it doesn't appends Z to the end so this is not fully complete ISO timestamp Commented Jan 7, 2020 at 10:34
  • 7
    date -In (same as date -Ins) or date -Is same as (date -Iseconds) etc. Commented Jun 13, 2021 at 19:08
  • 8
    "but it doesn't appends Z to the end so this is not fully complete ISO timestamp" Z is the UTC timezone. +00:00 is the same as Z. +00:00Z is not correct. Commented Jul 20, 2021 at 17:08
  • 3
    For a filename compatible format, I use $(date -u +"%Y-%m-%dT%H-%M-%SZ") Commented May 20, 2022 at 9:48
4

I think this is perfect, maybe you will also like it:

date -u --iso-8601=ns | sed s/+00:00/Z/ | sed s/,/./

Result example: 2022-12-10T20:44:36.753578818Z

0

Here's a few examples. I'll use -d@123456789 to make the command relative to that easy-to-remember Unix timestamp. This way this should reproduce on your system verbatim. -- Just leave that part out if you want to make date relative to the actual current date.

$ date -d@123456789 --iso-8601=seconds --utc
1973-11-29T21:33:09+00:00

Shorter:

$ date -d@123456789 -Is -u
1973-11-29T21:33:09+00:00

Shortest:

$ date -d@123456789 -uIs
1973-11-29T21:33:09+00:00

And if you want a Z instead of +00:00, then you can replace it...

$ date -d@123456789 -uIs | sed 's/+00:00/Z/'
1973-11-29T21:33:09Z

...or you can use the formatstring from the INFO page (linked below)...

$ date -d @123456789 '+%Y-%m-%dT%H:%M:%S%:z' --utc
1973-11-29T21:33:09+00:00

...and change that around a little and replace the variable %:z with just a literal letter Z:

$ date -d @123456789 '+%Y-%m-%dT%H:%M:%SZ' --utc
1973-11-29T21:33:09Z

Details:

I'm running MobaXterm on Win10. And my date says this:

$ date --version
date (GNU coreutils) 9.0
Packaged by Cygwin (9.0-1)
Copyright (C) 2021 Free Software Foundation, Inc.
License GPLv3+: GNU GPL version 3 or later <https://gnu.org/licenses/gpl.html>.
This is free software: you are free to change and redistribute it.
There is NO WARRANTY, to the extent permitted by law.

Written by David MacKenzie.

Builtin help:

$ date --help
Usage: /bin/date [OPTION]... [+FORMAT]
  or:  /bin/date [-u|--utc|--universal] [MMDDhhmm[[CC]YY][.ss]]
Display the current time in the given FORMAT, or set the system date.

Mandatory arguments to long options are mandatory for short options too.
  -d, --date=STRING          display time described by STRING, not 'now'
      --debug                annotate the parsed date,
                              and warn about questionable usage to stderr
  -f, --file=DATEFILE        like --date; once for each line of DATEFILE
  -I[FMT], --iso-8601[=FMT]  output date/time in ISO 8601 format.
                               FMT='date' for date only (the default),
                               'hours', 'minutes', 'seconds', or 'ns'
                               for date and time to the indicated precision.
                               Example: 2006-08-14T02:34:56-06:00
  -R, --rfc-email            output date and time in RFC 5322 format.
                               Example: Mon, 14 Aug 2006 02:34:56 -0600
      --rfc-3339=FMT         output date/time in RFC 3339 format.
                               FMT='date', 'seconds', or 'ns'
                               for date and time to the indicated precision.
                               Example: 2006-08-14 02:34:56-06:00
  -r, --reference=FILE       display the last modification time of FILE
  -s, --set=STRING           set time described by STRING
  -u, --utc, --universal     print or set Coordinated Universal Time (UTC)
      --help     display this help and exit
      --version  output version information and exit

FORMAT controls the output.  Interpreted sequences are:

  %%   a literal %
  %a   locale's abbreviated weekday name (e.g., Sun)
  %A   locale's full weekday name (e.g., Sunday)
  %b   locale's abbreviated month name (e.g., Jan)
  %B   locale's full month name (e.g., January)
  %c   locale's date and time (e.g., Thu Mar  3 23:05:25 2005)
  %C   century; like %Y, except omit last two digits (e.g., 20)
  %d   day of month (e.g., 01)
  %D   date; same as %m/%d/%y
  %e   day of month, space padded; same as %_d
  %F   full date; like %+4Y-%m-%d
  %g   last two digits of year of ISO week number (see %G)
  %G   year of ISO week number (see %V); normally useful only with %V
  %h   same as %b
  %H   hour (00..23)
  %I   hour (01..12)
  %j   day of year (001..366)
  %k   hour, space padded ( 0..23); same as %_H
  %l   hour, space padded ( 1..12); same as %_I
  %m   month (01..12)
  %M   minute (00..59)
  %n   a newline
  %N   nanoseconds (000000000..999999999)
  %p   locale's equivalent of either AM or PM; blank if not known
  %P   like %p, but lower case
  %q   quarter of year (1..4)
  %r   locale's 12-hour clock time (e.g., 11:11:04 PM)
  %R   24-hour hour and minute; same as %H:%M
  %s   seconds since the Epoch (1970-01-01 00:00 UTC)
  %S   second (00..60)
  %t   a tab
  %T   time; same as %H:%M:%S
  %u   day of week (1..7); 1 is Monday
  %U   week number of year, with Sunday as first day of week (00..53)
  %V   ISO week number, with Monday as first day of week (01..53)
  %w   day of week (0..6); 0 is Sunday
  %W   week number of year, with Monday as first day of week (00..53)
  %x   locale's date representation (e.g., 12/31/99)
  %X   locale's time representation (e.g., 23:13:48)
  %y   last two digits of year (00..99)
  %Y   year
  %z   +hhmm numeric time zone (e.g., -0400)
  %:z  +hh:mm numeric time zone (e.g., -04:00)
  %::z  +hh:mm:ss numeric time zone (e.g., -04:00:00)
  %:::z  numeric time zone with : to necessary precision (e.g., -04, +05:30)
  %Z   alphabetic time zone abbreviation (e.g., EDT)

By default, date pads numeric fields with zeroes.
The following optional flags may follow '%':

  -  (hyphen) do not pad the field
  _  (underscore) pad with spaces
  0  (zero) pad with zeros
  +  pad with zeros, and put '+' before future years with >4 digits
  ^  use upper case if possible
  #  use opposite case if possible

After any flags comes an optional field width, as a decimal number;
then an optional modifier, which is either
E to use the locale's alternate representations if available, or
O to use the locale's alternate numeric symbols if available.

Examples:
Convert seconds since the Epoch (1970-01-01 UTC) to a date
  $ date --date='@2147483647'

Show the time on the west coast of the US (use tzselect(1) to find TZ)
  $ TZ='America/Los_Angeles' date

Show the local time for 9AM next Friday on the west coast of the US
  $ date --date='TZ="America/Los_Angeles" 09:00 next Fri'

GNU coreutils online help: <https://www.gnu.org/software/coreutils/>
Full documentation <https://www.gnu.org/software/coreutils/date>
or available locally via: info '(coreutils) date invocation'

You must log in to answer this question.

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.