croniter 6.2.2


pip install croniter

  Latest version

Released: Mar 15, 2026

Project Links

Meta
Author: Matsumoto Taichi, kiorky, Ash Berlin-Taylor, Jarek Potiuk
Requires Python: >=3.9

Classifiers

Development Status
  • 4 - Beta

Intended Audience
  • Developers

License
  • OSI Approved :: MIT License

Operating System
  • POSIX

Programming Language
  • Python
  • Python :: 3.9
  • Python :: 3.10
  • Python :: 3.11
  • Python :: 3.12
  • Python :: 3.13

Introduction

croniter provides iteration for the datetime object with a cron like format.

                      _ _
  ___ _ __ ___  _ __ (_) |_ ___ _ __
 / __| '__/ _ \| '_ \| | __/ _ \ '__|
| (__| | | (_) | | | | | ||  __/ |
 \___|_|  \___/|_| |_|_|\__\___|_|

Website: https://github.com/pallets-eco/croniter

Build Badge

https://github.com/pallets-eco/croniter/actions/workflows/cicd.yml/badge.svg

Pallets Community Ecosystem

Usage

A simple example:

>>> from croniter import croniter
>>> from datetime import datetime
>>> base = datetime(2010, 1, 25, 4, 46)
>>> iter = croniter('*/5 * * * *', base)  # every 5 minutes
>>> print(iter.get_next(datetime))   # 2010-01-25 04:50:00
>>> print(iter.get_next(datetime))   # 2010-01-25 04:55:00
>>> print(iter.get_next(datetime))   # 2010-01-25 05:00:00
>>>
>>> iter = croniter('2 4 * * mon,fri', base)  # 04:02 on every Monday and Friday
>>> print(iter.get_next(datetime))   # 2010-01-26 04:02:00
>>> print(iter.get_next(datetime))   # 2010-01-30 04:02:00
>>> print(iter.get_next(datetime))   # 2010-02-02 04:02:00
>>>
>>> iter = croniter('2 4 1 * wed', base)  # 04:02 on every Wednesday OR on 1st day of month
>>> print(iter.get_next(datetime))   # 2010-01-27 04:02:00
>>> print(iter.get_next(datetime))   # 2010-02-01 04:02:00
>>> print(iter.get_next(datetime))   # 2010-02-03 04:02:00
>>>
>>> iter = croniter('2 4 1 * wed', base, day_or=False)  # 04:02 on every 1st day of the month if it is a Wednesday
>>> print(iter.get_next(datetime))   # 2010-09-01 04:02:00
>>> print(iter.get_next(datetime))   # 2010-12-01 04:02:00
>>> print(iter.get_next(datetime))   # 2011-06-01 04:02:00
>>>
>>> iter = croniter('0 0 * * sat#1,sun#2', base)  # 1st Saturday, and 2nd Sunday of the month
>>> print(iter.get_next(datetime))   # 2010-02-06 00:00:00
>>>
>>> iter = croniter('0 0 * * 5#3,L5', base)  # 3rd and last Friday of the month
>>> print(iter.get_next(datetime))   # 2010-01-29 00:00:00
>>> print(iter.get_next(datetime))   # 2010-02-19 00:00:00

All you need to know is how to use the constructor and the get_next method, the signature of these methods are listed below:

>>> def __init__(self, cron_format, start_time=time.time(), day_or=True)

croniter iterates along with cron_format from start_time. cron_format is min hour day month day_of_week, you can refer to http://en.wikipedia.org/wiki/Cron for more details. The day_or switch is used to control how croniter handles day and day_of_week entries. Default option is the cron behaviour, which connects those values using OR. If the switch is set to False, the values are connected using AND. This behaves like fcron and enables you to e.g. define a job that executes each 2nd Friday of a month by setting the days of month and the weekday.

>>> def get_next(self, ret_type=float)

get_next calculates the next value according to the cron expression and returns an object of type ret_type. ret_type should be a float or a datetime object.

Supported added for get_prev method. (>= 0.2.0):

>>> base = datetime(2010, 8, 25)
>>> itr = croniter('0 0 1 * *', base)
>>> print(itr.get_prev(datetime))  # 2010-08-01 00:00:00
>>> print(itr.get_prev(datetime))  # 2010-07-01 00:00:00
>>> print(itr.get_prev(datetime))  # 2010-06-01 00:00:00

You can validate your crons using is_valid class method. (>= 0.3.18):

>>> croniter.is_valid('0 0 1 * *')  # True
>>> croniter.is_valid('0 wrong_value 1 * *')  # False

Strict validation

By default, is_valid and expand only check that each field is within its own range (e.g. day 1-31, month 1-12). They do not cross-validate fields, so expressions like 0 0 31 2 * (February 31st) are considered valid even though they can never match a real date.

Use strict=True to enable cross-field validation that rejects impossible day/month combinations:

>>> croniter.is_valid('0 0 31 2 *')  # True (default: syntax only)
>>> croniter.is_valid('0 0 31 2 *', strict=True)  # False (Feb has at most 29 days)
>>> croniter.is_valid('0 0 31 4 *', strict=True)  # False (Apr has 30 days)
>>> croniter.is_valid('0 0 30 1,2 *', strict=True)  # True (day 30 is valid in Jan)

When strict=True and the expression does not include a year field, February is assumed to have 29 days (since leap years exist):

>>> croniter.is_valid('0 0 29 2 *', strict=True)  # True (leap years exist)

If the expression includes a year field (7-field format), leap year status is determined from the specified years:

>>> croniter.is_valid('0 0 29 2 * 0 2024', strict=True)  # True (2024 is a leap year)
>>> croniter.is_valid('0 0 29 2 * 0 2023', strict=True)  # False (2023 is not)
>>> croniter.is_valid('0 0 29 2 * 0 2023-2025', strict=True)  # True (2024 in range is a leap year)

For 5- or 6-field expressions, you can pass strict_year to provide the year(s) for leap year checking without adding a year field to the expression:

>>> croniter.is_valid('0 0 29 2 *', strict=True, strict_year=2024)  # True (leap year)
>>> croniter.is_valid('0 0 29 2 *', strict=True, strict_year=2023)  # False (not a leap year)
>>> croniter.is_valid('0 0 29 2 *', strict=True, strict_year=[2023, 2024])  # True (2024 is a leap year)

The strict and strict_year parameters are also available on expand():

>>> croniter.expand('0 0 31 2 *', strict=True)  # raises CroniterBadCronError

Nearest weekday (W)

The W character is supported in the day-of-month field to specify the nearest weekday (Monday-Friday) to the given day. Both nW and Wn formats are accepted:

>>> base = datetime(2024, 6, 1)
>>> itr = croniter('0 9 15W * *', base)
>>> itr.get_next(datetime)   # Jun 15 2024 is Saturday -> fires on Friday 14th
datetime.datetime(2024, 6, 14, 9, 0)

Rules:

  • If the specified day falls on a weekday, the trigger fires on that day.

  • If the specified day falls on Saturday, the trigger fires on the preceding Friday.

  • If the specified day falls on Sunday, the trigger fires on the following Monday.

  • The nearest weekday never crosses month boundaries. If the 1st is a Saturday, the trigger fires on Monday the 3rd. If the last day of the month is a Sunday, the trigger fires on the preceding Friday.

  • W can only be used with a single day value, not in a range or list.

Examples:

>>> croniter('0 9 1W * *', datetime(2024, 5, 31)).get_next(datetime)   # Jun 1 is Sat -> Mon 3rd
datetime.datetime(2024, 6, 3, 9, 0)
>>> croniter('0 9 W15 * *', datetime(2024, 1, 1)).get_next(datetime)   # Wn format also works
datetime.datetime(2024, 1, 15, 9, 0)

Day-of-month and day-of-week

When both the day-of-month and day-of-week fields are restricted (not *), the default POSIX cron behavior is to match when either field matches (OR). This is controlled by the day_or parameter:

>>> # OR (default): fires on every Wednesday OR on the 1st of the month
>>> iter = croniter('2 4 1 * wed', datetime(2010, 1, 25))
>>> print(iter.get_next(datetime))   # 2010-01-27 04:02:00 (Wed)
>>> print(iter.get_next(datetime))   # 2010-02-01 04:02:00 (1st)

>>> # AND: fires only on the 1st of the month IF it is a Wednesday
>>> iter = croniter('2 4 1 * wed', datetime(2010, 1, 25), day_or=False)
>>> print(iter.get_next(datetime))   # 2010-09-01 04:02:00 (Wed AND 1st)

This can be used to express patterns like “the first Tuesday of the month” by combining a day range with a weekday:

>>> # First Tuesday of each month: days 1-7 AND Tuesday
>>> iter = croniter('1 1 1-7 * 2', datetime(2024, 7, 12), day_or=False)
>>> print(iter.get_next(datetime))   # 2024-08-06 01:01:00
>>> print(iter.get_next(datetime))   # 2024-09-03 01:01:00
>>> print(iter.get_next(datetime))   # 2024-10-01 01:01:00
>>> print(iter.get_next(datetime))   # 2024-11-05 01:01:00

Vixie cron bug compatibility

Some vixie/ISC cron implementations have a known bug where expressions that start with * in the day-of-month or day-of-week field (e.g. */32,1-7) use AND logic instead of OR, even though those fields are technically restricted.

If you need to replicate this behavior (e.g. */32,1-7 as a hack for days 1-7), use implement_cron_bug=True:

>>> iter = croniter('1 1 */32,1-7 * 2', datetime(2024, 7, 12), implement_cron_bug=True)
>>> print(iter.get_next(datetime))   # 2024-08-06 01:01:00
>>> print(iter.get_next(datetime))   # 2024-09-03 01:01:00

About DST

Be sure to init your croniter instance with a TZ aware datetime for this to work!

Example using zoneinfo:

>>> import zoneinfo
>>> tz = zoneinfo.ZoneInfo("Europe/Berlin")
>>> local_date = datetime(2017, 3, 26, tzinfo=tz)
>>> val = croniter('0 0 * * *', local_date).get_next(datetime)

Example using pytz:

>>> import pytz
>>> tz = pytz.timezone("Europe/Paris")
>>> local_date = tz.localize(datetime(2017, 3, 26))
>>> val = croniter('0 0 * * *', local_date).get_next(datetime)

Example using python_dateutil:

>>> import dateutil.tz
>>> tz = dateutil.tz.gettz('Asia/Tokyo')
>>> local_date = datetime(2017, 3, 26, tzinfo=tz)
>>> val = croniter('0 0 * * *', local_date).get_next(datetime)

Example using python built in module:

>>> from datetime import datetime, timezone
>>> local_date = datetime(2017, 3, 26, tzinfo=timezone.utc)
>>> val = croniter('0 0 * * *', local_date).get_next(datetime)

About second repeats

Croniter is able to do second repetition crontabs form and by default seconds are the 6th field:

>>> base = datetime(2012, 4, 6, 13, 26, 10)
>>> itr = croniter('* * * * * 15,25', base)
>>> itr.get_next(datetime) # 4/6 13:26:15
>>> itr.get_next(datetime) # 4/6 13:26:25
>>> itr.get_next(datetime) # 4/6 13:27:15

You can also note that this expression will repeat every second from the start datetime.:

>>> croniter('* * * * * *', local_date).get_next(datetime)

You can also use seconds as first field:

>>> itr = croniter('15,25 * * * * *', base, second_at_beginning=True)

About year

Croniter also support year field. Year presents at the seventh field, which is after second repetition. The range of year field is from 1970 to 2099. To ignore second repetition, simply set second to 0 or any other const:

>>> base = datetime(2012, 4, 6, 2, 6, 59)
>>> itr = croniter('0 0 1 1 * 0 2020/2', base)
>>> itr.get_next(datetime) # 2020 1/1 0:0:0
>>> itr.get_next(datetime) # 2022 1/1 0:0:0
>>> itr.get_next(datetime) # 2024 1/1 0:0:0

Support for start_time shifts

See https://github.com/pallets-eco/croniter/pull/76, You can set start_time=, then expand_from_start_time=True for your generations to be computed from start_time instead of calendar days:

>>> from pprint import pprint
>>> iter = croniter('0 0 */7 * *', start_time=datetime(2024, 7, 11), expand_from_start_time=True);pprint([iter.get_next(datetime) for a in range(10)])
[datetime.datetime(2024, 7, 18, 0, 0),
 datetime.datetime(2024, 7, 25, 0, 0),
 datetime.datetime(2024, 8, 4, 0, 0),
 datetime.datetime(2024, 8, 11, 0, 0),
 datetime.datetime(2024, 8, 18, 0, 0),
 datetime.datetime(2024, 8, 25, 0, 0),
 datetime.datetime(2024, 9, 4, 0, 0),
 datetime.datetime(2024, 9, 11, 0, 0),
 datetime.datetime(2024, 9, 18, 0, 0),
 datetime.datetime(2024, 9, 25, 0, 0)]
>>> # INSTEAD OF THE DEFAULT BEHAVIOR:
>>> iter = croniter('0 0 */7 * *', start_time=datetime(2024, 7, 11), expand_from_start_time=False);pprint([iter.get_next(datetime) for a in range(10)])
[datetime.datetime(2024, 7, 15, 0, 0),
 datetime.datetime(2024, 7, 22, 0, 0),
 datetime.datetime(2024, 7, 29, 0, 0),
 datetime.datetime(2024, 8, 1, 0, 0),
 datetime.datetime(2024, 8, 8, 0, 0),
 datetime.datetime(2024, 8, 15, 0, 0),
 datetime.datetime(2024, 8, 22, 0, 0),
 datetime.datetime(2024, 8, 29, 0, 0),
 datetime.datetime(2024, 9, 1, 0, 0),
 datetime.datetime(2024, 9, 8, 0, 0)]

Testing if a date matches a crontab

Test for a match with (>=0.3.32):

>>> croniter.match("0 0 * * *", datetime(2019, 1, 14, 0, 0, 0, 0))
True
>>> croniter.match("0 0 * * *", datetime(2019, 1, 14, 0, 2, 0, 0))
False
>>>
>>> croniter.match("2 4 1 * wed", datetime(2019, 1, 1, 4, 2, 0, 0)) # 04:02 on every Wednesday OR on 1st day of month
True
>>> croniter.match("2 4 1 * wed", datetime(2019, 1, 1, 4, 2, 0, 0), day_or=False) # 04:02 on every 1st day of the month if it is a Wednesday
False

By default, match uses a precision of 60 seconds for standard 5-field cron expressions and 1 second for 6-field expressions (with seconds). This means a datetime up to 60 (or 1) seconds after the scheduled time will still match.

You can override this with the precision_in_seconds parameter:

>>> # With default precision (60s), 59 seconds past midnight still matches
>>> croniter.match("0 0 * * *", datetime(2019, 1, 14, 0, 0, 59, 0))
True
>>> # With precision=1, only an exact match within 1 second works
>>> croniter.match("0 0 * * *", datetime(2019, 1, 14, 0, 0, 59, 0), precision_in_seconds=1)
False

Testing if a crontab matches in datetime range

Test for a match_range with (>=2.0.3):

>>> croniter.match_range("0 0 * * *", datetime(2019, 1, 13, 0, 59, 0, 0), datetime(2019, 1, 14, 0, 1, 0, 0))
True
>>> croniter.match_range("0 0 * * *", datetime(2019, 1, 13, 0, 1, 0, 0), datetime(2019, 1, 13, 0, 59, 0, 0))
False
>>> croniter.match_range("2 4 1 * wed", datetime(2019, 1, 1, 3, 2, 0, 0), datetime(2019, 1, 1, 5, 1, 0, 0))
# 04:02 on every Wednesday OR on 1st day of month
True
>>> croniter.match_range("2 4 1 * wed", datetime(2019, 1, 1, 3, 2, 0, 0), datetime(2019, 1, 1, 5, 2, 0, 0), day_or=False)
# 04:02 on every 1st day of the month if it is a Wednesday
False

The precision_in_seconds parameter is also available on match_range.

Gaps between date matches

For performance reasons, croniter limits the amount of CPU cycles spent attempting to find the next match. Starting in v0.3.35, this behavior is configurable via the max_years_between_matches parameter, and the default window has been increased from 1 year to 50 years.

The defaults should be fine for many use cases. Applications that evaluate multiple cron expressions or handle cron expressions from untrusted sources or end-users should use this parameter. Iterating over sparse cron expressions can result in increased CPU consumption or a raised CroniterBadDateError exception which indicates that croniter has given up attempting to find the next (or previous) match. Explicitly specifying max_years_between_matches provides a way to limit CPU utilization and simplifies the iterable interface by eliminating the need for CroniterBadDateError. The difference in the iterable interface is based on the reasoning that whenever max_years_between_matches is explicitly agreed upon, there is no need for croniter to signal that it has given up; simply stopping the iteration is preferable.

This example matches 4 AM Friday, January 1st. Since January 1st isn’t often a Friday, there may be a few years between each occurrence. Setting the limit to 15 years ensures all matches:

>>> it = croniter("0 4 1 1 fri", datetime(2000,1,1), day_or=False, max_years_between_matches=15).all_next(datetime)
>>> for i in range(5):
...     print(next(it))
...
2010-01-01 04:00:00
2016-01-01 04:00:00
2021-01-01 04:00:00
2027-01-01 04:00:00
2038-01-01 04:00:00

However, when only concerned with dates within the next 5 years, simply set max_years_between_matches=5 in the above example. This will result in no matches found, but no additional cycles will be wasted on unwanted matches far in the future.

Iterating over a range using cron

Find matches within a range using the croniter_range() function. This is much like the builtin range(start,stop,step) function, but for dates. The step argument is a cron expression. Added in (>=0.3.34)

List the first Saturday of every month in 2019:

>>> from croniter import croniter_range
>>> for dt in croniter_range(datetime(2019, 1, 1), datetime(2019, 12, 31), "0 0 * * sat#1"):
>>>     print(dt)

Hashed expressions

croniter supports Jenkins-style hashed expressions, using the “H” definition keyword and the required hash_id keyword argument. Hashed expressions remain consistent, given the same hash_id, but different hash_ids will evaluate completely different to each other. This allows, for example, for an even distribution of differently-named jobs without needing to manually spread them out.

>>> itr = croniter("H H * * *", hash_id="hello")
>>> itr.get_next(datetime)
datetime.datetime(2021, 4, 10, 11, 10)
>>> itr.get_next(datetime)
datetime.datetime(2021, 4, 11, 11, 10)
>>> itr = croniter("H H * * *", hash_id="hello")
>>> itr.get_next(datetime)
datetime.datetime(2021, 4, 10, 11, 10)
>>> itr = croniter("H H * * *", hash_id="bonjour")
>>> itr.get_next(datetime)
datetime.datetime(2021, 4, 10, 20, 52)

Random expressions

Random “R” definition keywords are supported, and remain consistent only within their croniter() instance.

>>> itr = croniter("R R * * *")
>>> itr.get_next(datetime)
datetime.datetime(2021, 4, 10, 22, 56)
>>> itr.get_next(datetime)
datetime.datetime(2021, 4, 11, 22, 56)
>>> itr = croniter("R R * * *")
>>> itr.get_next(datetime)
datetime.datetime(2021, 4, 11, 4, 19)

Note about Ranges

Note that as a deviation from cron standard, croniter is somehow laxist with ranges and will allow ranges of Jan-Dec, & Sun-Sat in reverse way and interpret them as following examples:

  • Apr-Jan: from April to january

  • Sat-Sun: Saturday, Sunday

  • Wed-Sun: Wednesday to Saturday, Sunday

Please note that if a /step is given, it will be respected.

Note about Sunday

Note that as a deviation from cron standard, croniter like numerous cron implementations supports SUNDAY to be expressed as DAY7, allowing such expressions:

  • 0 0 * * 7

  • 0 0 * * 6-7

  • 0 0 * * 6,7

Keyword expressions

Vixie cron-style “@” keyword expressions are supported. What they evaluate to depends on whether you supply hash_id: no hash_id corresponds to Vixie cron definitions (exact times, minute resolution), while with hash_id corresponds to Jenkins definitions (hashed within the period, second resolution).

Keyword

No hash_id

With hash_id

@midnight

0 0 * * *

H H(0-2) * * * H

@hourly

0 * * * *

H * * * * H

@daily

0 0 * * *

H H * * * H

@weekly

0 0 * * 0

H H * * H H

@monthly

0 0 1 * *

H H H * * H

@yearly

0 0 1 1 *

H H H H * H

@annually

0 0 1 1 *

H H H H * H

Upgrading

To 2.0.0

  • Install or upgrade pytz by using version specified requirements/base.txt if you have it installed <=2021.1.

Develop this package

git clone https://github.com/pallets-eco/croniter.git
cd croniter
virtualenv --no-site-packages venv3
venv3/bin/pip install --upgrade -r requirements/test.txt -r requirements/lint.txt -r requirements/format.txt -r requirements/tox.txt
venv3/bin/black src/
venv3/bin/isort src/
venv3/bin/tox --current-env -e fmt,lint,test

Make a new release

We use zest.fullreleaser, a great release infrastructure.

Do and follow these instructions

venv3/bin/pip install --upgrade -r requirements/release.txt
./release.sh

Contributors

Thanks to all who have contributed to this project! If you have contributed and your name is not listed below please let us know.

  • Aarni Koskela (akx)

  • ashb

  • bdrung

  • chris-baynes

  • djmitche

  • evanpurkhiser

  • GreatCombinator

  • Hinnack

  • ipartola

  • jlsandell

  • kiorky

  • lowell80 (Kintyre)

  • mag009

  • mrmachine

  • potiuk

  • Ryan Finnie (rfinnie)

  • salitaba

  • scop

  • shazow

  • yuzawa-san

  • zed2015

6.2.2 Mar 15, 2026
6.2.1 Mar 15, 2026
6.2.0 Mar 14, 2026
6.1.0 Mar 14, 2026
6.1.0rc1 Mar 14, 2026
6.0.0 Dec 17, 2024
5.0.1 Oct 29, 2024
4.0.0 Oct 28, 2024
3.0.4 Oct 25, 2024
3.0.3 Jul 26, 2024
3.0.2 Jul 26, 2024
3.0.1 Jul 25, 2024
3.0.0 Jul 23, 2024
2.0.7 Jul 16, 2024
2.0.6 Jul 16, 2024
2.0.5 Apr 20, 2024
2.0.4 Apr 20, 2024
2.0.3 Mar 19, 2024
2.0.2 Feb 29, 2024
2.0.1 Oct 11, 2023
2.0.0 Oct 10, 2023
1.4.1 Jun 15, 2023
1.4.0 Jun 15, 2023
1.3.15 May 25, 2023
1.3.14 Apr 12, 2023
1.3.13 Apr 12, 2023
1.3.12 Apr 12, 2023
1.3.11 Apr 12, 2023
1.3.10 Apr 07, 2023
1.3.8 Nov 22, 2022
1.3.7 Sep 06, 2022
1.3.6 Sep 06, 2022
1.3.5 May 14, 2022
1.3.4 Feb 18, 2022
1.3.3 Feb 18, 2022
1.3.2 Feb 18, 2022
1.3.1 Feb 15, 2022
1.3.0 Feb 15, 2022
1.2.0 Jan 14, 2022
1.1.0 Dec 03, 2021
1.0.15 Jun 25, 2021
1.0.14 Jun 25, 2021
1.0.13 May 06, 2021
1.0.13.dev0 Apr 14, 2021
1.0.12 Apr 13, 2021
1.0.11 Apr 07, 2021
1.0.10 Mar 25, 2021
1.0.9 Mar 23, 2021
1.0.8 Mar 06, 2021
1.0.7 Mar 02, 2021
1.0.6 Feb 01, 2021
1.0.5 Jan 29, 2021
1.0.4 Jan 29, 2021
1.0.3 Jan 29, 2021
1.0.2 Jan 19, 2021
1.0.1 Jan 06, 2021
0.3.37 Dec 31, 2020
0.3.36 Nov 02, 2020
0.3.35 Oct 11, 2020
0.3.34 Jun 19, 2020
0.3.33 Jun 15, 2020
0.3.32 May 27, 2020
0.3.31 Jan 02, 2020
0.3.30 Apr 20, 2019
0.3.29 Mar 26, 2019
0.3.28 Mar 19, 2019
0.3.25 Aug 07, 2018
0.3.24 Jun 20, 2018
0.3.23 May 23, 2018
0.3.22 May 16, 2018
0.3.20 Nov 06, 2017
0.3.19 Aug 31, 2017
0.3.18 Aug 31, 2017
0.3.17 May 22, 2017
0.3.16 Mar 15, 2017
0.3.15 Feb 16, 2017
0.3.14 Jan 25, 2017
0.3.13 Nov 01, 2016
0.3.12 Mar 10, 2016
0.3.11 Jan 13, 2016
0.3.10 Nov 29, 2015
0.3.9 Nov 19, 2015
0.3.8 Jun 23, 2015
0.3.7 Jun 01, 2015
0.3.6 May 29, 2015
0.3.5 Aug 01, 2014
0.3.4 Jan 29, 2014
0.3.3 Sep 29, 2012
0.3.2 Sep 26, 2012
0.3.1 Sep 26, 2012
0.3.0 Apr 06, 2012
0.2.7 Feb 24, 2012
0.2.6 Oct 29, 2011
0.2.5 Jan 27, 2011
0.2.4 Jan 26, 2011
0.2.2 Jan 26, 2011
0.2.0 Aug 25, 2010
0.1.6 Aug 10, 2010
0.1.5 Jan 27, 2010
0.1.4 Jan 27, 2010
0.1.3 Jan 24, 2010
0.1.2 Jan 24, 2010
0.1.1 Jan 24, 2010
0.1.4.macosx Jan 27, 2010

Wheel compatibility matrix

Platform Python 3
any

Files in release

Extras: None
Dependencies:
python-dateutil