Case Study: ISO 8601 Date + Time#
Let us make use of what we have explored so far in a more elaborate case study. ISO 8601 is an international standard for exchange and communication of date and time-related data, providing an unambiguous method of representing calendar dates and times.
In this chapter, we will define a full Fandango spec for ISO 8601 date and time formats, ensuring both syntactic and semantic validity. To make matters more interesting, we will create the spec programmatically - that is, by having a number of Python functions that generate the spec for us.
Creating Grammars Programmatically#
We start with a number of functions that help us create fragments of the grammar.
make_rule()
creates a grammar rule from symbol
and its possible expansions
:
import sys
def print_s(s: str) -> str:
print(s, end="")
return s
def make_rule(symbol: str, expansions: list[str], sep: str = '') -> str:
return print_s(f"{sep}<{symbol}> ::= " + " | ".join(expansions) + "\n")
make_rule("start", ["<iso8601datetime>"]); # a final ";" suppresses result
<start> ::= <iso8601datetime>
We can also use make_rule()
to quickly create a list of expansions:
make_rule("digit", [f"'{digit}'" for digit in range(0, 10)]);
<digit> ::= '0' | '1' | '2' | '3' | '4' | '5' | '6' | '7' | '8' | '9'
These additional functions help in adding non-grammar elements such as headers, code, and constraints:
def make_header(title: str) -> str:
return print_s(f"\n# {title}\n")
def make_comment(comment: str, sep: str = '') -> str:
return print_s(f"{sep}# {comment}\n")
def make_constraint(constraint: str) -> str:
return print_s(f"where {constraint}\n")
def make_code(code: str) -> str:
return print_s(f"{code}\n")
make_header("ISO 8601 grammar");
# ISO 8601 grammar
Spec Header#
Let us create a Fandango spec.
For now, we store the final spec in the iso8601lib
string variable, to be saved in a file at the end.
Every time we add a new fragment, the new fragment will be output here, so we can see the actual content of iso8601lib
incrementally.
iso8601lib = make_header("ISO 8601 date and time")
iso8601lib += make_comment("Generated by docs/ISO8601.md. Do not edit.")
# ISO 8601 date and time
# Generated by docs/ISO8601.md. Do not edit.
We will need the datetime
library below, so we import it.
iso8601lib += make_code("\nimport datetime\n")
import datetime
Date#
We start with a <start>
symbol.
An ISO 8601 date/time spec starts with a date, and an optional time, separated by T
:
iso8601lib += make_rule("start", ["<iso8601datetime>"])
iso8601lib += make_rule("iso8601datetime",
["<iso8601date> ('T' <iso8601time>)?"])
<start> ::= <iso8601datetime>
<iso8601datetime> ::= <iso8601date> ('T' <iso8601time>)?
A date can either be a calendar date, but also a week date or an ordinal date.
iso8601lib += make_rule("iso8601date",
["<iso8601calendardate>",
"<iso8601weekdate>",
"<iso8601ordinaldate>"], sep='\n')
<iso8601date> ::= <iso8601calendardate> | <iso8601weekdate> | <iso8601ordinaldate>
Calendar Dates#
An ISO 8601 calendar date has the format YYYY-MM-DD, but can also be YYYY-MM or YYYYMMDD.
The year can be prefixed with +
or -
to indicate AC/BC (or CE/BCE) designators.
iso8601lib += make_rule("iso8601calendardate",
["<iso8601year> '-' <iso8601month> ('-' <iso8601day>)?",
"<iso8601year> <iso8601month> <iso8601day>"], sep='\n')
iso8601lib += make_rule("iso8601year", ["('+'|'-')? <digit>{4}"])
<iso8601calendardate> ::= <iso8601year> '-' <iso8601month> ('-' <iso8601day>)? | <iso8601year> <iso8601month> <iso8601day>
<iso8601year> ::= ('+'|'-')? <digit>{4}
Months#
Months are easy:
iso8601lib += make_rule("iso8601month",
[f"'{month:02d}'" for month in range(1, 13)])
<iso8601month> ::= '01' | '02' | '03' | '04' | '05' | '06' | '07' | '08' | '09' | '10' | '11' | '12'
Days#
As with months, we define each day individually to avoid biasing the grammar. The fact that some months have only 28, 29, or 30 days will be handled later on in a constraint.
iso8601lib += make_rule("iso8601day",
[f"'{day:02d}'" for day in range(1, 32)])
<iso8601day> ::= '01' | '02' | '03' | '04' | '05' | '06' | '07' | '08' | '09' | '10' | '11' | '12' | '13' | '14' | '15' | '16' | '17' | '18' | '19' | '20' | '21' | '22' | '23' | '24' | '25' | '26' | '27' | '28' | '29' | '30' | '31'
Tip
For testing purposes, it makes sense to test with extreme values - in our case, January 1, February 28 or 29, and December 31.
Week Dates#
ISO 8601 also allows specifying dates by week numbers (1 - 53) and optional days of the week (1 - 7).
2025-W12-1
is the Monday of the 12th week in 2025.
iso8601lib += make_rule("iso8601weekdate",
["<iso8601year> '-'? 'W' <iso8601week> ('-' <iso8601weekday>)?"], sep='\n')
iso8601lib += make_rule("iso8601week",
[f"'{week:02d}'" for week in range(1, 54)])
iso8601lib += make_rule("iso8601weekday",
[f"'{weekday:1d}'" for weekday in range(1, 8)])
<iso8601weekdate> ::= <iso8601year> '-'? 'W' <iso8601week> ('-' <iso8601weekday>)?
<iso8601week> ::= '01' | '02' | '03' | '04' | '05' | '06' | '07' | '08' | '09' | '10' | '11' | '12' | '13' | '14' | '15' | '16' | '17' | '18' | '19' | '20' | '21' | '22' | '23' | '24' | '25' | '26' | '27' | '28' | '29' | '30' | '31' | '32' | '33' | '34' | '35' | '36' | '37' | '38' | '39' | '40' | '41' | '42' | '43' | '44' | '45' | '46' | '47' | '48' | '49' | '50' | '51' | '52' | '53'
<iso8601weekday> ::= '1' | '2' | '3' | '4' | '5' | '6' | '7'
Ordinal Dates#
ISO 8601 also allows specifying dates by day numbers (1 - 366).
2000-366
, for instance, specifies the last day in 2000 (which was a leap year).
Again, we specify each day programmatically (and individually).
iso8601lib += make_rule("iso8601ordinaldate",
["<iso8601year> ('-'? <iso8601ordinalday>)?"], sep='\n')
iso8601lib += make_rule("iso8601ordinalday",
[f"'{day:03d}'" for day in range(1, 367)])
<iso8601ordinaldate> ::= <iso8601year> ('-'? <iso8601ordinalday>)?
<iso8601ordinalday> ::= '001' | '002' | '003' | '004' | '005' | '006' | '007' | '008' | '009' | '010' | '011' | '012' | '013' | '014' | '015' | '016' | '017' | '018' | '019' | '020' | '021' | '022' | '023' | '024' | '025' | '026' | '027' | '028' | '029' | '030' | '031' | '032' | '033' | '034' | '035' | '036' | '037' | '038' | '039' | '040' | '041' | '042' | '043' | '044' | '045' | '046' | '047' | '048' | '049' | '050' | '051' | '052' | '053' | '054' | '055' | '056' | '057' | '058' | '059' | '060' | '061' | '062' | '063' | '064' | '065' | '066' | '067' | '068' | '069' | '070' | '071' | '072' | '073' | '074' | '075' | '076' | '077' | '078' | '079' | '080' | '081' | '082' | '083' | '084' | '085' | '086' | '087' | '088' | '089' | '090' | '091' | '092' | '093' | '094' | '095' | '096' | '097' | '098' | '099' | '100' | '101' | '102' | '103' | '104' | '105' | '106' | '107' | '108' | '109' | '110' | '111' | '112' | '113' | '114' | '115' | '116' | '117' | '118' | '119' | '120' | '121' | '122' | '123' | '124' | '125' | '126' | '127' | '128' | '129' | '130' | '131' | '132' | '133' | '134' | '135' | '136' | '137' | '138' | '139' | '140' | '141' | '142' | '143' | '144' | '145' | '146' | '147' | '148' | '149' | '150' | '151' | '152' | '153' | '154' | '155' | '156' | '157' | '158' | '159' | '160' | '161' | '162' | '163' | '164' | '165' | '166' | '167' | '168' | '169' | '170' | '171' | '172' | '173' | '174' | '175' | '176' | '177' | '178' | '179' | '180' | '181' | '182' | '183' | '184' | '185' | '186' | '187' | '188' | '189' | '190' | '191' | '192' | '193' | '194' | '195' | '196' | '197' | '198' | '199' | '200' | '201' | '202' | '203' | '204' | '205' | '206' | '207' | '208' | '209' | '210' | '211' | '212' | '213' | '214' | '215' | '216' | '217' | '218' | '219' | '220' | '221' | '222' | '223' | '224' | '225' | '226' | '227' | '228' | '229' | '230' | '231' | '232' | '233' | '234' | '235' | '236' | '237' | '238' | '239' | '240' | '241' | '242' | '243' | '244' | '245' | '246' | '247' | '248' | '249' | '250' | '251' | '252' | '253' | '254' | '255' | '256' | '257' | '258' | '259' | '260' | '261' | '262' | '263' | '264' | '265' | '266' | '267' | '268' | '269' | '270' | '271' | '272' | '273' | '274' | '275' | '276' | '277' | '278' | '279' | '280' | '281' | '282' | '283' | '284' | '285' | '286' | '287' | '288' | '289' | '290' | '291' | '292' | '293' | '294' | '295' | '296' | '297' | '298' | '299' | '300' | '301' | '302' | '303' | '304' | '305' | '306' | '307' | '308' | '309' | '310' | '311' | '312' | '313' | '314' | '315' | '316' | '317' | '318' | '319' | '320' | '321' | '322' | '323' | '324' | '325' | '326' | '327' | '328' | '329' | '330' | '331' | '332' | '333' | '334' | '335' | '336' | '337' | '338' | '339' | '340' | '341' | '342' | '343' | '344' | '345' | '346' | '347' | '348' | '349' | '350' | '351' | '352' | '353' | '354' | '355' | '356' | '357' | '358' | '359' | '360' | '361' | '362' | '363' | '364' | '365' | '366'
Tip
For testing purposes, it makes sense to test with extreme values - in our case, 1
, 365
and 366
.
Time#
Now for time specifications.
In ISO 8601, time is specified as HH:MM:SS
, where HH
comes in 24-hour format.
MM
and SS
are optional, as is the colon separator in an âunambiguousâ context.
iso8601lib += make_rule("iso8601time",
["'T'? <iso8601hour> (':'? <iso8601minute> (':'? <iso8601second> (('.' | ',') <iso8601fraction>)? )? )? <iso8601timezone>?"], sep='\n')
<iso8601time> ::= 'T'? <iso8601hour> (':'? <iso8601minute> (':'? <iso8601second> (('.' | ',') <iso8601fraction>)? )? )? <iso8601timezone>?
Hours#
Hours normally go from 00
to 23
, but 24:00:00
is allowed to represent midnight at the end of a day.
iso8601lib += make_comment("24:00:00 is allowed to represent midnight at the end of a day", sep='\n')
iso8601lib += make_rule("iso8601hour",
[f"'{hour:02d}'" for hour in range(0, 25)])
# 24:00:00 is allowed to represent midnight at the end of a day
<iso8601hour> ::= '00' | '01' | '02' | '03' | '04' | '05' | '06' | '07' | '08' | '09' | '10' | '11' | '12' | '13' | '14' | '15' | '16' | '17' | '18' | '19' | '20' | '21' | '22' | '23' | '24'
Minutes#
Minutes go from 00
to 59
.
Nothing special here.
iso8601lib += make_rule("iso8601minute",
[f"'{minute:02d}'" for minute in range(0, 60)])
<iso8601minute> ::= '00' | '01' | '02' | '03' | '04' | '05' | '06' | '07' | '08' | '09' | '10' | '11' | '12' | '13' | '14' | '15' | '16' | '17' | '18' | '19' | '20' | '21' | '22' | '23' | '24' | '25' | '26' | '27' | '28' | '29' | '30' | '31' | '32' | '33' | '34' | '35' | '36' | '37' | '38' | '39' | '40' | '41' | '42' | '43' | '44' | '45' | '46' | '47' | '48' | '49' | '50' | '51' | '52' | '53' | '54' | '55' | '56' | '57' | '58' | '59'
Seconds#
Seconds normally go from 00
to 59
, but 60
can be used to represent leap seconds.
iso8601lib += make_comment("xx:yy:60 is allowed to represent leap seconds", sep='\n')
iso8601lib += make_rule("iso8601second",
[f"'{second:02d}'" for second in range(0, 61)])
# xx:yy:60 is allowed to represent leap seconds
<iso8601second> ::= '00' | '01' | '02' | '03' | '04' | '05' | '06' | '07' | '08' | '09' | '10' | '11' | '12' | '13' | '14' | '15' | '16' | '17' | '18' | '19' | '20' | '21' | '22' | '23' | '24' | '25' | '26' | '27' | '28' | '29' | '30' | '31' | '32' | '33' | '34' | '35' | '36' | '37' | '38' | '39' | '40' | '41' | '42' | '43' | '44' | '45' | '46' | '47' | '48' | '49' | '50' | '51' | '52' | '53' | '54' | '55' | '56' | '57' | '58' | '59' | '60'
Seconds can be followed by ,
or a .
(see <iso8601time>
, above) and a fraction - an arbitrary number of digits.
iso8601lib += make_rule("iso8601fraction", ["<digit>+"])
<iso8601fraction> ::= <digit>+
Tip
Testing with extreme values would mandate times such as 20:15:00,99999999999999999999
to test for possible buffer overflows.
Time Zones#
Any time can be followed by a time zone designator.
This is either Z
, indicating Coordinated Universal Time (UTC), or an offset designating the time zone.
iso8601lib += make_rule("iso8601timezone", ["'Z'",
"'+' <iso8601hour> (':'? <iso8601minute>)?",
"'-' <iso8601hour> (':'? <iso8601minute>)?"], sep='\n')
<iso8601timezone> ::= 'Z' | '+' <iso8601hour> (':'? <iso8601minute>)? | '-' <iso8601hour> (':'? <iso8601minute>)?
Ensuring Validity#
So far, our dates and times are syntactically valid, but not necessarily semantically.
We can easily create invalid dates such as 2025-02-31
(February 31) or invalid times such as 24:10:00
(10 minutes past midnight - only 24:00:00
is allowed).
We could now add a number of additional rules to check for all these properties, and/or extend the grammar accordingly.
However, we can also simply be lazy and have the existing Python datetime
module check all this for us:
import datetime
def is_valid_iso8601datetime(iso8601datetime: str) -> bool:
"""Return True iff `iso8601datetime` is valid."""
try:
datetime.datetime.fromisoformat(iso8601datetime)
return True
except ValueError:
return False
is_valid_iso8601datetime("2025-01-01T14:00")
True
is_valid_iso8601datetime("2025-02-31")
False
is_valid_iso8601datetime("2000-02-29T00:00:00")
True
With this, we can now add a constraint that limits our generator to only valid dates and times:
iso8601lib += make_constraint("is_valid_iso8601datetime(str(<iso8601datetime>))")
where is_valid_iso8601datetime(str(<iso8601datetime>))
iso8601lib += '''
def is_valid_iso8601datetime(iso8601datetime: str) -> bool:
"""Return True iff `iso8601datetime` is valid."""
try:
datetime.datetime.fromisoformat(iso8601datetime)
return True
except ValueError:
return False
'''
Even Better Validity#
The Python datetime
module has a some limitations, which extend to our spec.
For instance, datetime
does not support 24:00:00
as a valid time:
is_valid_iso8601datetime("2024-12-31T24:00:00")
False
Hence, our iso8601.fan
spec may miss out a number of ISO 8601 features.
The dateutil
alternative provides an ISO 8601 parser without these deficiencies.
Let us redefine is_valid_iso8601datetime()
to make use of the dateutil
parser:
def is_valid_iso8601datetime(iso8601datetime: str) -> bool:
"""Return True iff `iso8601datetime` is valid."""
try:
dateutil.parser.isoparse(iso8601datetime)
return True
except ValueError:
return False
Let us see if the above example now works:
is_valid_iso8601datetime("2024-12-31T24:00:00")
True
This now works! How about the earlier examples?
is_valid_iso8601datetime("2025-01-01T14:00")
True
is_valid_iso8601datetime("2025-02-31")
False
is_valid_iso8601datetime("2000-02-29T00:00:00")
True
These also work.
Let us fix the spec to use dateutil
instead:
iso8601lib = iso8601lib.replace('datetime.datetime.fromisoformat', 'dateutil.parser.isoparse')
iso8601lib = iso8601lib.replace('import datetime', 'import dateutil # See https://dateutil.readthedocs.io');
Fuzzing Dates and Times#
Our ISO 8601 spec is now complete.
Let us write it into a .fan
file, so we can use it for fuzzing:
open('ISO8601.fan', 'w').write(iso8601lib);
Here comes iso8601.fan
in all its glory:
# ISO 8601 date and time
# Generated by docs/ISO8601.md. Do not edit.
import dateutil # See https://dateutil.readthedocs.io
<start> ::= <iso8601datetime>
<iso8601datetime> ::= <iso8601date> ('T' <iso8601time>)?
<iso8601date> ::= <iso8601calendardate> | <iso8601weekdate> |
<iso8601ordinaldate>
<iso8601calendardate> ::= <iso8601year> '-' <iso8601month> ('-' <iso8601day>)?
| <iso8601year> <iso8601month> <iso8601day>
<iso8601year> ::= ('+'|'-')? <digit>{4}
<iso8601month> ::= '01' | '02' | '03' | '04' | '05' | '06' | '07' | '08' | '09'
| '10' | '11' | '12'
<iso8601day> ::= '01' | '02' | '03' | '04' | '05' | '06' | '07' | '08' | '09' |
'10' | '11' | '12' | '13' | '14' | '15' | '16' | '17' | '18' | '19' | '20' |
'21' | '22' | '23' | '24' | '25' | '26' | '27' | '28' | '29' | '30' | '31'
<iso8601weekdate> ::= <iso8601year> '-'? 'W' <iso8601week> ('-'
<iso8601weekday>)?
<iso8601week> ::= '01' | '02' | '03' | '04' | '05' | '06' | '07' | '08' | '09'
| '10' | '11' | '12' | '13' | '14' | '15' | '16' | '17' | '18' | '19' | '20' |
'21' | '22' | '23' | '24' | '25' | '26' | '27' | '28' | '29' | '30' | '31' |
'32' | '33' | '34' | '35' | '36' | '37' | '38' | '39' | '40' | '41' | '42' |
'43' | '44' | '45' | '46' | '47' | '48' | '49' | '50' | '51' | '52' | '53'
<iso8601weekday> ::= '1' | '2' | '3' | '4' | '5' | '6' | '7'
<iso8601ordinaldate> ::= <iso8601year> ('-'? <iso8601ordinalday>)?
<iso8601ordinalday> ::= '001' | '002' | '003' | '004' | '005' | '006' | '007' |
'008' | '009' | '010' | '011' | '012' | '013' | '014' | '015' | '016' | '017' |
'018' | '019' | '020' | '021' | '022' | '023' | '024' | '025' | '026' | '027' |
'028' | '029' | '030' | '031' | '032' | '033' | '034' | '035' | '036' | '037' |
'038' | '039' | '040' | '041' | '042' | '043' | '044' | '045' | '046' | '047' |
'048' | '049' | '050' | '051' | '052' | '053' | '054' | '055' | '056' | '057' |
'058' | '059' | '060' | '061' | '062' | '063' | '064' | '065' | '066' | '067' |
'068' | '069' | '070' | '071' | '072' | '073' | '074' | '075' | '076' | '077' |
'078' | '079' | '080' | '081' | '082' | '083' | '084' | '085' | '086' | '087' |
'088' | '089' | '090' | '091' | '092' | '093' | '094' | '095' | '096' | '097' |
'098' | '099' | '100' | '101' | '102' | '103' | '104' | '105' | '106' | '107' |
'108' | '109' | '110' | '111' | '112' | '113' | '114' | '115' | '116' | '117' |
'118' | '119' | '120' | '121' | '122' | '123' | '124' | '125' | '126'
| '127' |
'128' | '129' | '130' | '131' | '132' | '133' | '134' | '135' | '136' | '137' |
'138' | '139' | '140' | '141' | '142' | '143' | '144' | '145' | '146' | '147' |
'148' | '149' | '150' | '151' | '152' | '153' | '154' | '155' | '156' | '157' |
'158' | '159' | '160' | '161' | '162' | '163' | '164' | '165' | '166' | '167' |
'168' | '169' | '170' | '171' | '172' | '173' | '174' | '175' | '176' | '177' |
'178' | '179' | '180' | '181' | '182' | '183' | '184' | '185' | '186' | '187' |
'188' | '189' | '190' | '191' | '192' | '193' | '194' | '195' | '196' | '197' |
'198' | '199' | '200' | '201' | '202' | '203' | '204' | '205' | '206' | '207' |
'208' | '209' | '210' | '211' | '212' | '213' | '214' | '215' | '216' | '217' |
'218' | '219' | '220' | '221' | '222' | '223' | '224' | '225' | '226' | '227' |
'228' | '229' | '230' | '231' | '232' | '233' | '234' | '235' | '236' | '237' |
'238' | '239' | '240' | '241' | '242' | '243' | '244' | '245' | '246' | '247' |
'248' | '249' | '250' | '251' | '252' | '253' | '254' | '255' | '256' | '257' |
'258' | '259' | '260' | '261' | '262' | '263' | '264' | '265' | '266' | '267' |
'268' | '269' | '270' | '271' | '272' | '273' | '274' | '275' | '276' | '277' |
'278' | '279' | '280' | '281' | '282' | '283' | '284' | '285' | '286' | '287' |
'288' | '289' | '290' | '291' | '292' | '293' | '294' | '295' | '296' | '297' |
'298' | '299' | '300' | '301' | '302' | '303' | '304' | '305' | '306' | '307' |
'308' | '309' | '310' | '311' | '312' | '313' | '314' | '315' | '316' | '317' |
'318' | '319' | '320' | '321' | '322' | '323' | '324' | '325' | '326' | '327' |
'328' | '329' | '330' | '331' | '332' | '333' | '334' | '335' | '336' | '337' |
'338' | '339' | '340' | '341' | '342' | '343' | '344' | '345' | '346' | '347' |
'348' | '349' | '350' | '351' | '352' | '353' | '354' | '355' | '356' | '357' |
'358' | '359' | '360' | '361' | '362' | '363' | '364' | '365' | '366'
<iso8601time> ::= 'T'? <iso8601hour> (':'? <iso8601minute> (':'?
<iso8601second> (('.' | ',') <iso8601fraction>)? )? )? <iso8601timezone>?
# 24:00:00 is allowed to represent midnight at the end of a day
<iso8601hour> ::= '00' | '01' | '02' | '03' | '04' | '05' | '06' | '07' | '08'
| '09' | '10' | '11' | '12' | '13' | '14' | '15' | '16' | '17' | '18' | '19' |
'20' | '21' | '22' | '23' | '24'
<iso8601minute> ::= '00' | '01' | '02' | '03' | '04' | '05' | '06' | '07' |
'08' | '09' | '10' | '11' | '12' | '13' | '14' | '15' | '16' | '17' | '18' |
'19' | '20' | '21' | '22' | '23' | '24' | '25' | '26' | '27' | '28' | '29' |
'30' | '31' | '32' | '33' | '34' | '35' | '36' | '37' | '38' | '39' | '40' |
'41' | '42' | '43' | '44' | '45' | '46' | '47' | '48' | '49' | '50' | '51' |
'52' | '53' | '54' | '55' | '56' | '57' | '58' | '59'
# xx:yy:60 is allowed to represent leap seconds
<iso8601second> ::= '00' | '01' | '02' | '03' | '04' | '05' | '06' | '07' |
'08' | '09' | '10' | '11' | '12' | '13' | '14' | '15' | '16' | '17' | '18' |
'19' | '20' | '21' | '22' | '23' | '24' | '25' | '26' | '27' | '28' | '29' |
'30' | '31' | '32' | '33' | '34' | '35' | '36' | '37' | '38' | '39' | '40' |
'41' | '42' | '43' | '44' | '45' | '46' | '47' | '48' | '49' | '50' | '51' |
'52' | '53' | '54' | '55' | '56' | '57' | '58' | '59' | '60'
<iso8601fraction> ::= <digit>+
<iso8601timezone> ::= 'Z' | '+' <iso8601hour> (':'? <iso8601minute>)? | '-'
<iso8601hour> (':'? <iso8601minute>)?
where is_valid_iso8601datetime(str(<iso8601datetime>))
def is_valid_iso8601datetime(iso8601datetime: str) -> bool:
"""Return True iff `iso8601datetime` is valid."""
try:
dateutil.parser.isoparse(iso8601datetime)
return True
except ValueError:
return False
With this, we can now create a bunch of random date/time elements:
$ fandango fuzz -f iso8601.fan -n 10
6899-03-28
40570303
7306-11-14
1405-083T17-08
7550-W49-5T14:27
6964358
5201W27
4190
3661-01
3165-04-28
And we can use additional constraints to further narrow down date intervals:
$ fandango fuzz -f ISO8601.fan -n 10 -c 'int(<iso8601year>) > 1950 and int(<iso8601year>) < 2000'
1969-09-12
1953-04-18
1998-W31-5
1967-W24
1953-04-30
1970-12-22
1964-187
1988-W35-1
1989186T22:47:54,42712214087357775249526290419518242533119305513952671-23:32
1976-12-22
Or produce individual elements, again with individual constraints:
$ fandango fuzz -f ISO8601.fan -n 10 --start-symbol='<iso8601time>' -c '<iso8601hour> == "00"'
fandango:WARNING: Symbol <start> defined, but not used
00Z
T00
T004114
T00:44
0010
00:49:56,68-00
T00Z
0000-00
00:37:10Z
00
Try out more constraints for yourself!
The generated ISO8601.fan
file is available for download.