Testland
Browse all skills & agents

iso-8601-vs-rfc-3339-reference

Pure-reference catalog of the ISO 8601 vs RFC 3339 distinction. Covers the relationship (RFC 3339 is a strict subset of ISO 8601 designed for internet protocols), the syntactic differences (RFC 3339 disallows ISO 8601's '+02' offset short-form requires '+02:00'; RFC 3339 mandates a date-time separator T or space; ISO 8601 allows much more), the canonical date-time string format (YYYY-MM-DDTHH:MM:SS[.fff]±HH:MM or Z), per-language parser behaviour (Python isoformat, Java Instant.parse, JS Date.parse non-spec), and serialisation rules for APIs. Use when choosing a wire format, parsing third-party datetimes, or auditing time-string handling.

iso-8601-vs-rfc-3339-reference

Overview

ISO 8601 and RFC 3339 are often used interchangeably but they are not the same. RFC 3339 is a stricter subset of ISO 8601 designed specifically for internet protocols. ISO 8601 has many optional variations (ordinal dates, week-dates, no separators, etc.) that RFC 3339 forbids.

Per RFC 3339 §5.6: it "defines a date and time format for use in Internet protocols that is a profile of the ISO 8601 standard."

For APIs, always use RFC 3339. Parsers handle it predictably; ISO 8601 in full generality is parsing hell.

When to use

  • Picking a wire format for a new API.
  • Parsing a third-party datetime string of unclear provenance.
  • Auditing existing datetime handling.
  • Reviewing API specs / schemas.

The canonical format

Per RFC 3339:

2026-05-20T14:30:00Z                  # UTC (Z = +00:00)
2026-05-20T14:30:00.123456Z           # microsecond precision
2026-05-20T14:30:00+02:00             # CEST
2026-05-20T14:30:00-05:00             # EST
2026-05-20 14:30:00Z                  # space-separated (allowed)

Per RFC 3339:

  • Date format: YYYY-MM-DD (extended; no compact YYYYMMDD)
  • Date-time separator: T or space (T recommended)
  • Time-zone offset: Z or ±HH:MM (full form, with the colon)
  • Optional fractional seconds: .fff (any digits)

What RFC 3339 forbids that ISO 8601 allows

ISO 8601 validRFC 3339
20260520T143000Z (compact, no separators)Forbidden - needs hyphens + colons
2026-05-20T14:30:00+02 (offset short form)Forbidden - must be +02:00
2026-W21-3 (week date)Forbidden - week dates not supported
2026-140 (ordinal date)Forbidden - ordinal dates not supported
2026-05-20T14:30:00,123Z (comma decimal)Forbidden - period only
--05-20 (omitted year)Forbidden - year required
+002026-05-20T... (extended year)Forbidden - 4 digits
24:00:00 (midnight as end-of-day)Forbidden - only 00:00:00 (start)

Per-language parser support

LanguageRFC 3339 strictISO 8601 fullTolerance
Python datetime.fromisoformat (3.11+)yespartialAccepts most RFC 3339; pre-3.11 didn't accept Z
Python dateutil.parseryesmostly yesLenient
Java Instant.parseyes (RFC 3339 + Z)noStrict ISO 8601 subset
Java OffsetDateTime.parseyesmostly yesLenient
JavaScript Date.parseplatform-dependentNONon-spec; varies by browser
Rust chronoyespartialLenient
Go time.Parse(time.RFC3339, ...)yesnoStrict; use RFC3339Nano for fractional
.NET DateTimeOffset.Parseyesmostly yesLenient

JavaScript Date.parse is the worst. Per developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Date/parse: "implementation-specific... The exact behavior of this function varies between implementations." Use a library (date-fns, dayjs) or the Temporal proposal where available.

Serialisation rules for APIs

RuleWhy
Always include time-zone offsetWithout it, the receiver guesses
Prefer UTC (Z) for storage / wireAvoids per-region drift
Use T separatorMore universally accepted
Include microseconds for distributed-systems useSubsecond resolution for ordering
Round-trip safely: parse + emit produces the same stringTest this; some libraries don't
For local-time semantics, emit offset (not zone name)+02:00 is portable; Europe/Berlin requires per-receiver zoneinfo

Common pitfalls

"Local time without offset"

2026-05-20T14:30:00

No Z, no +02:00. Ambiguous. Different libraries interpret differently:

  • Java Instant.parse → throws (offset required)
  • Python datetime.fromisoformat → returns naive datetime
  • JavaScript new Date(...) → assumes browser-local timezone

For wire format, always include offset.

Sortability

UTC strings (...Z) are lexically sortable. Mixed-offset strings are not - 2026-05-20T14:30:00+02:00 sorts before 2026-05-20T13:00:00Z even though it's later in time.

Always store UTC. Display local if needed.

Date-only

ISO 8601 allows 2026-05-20 (date without time). RFC 3339 §5.6 calls this "full-date" and allows it. But:

  • Some receivers parse it as 2026-05-20T00:00:00
  • Others as 2026-05-20T12:00:00 (noon to avoid timezone-flip surprises)

If you need date-only, document explicitly and test parsing.

Testable behaviours

BehaviourTest
Parser accepts all RFC 3339 formsT separator, space separator, microseconds, Z, +HH:MM
Parser rejects ISO-8601-only formsWeek date, ordinal date, compact T143000
Round-trip preserves precisionparse + serialise = original
Sortability holds for UTC stringsSort 1000 random UTC timestamps; verify lexical = chronological
API spec documents the formatOpenAPI uses format: date-time (RFC 3339)
Per-language client + server agreeCross-language test fixture

Anti-patterns

Anti-patternWhy it failsFix
Date.parse('2026-05-20') in JSImplementation-dependent (browser-local? UTC?)Use a library; specify offset
Mix UTC and local-with-offset in same fieldSortability broken; consumer confusionPick one wire format
Skip the offset (2026-05-20T14:30:00)AmbiguousAlways offset
Storing local-format stringsLose tz info; can't reconstructStore UTC + zone name separately if local matters
4-digit milliseconds (2026...000)Not all parsers acceptUse 3 (milli) or 6 (micro) digits
Sub-second precision but UTC stringLose subsec on round-trip in some librariesTest the round-trip
Trusting Date.parse('5/20/2026')US format; ambiguousUse RFC 3339 always
Date-only without explicit timeReceiver-defined behaviourSpecify T00:00:00Z or document

Limitations

  • RFC 3339 is silent on leap seconds in the wire format. Allows 60 in the seconds field; parsers vary on acceptance.
  • No native "date only" or "time only" in RFC 3339. ISO 8601 has both; receivers need both.
  • No durations or intervals. ISO 8601 has P1Y2M3D, RFC 3339 doesn't directly speak to these.
  • 2038 problem (32-bit time_t) is orthogonal but worth noting; affects parsing of timestamps near 2038-01-19.

References