123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296 |
- # -*- coding: utf-8 -*-
- """
- Implements the :class:`ArrowFactory <arrow.factory.ArrowFactory>` class,
- providing factory methods for common :class:`Arrow <arrow.arrow.Arrow>`
- construction scenarios.
- """
- from __future__ import absolute_import
- import calendar
- from datetime import date, datetime
- from datetime import tzinfo as dt_tzinfo
- from time import struct_time
- from dateutil import tz as dateutil_tz
- from arrow import parser
- from arrow.arrow import Arrow
- from arrow.util import is_timestamp, iso_to_gregorian, isstr
- class ArrowFactory(object):
- """ A factory for generating :class:`Arrow <arrow.arrow.Arrow>` objects.
- :param type: (optional) the :class:`Arrow <arrow.arrow.Arrow>`-based class to construct from.
- Defaults to :class:`Arrow <arrow.arrow.Arrow>`.
- """
- def __init__(self, type=Arrow):
- self.type = type
- def get(self, *args, **kwargs):
- """ Returns an :class:`Arrow <arrow.arrow.Arrow>` object based on flexible inputs.
- :param locale: (optional) a ``str`` specifying a locale for the parser. Defaults to
- 'en_us'.
- :param tzinfo: (optional) a :ref:`timezone expression <tz-expr>` or tzinfo object.
- Replaces the timezone unless using an input form that is explicitly UTC or specifies
- the timezone in a positional argument. Defaults to UTC.
- Usage::
- >>> import arrow
- **No inputs** to get current UTC time::
- >>> arrow.get()
- <Arrow [2013-05-08T05:51:43.316458+00:00]>
- **None** to also get current UTC time::
- >>> arrow.get(None)
- <Arrow [2013-05-08T05:51:49.016458+00:00]>
- **One** :class:`Arrow <arrow.arrow.Arrow>` object, to get a copy.
- >>> arw = arrow.utcnow()
- >>> arrow.get(arw)
- <Arrow [2013-10-23T15:21:54.354846+00:00]>
- **One** ``float`` or ``int``, convertible to a floating-point timestamp, to get
- that timestamp in UTC::
- >>> arrow.get(1367992474.293378)
- <Arrow [2013-05-08T05:54:34.293378+00:00]>
- >>> arrow.get(1367992474)
- <Arrow [2013-05-08T05:54:34+00:00]>
- **One** ISO 8601-formatted ``str``, to parse it::
- >>> arrow.get('2013-09-29T01:26:43.830580')
- <Arrow [2013-09-29T01:26:43.830580+00:00]>
- **One** ISO 8601-formatted ``str``, in basic format, to parse it::
- >>> arrow.get('20160413T133656.456289')
- <Arrow [2016-04-13T13:36:56.456289+00:00]>
- **One** ``tzinfo``, to get the current time **converted** to that timezone::
- >>> arrow.get(tz.tzlocal())
- <Arrow [2013-05-07T22:57:28.484717-07:00]>
- **One** naive ``datetime``, to get that datetime in UTC::
- >>> arrow.get(datetime(2013, 5, 5))
- <Arrow [2013-05-05T00:00:00+00:00]>
- **One** aware ``datetime``, to get that datetime::
- >>> arrow.get(datetime(2013, 5, 5, tzinfo=tz.tzlocal()))
- <Arrow [2013-05-05T00:00:00-07:00]>
- **One** naive ``date``, to get that date in UTC::
- >>> arrow.get(date(2013, 5, 5))
- <Arrow [2013-05-05T00:00:00+00:00]>
- **One** time.struct time::
- >>> arrow.get(gmtime(0))
- <Arrow [1970-01-01T00:00:00+00:00]>
- **One** iso calendar ``tuple``, to get that week date in UTC::
- >>> arrow.get((2013, 18, 7))
- <Arrow [2013-05-05T00:00:00+00:00]>
- **Two** arguments, a naive or aware ``datetime``, and a replacement
- :ref:`timezone expression <tz-expr>`::
- >>> arrow.get(datetime(2013, 5, 5), 'US/Pacific')
- <Arrow [2013-05-05T00:00:00-07:00]>
- **Two** arguments, a naive ``date``, and a replacement
- :ref:`timezone expression <tz-expr>`::
- >>> arrow.get(date(2013, 5, 5), 'US/Pacific')
- <Arrow [2013-05-05T00:00:00-07:00]>
- **Two** arguments, both ``str``, to parse the first according to the format of the second::
- >>> arrow.get('2013-05-05 12:30:45 America/Chicago', 'YYYY-MM-DD HH:mm:ss ZZZ')
- <Arrow [2013-05-05T12:30:45-05:00]>
- **Two** arguments, first a ``str`` to parse and second a ``list`` of formats to try::
- >>> arrow.get('2013-05-05 12:30:45', ['MM/DD/YYYY', 'YYYY-MM-DD HH:mm:ss'])
- <Arrow [2013-05-05T12:30:45+00:00]>
- **Three or more** arguments, as for the constructor of a ``datetime``::
- >>> arrow.get(2013, 5, 5, 12, 30, 45)
- <Arrow [2013-05-05T12:30:45+00:00]>
- """
- arg_count = len(args)
- locale = kwargs.pop("locale", "en_us")
- tz = kwargs.get("tzinfo", None)
- # if kwargs given, send to constructor unless only tzinfo provided
- if len(kwargs) > 1:
- arg_count = 3
- # tzinfo kwarg is not provided
- if len(kwargs) == 1 and tz is None:
- arg_count = 3
- # () -> now, @ utc.
- if arg_count == 0:
- if isstr(tz):
- tz = parser.TzinfoParser.parse(tz)
- return self.type.now(tz)
- if isinstance(tz, dt_tzinfo):
- return self.type.now(tz)
- return self.type.utcnow()
- if arg_count == 1:
- arg = args[0]
- # (None) -> now, @ utc.
- if arg is None:
- return self.type.utcnow()
- # try (int, float) -> from timestamp with tz
- elif not isstr(arg) and is_timestamp(arg):
- if tz is None:
- # set to UTC by default
- tz = dateutil_tz.tzutc()
- return self.type.fromtimestamp(arg, tzinfo=tz)
- # (Arrow) -> from the object's datetime.
- elif isinstance(arg, Arrow):
- return self.type.fromdatetime(arg.datetime)
- # (datetime) -> from datetime.
- elif isinstance(arg, datetime):
- return self.type.fromdatetime(arg)
- # (date) -> from date.
- elif isinstance(arg, date):
- return self.type.fromdate(arg)
- # (tzinfo) -> now, @ tzinfo.
- elif isinstance(arg, dt_tzinfo):
- return self.type.now(arg)
- # (str) -> parse.
- elif isstr(arg):
- dt = parser.DateTimeParser(locale).parse_iso(arg)
- return self.type.fromdatetime(dt, tz)
- # (struct_time) -> from struct_time
- elif isinstance(arg, struct_time):
- return self.type.utcfromtimestamp(calendar.timegm(arg))
- # (iso calendar) -> convert then from date
- elif isinstance(arg, tuple) and len(arg) == 3:
- dt = iso_to_gregorian(*arg)
- return self.type.fromdate(dt)
- else:
- raise TypeError(
- "Can't parse single argument of type '{}'".format(type(arg))
- )
- elif arg_count == 2:
- arg_1, arg_2 = args[0], args[1]
- if isinstance(arg_1, datetime):
- # (datetime, tzinfo/str) -> fromdatetime replace tzinfo.
- if isinstance(arg_2, dt_tzinfo) or isstr(arg_2):
- return self.type.fromdatetime(arg_1, arg_2)
- else:
- raise TypeError(
- "Can't parse two arguments of types 'datetime', '{}'".format(
- type(arg_2)
- )
- )
- elif isinstance(arg_1, date):
- # (date, tzinfo/str) -> fromdate replace tzinfo.
- if isinstance(arg_2, dt_tzinfo) or isstr(arg_2):
- return self.type.fromdate(arg_1, tzinfo=arg_2)
- else:
- raise TypeError(
- "Can't parse two arguments of types 'date', '{}'".format(
- type(arg_2)
- )
- )
- # (str, format) -> parse.
- elif isstr(arg_1) and (isstr(arg_2) or isinstance(arg_2, list)):
- dt = parser.DateTimeParser(locale).parse(args[0], args[1])
- return self.type.fromdatetime(dt, tzinfo=tz)
- else:
- raise TypeError(
- "Can't parse two arguments of types '{}' and '{}'".format(
- type(arg_1), type(arg_2)
- )
- )
- # 3+ args -> datetime-like via constructor.
- else:
- return self.type(*args, **kwargs)
- def utcnow(self):
- """Returns an :class:`Arrow <arrow.arrow.Arrow>` object, representing "now" in UTC time.
- Usage::
- >>> import arrow
- >>> arrow.utcnow()
- <Arrow [2013-05-08T05:19:07.018993+00:00]>
- """
- return self.type.utcnow()
- def now(self, tz=None):
- """Returns an :class:`Arrow <arrow.arrow.Arrow>` object, representing "now" in the given
- timezone.
- :param tz: (optional) A :ref:`timezone expression <tz-expr>`. Defaults to local time.
- Usage::
- >>> import arrow
- >>> arrow.now()
- <Arrow [2013-05-07T22:19:11.363410-07:00]>
- >>> arrow.now('US/Pacific')
- <Arrow [2013-05-07T22:19:15.251821-07:00]>
- >>> arrow.now('+02:00')
- <Arrow [2013-05-08T07:19:25.618646+02:00]>
- >>> arrow.now('local')
- <Arrow [2013-05-07T22:19:39.130059-07:00]>
- """
- if tz is None:
- tz = dateutil_tz.tzlocal()
- elif not isinstance(tz, dt_tzinfo):
- tz = parser.TzinfoParser.parse(tz)
- return self.type.now(tz)
|