PageRenderTime 53ms CodeModel.GetById 26ms RepoModel.GetById 1ms app.codeStats 0ms

/apps/external_apps/arcade/util.py

http://django-hotclub.googlecode.com/
Python | 113 lines | 105 code | 1 blank | 7 comment | 0 complexity | 4d360afd79befa14bc829738fdf87311 MD5 | raw file
Possible License(s): BSD-3-Clause, GPL-2.0, BSD-2-Clause, Apache-2.0, AGPL-3.0
  1. """
  2. ISO8601 Date Parsing included from http://code.google.com/p/pyiso8601/
  3. """
  4. """ISO 8601 date time string parsing
  5. Basic usage:
  6. >>> import iso8601
  7. >>> iso8601.parse_date("2007-01-25T12:00:00Z")
  8. datetime.datetime(2007, 1, 25, 12, 0, tzinfo=<iso8601.iso8601.Utc ...>)
  9. >>>
  10. """
  11. from datetime import datetime, timedelta, tzinfo
  12. import re
  13. # Adapted from http://delete.me.uk/2005/03/iso8601.html
  14. ISO8601_REGEX = re.compile(r"(?P<year>[0-9]{4})(-(?P<month>[0-9]{1,2})(-(?P<day>[0-9]{1,2})"
  15. r"((?P<separator>.)(?P<hour>[0-9]{2}):(?P<minute>[0-9]{2})(:(?P<second>[0-9]{2})(\.(?P<fraction>[0-9]+))?)?"
  16. r"(?P<timezone>Z|(([-+])([0-9]{2}):([0-9]{2})))?)?)?)?"
  17. )
  18. TIMEZONE_REGEX = re.compile("(?P<prefix>[+-])(?P<hours>[0-9]{2}).(?P<minutes>[0-9]{2})")
  19. class ParseError(Exception):
  20. """Raised when there is a problem parsing a date string"""
  21. # Yoinked from python docs
  22. ZERO = timedelta(0)
  23. class Utc(tzinfo):
  24. """UTC
  25. """
  26. def utcoffset(self, dt):
  27. return ZERO
  28. def tzname(self, dt):
  29. return "UTC"
  30. def dst(self, dt):
  31. return ZERO
  32. UTC = Utc()
  33. class FixedOffset(tzinfo):
  34. """Fixed offset in hours and minutes from UTC
  35. """
  36. def __init__(self, offset_hours, offset_minutes, name):
  37. self.__offset = timedelta(hours=offset_hours, minutes=offset_minutes)
  38. self.__name = name
  39. def utcoffset(self, dt):
  40. return self.__offset
  41. def tzname(self, dt):
  42. return self.__name
  43. def dst(self, dt):
  44. return ZERO
  45. def __repr__(self):
  46. return "<FixedOffset %r>" % self.__name
  47. def parse_timezone(tzstring, default_timezone=UTC):
  48. """Parses ISO 8601 time zone specs into tzinfo offsets
  49. """
  50. if tzstring == "Z":
  51. return default_timezone
  52. # This isn't strictly correct, but it's common to encounter dates without
  53. # timezones so I'll assume the default (which defaults to UTC).
  54. # Addresses issue 4.
  55. if tzstring is None:
  56. return default_timezone
  57. m = TIMEZONE_REGEX.match(tzstring)
  58. prefix, hours, minutes = m.groups()
  59. hours, minutes = int(hours), int(minutes)
  60. if prefix == "-":
  61. hours = -hours
  62. minutes = -minutes
  63. return FixedOffset(hours, minutes, tzstring)
  64. def parse_date(datestring, default_timezone=UTC):
  65. """Parses ISO 8601 dates into datetime objects
  66. The timezone is parsed from the date string. However it is quite common to
  67. have dates without a timezone (not strictly correct). In this case the
  68. default timezone specified in default_timezone is used. This is UTC by
  69. default.
  70. """
  71. if not isinstance(datestring, basestring):
  72. raise ParseError("Expecting a string %r" % datestring)
  73. m = ISO8601_REGEX.match(datestring)
  74. if not m:
  75. raise ParseError("Unable to parse date string %r" % datestring)
  76. groups = m.groupdict()
  77. tz = parse_timezone(groups["timezone"], default_timezone=default_timezone)
  78. if groups["fraction"] is None:
  79. groups["fraction"] = 0
  80. else:
  81. groups["fraction"] = int(float("0.%s" % groups["fraction"]) * 1e6)
  82. return datetime(int(groups["year"]), int(groups["month"]), int(groups["day"]),
  83. int(groups["hour"]), int(groups["minute"]), int(groups["second"]),
  84. int(groups["fraction"]), tz)
  85. def iso8601(d):
  86. if isinstance(d, _datetime.date) and not isinstance(d, _datetime.datetime):
  87. return d.strftime('%Y-%m-%d')
  88. if d.tzinfo is None:
  89. d = _datetime.datetime(*_time.gmtime(_time.mktime(d.timetuple()))[:7])
  90. else:
  91. d = d.astimezone(_pytz.utc)
  92. return d.strftime('%Y-%m-%dT%H:%M:%SZ')