timezones.txt 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702
  1. .. _time-zones:
  2. ==========
  3. Time zones
  4. ==========
  5. .. versionadded:: 1.4
  6. .. _time-zones-overview:
  7. Overview
  8. ========
  9. When support for time zones is enabled, Django stores date and time
  10. information in UTC in the database, uses time-zone-aware datetime objects
  11. internally, and translates them to the end user's time zone in templates and
  12. forms.
  13. This is handy if your users live in more than one time zone and you want to
  14. display date and time information according to each user's wall clock.
  15. Even if your Web site is available in only one time zone, it's still good
  16. practice to store data in UTC in your database. One main reason is Daylight
  17. Saving Time (DST). Many countries have a system of DST, where clocks are moved
  18. forward in spring and backward in autumn. If you're working in local time,
  19. you're likely to encounter errors twice a year, when the transitions happen.
  20. (The pytz_ documentation discusses `these issues`_ in greater detail.) This
  21. probably doesn't matter for your blog, but it's a problem if you over-bill or
  22. under-bill your customers by one hour, twice a year, every year. The solution
  23. to this problem is to use UTC in the code and use local time only when
  24. interacting with end users.
  25. Time zone support is disabled by default. To enable it, set :setting:`USE_TZ =
  26. True <USE_TZ>` in your settings file. Installing pytz_ is highly recommended,
  27. but not mandatory. It's as simple as:
  28. .. code-block:: bash
  29. $ sudo pip install pytz
  30. .. note::
  31. The default :file:`settings.py` file created by :djadmin:`django-admin.py
  32. startproject <startproject>` includes :setting:`USE_TZ = True <USE_TZ>`
  33. for convenience.
  34. .. note::
  35. There is also an independent but related :setting:`USE_L10N` setting that
  36. controls whether Django should activate format localization. See
  37. :doc:`/topics/i18n/formatting` for more details.
  38. If you're wrestling with a particular problem, start with the :ref:`time zone
  39. FAQ <time-zones-faq>`.
  40. Concepts
  41. ========
  42. Naive and aware datetime objects
  43. --------------------------------
  44. Python's :class:`datetime.datetime` objects have a ``tzinfo`` attribute that
  45. can be used to store time zone information, represented as an instance of a
  46. subclass of :class:`datetime.tzinfo`. When this attribute is set and describes
  47. an offset, a datetime object is **aware**. Otherwise, it's **naive**.
  48. You can use :func:`~django.utils.timezone.is_aware` and
  49. :func:`~django.utils.timezone.is_naive` to determine whether datetimes are
  50. aware or naive.
  51. When time zone support is disabled, Django uses naive datetime objects in local
  52. time. This is simple and sufficient for many use cases. In this mode, to obtain
  53. the current time, you would write::
  54. import datetime
  55. now = datetime.datetime.now()
  56. When time zone support is enabled, Django uses time-zone-aware datetime
  57. objects. If your code creates datetime objects, they should be aware too. In
  58. this mode, the example above becomes::
  59. import datetime
  60. from django.utils.timezone import utc
  61. now = datetime.datetime.utcnow().replace(tzinfo=utc)
  62. .. note::
  63. :mod:`django.utils.timezone` provides a
  64. :func:`~django.utils.timezone.now()` function that returns a naive or
  65. aware datetime object according to the value of :setting:`USE_TZ`.
  66. .. warning::
  67. Dealing with aware datetime objects isn't always intuitive. For instance,
  68. the ``tzinfo`` argument of the standard datetime constructor doesn't work
  69. reliably for time zones with DST. Using UTC is generally safe; if you're
  70. using other time zones, you should review the `pytz`_ documentation
  71. carefully.
  72. .. note::
  73. Python's :class:`datetime.time` objects also feature a ``tzinfo``
  74. attribute, and PostgreSQL has a matching ``time with time zone`` type.
  75. However, as PostgreSQL's docs put it, this type "exhibits properties which
  76. lead to questionable usefulness".
  77. Django only supports naive time objects and will raise an exception if you
  78. attempt to save an aware time object.
  79. .. _naive-datetime-objects:
  80. Interpretation of naive datetime objects
  81. ----------------------------------------
  82. When :setting:`USE_TZ` is ``True``, Django still accepts naive datetime
  83. objects, in order to preserve backwards-compatibility. When the database layer
  84. receives one, it attempts to make it aware by interpreting it in the
  85. :ref:`default time zone <default-current-time-zone>` and raises a warning.
  86. Unfortunately, during DST transitions, some datetimes don't exist or are
  87. ambiguous. In such situations, pytz_ raises an exception. Other
  88. :class:`~datetime.tzinfo` implementations, such as the local time zone used as
  89. a fallback when pytz_ isn't installed, may raise an exception or return
  90. inaccurate results. That's why you should always create aware datetime objects
  91. when time zone support is enabled.
  92. In practice, this is rarely an issue. Django gives you aware datetime objects
  93. in the models and forms, and most often, new datetime objects are created from
  94. existing ones through :class:`~datetime.timedelta` arithmetic. The only
  95. datetime that's often created in application code is the current time, and
  96. :func:`timezone.now() <django.utils.timezone.now>` automatically does the
  97. right thing.
  98. .. _default-current-time-zone:
  99. Default time zone and current time zone
  100. ---------------------------------------
  101. The **default time zone** is the time zone defined by the :setting:`TIME_ZONE`
  102. setting.
  103. The **current time zone** is the time zone that's used for rendering.
  104. You should set the current time zone to the end user's actual time zone with
  105. :func:`~django.utils.timezone.activate`. Otherwise, the default time zone is
  106. used.
  107. .. note::
  108. As explained in the documentation of :setting:`TIME_ZONE`, Django sets
  109. environment variables so that its process runs in the default time zone.
  110. This happens regardless of the value of :setting:`USE_TZ` and of the
  111. current time zone.
  112. When :setting:`USE_TZ` is ``True``, this is useful to preserve
  113. backwards-compatibility with applications that still rely on local time.
  114. However, :ref:`as explained above <naive-datetime-objects>`, this isn't
  115. entirely reliable, and you should always work with aware datetimes in UTC
  116. in your own code. For instance, use
  117. :meth:`~datetime.datetime.utcfromtimestamp` instead of
  118. :meth:`~datetime.datetime.fromtimestamp` -- and don't forget to set
  119. ``tzinfo`` to :data:`~django.utils.timezone.utc`.
  120. Selecting the current time zone
  121. -------------------------------
  122. The current time zone is the equivalent of the current :term:`locale <locale
  123. name>` for translations. However, there's no equivalent of the
  124. ``Accept-Language`` HTTP header that Django could use to determine the user's
  125. time zone automatically. Instead, Django provides :ref:`time zone selection
  126. functions <time-zone-selection-functions>`. Use them to build the time zone
  127. selection logic that makes sense for you.
  128. Most Web sites that care about time zones just ask users in which time zone they
  129. live and store this information in the user's profile. For anonymous users,
  130. they use the time zone of their primary audience or UTC. pytz_ provides
  131. helpers_, like a list of time zones per country, that you can use to pre-select
  132. the most likely choices.
  133. Here's an example that stores the current timezone in the session. (It skips
  134. error handling entirely for the sake of simplicity.)
  135. Add the following middleware to :setting:`MIDDLEWARE_CLASSES`::
  136. from django.utils import timezone
  137. class TimezoneMiddleware(object):
  138. def process_request(self, request):
  139. tz = request.session.get('django_timezone')
  140. if tz:
  141. timezone.activate(tz)
  142. Create a view that can set the current timezone::
  143. import pytz
  144. from django.shortcuts import redirect, render
  145. def set_timezone(request):
  146. if request.method == 'POST':
  147. request.session['django_timezone'] = pytz.timezone(request.POST['timezone'])
  148. return redirect('/')
  149. else:
  150. return render(request, 'template.html', {'timezones': pytz.common_timezones})
  151. Include a form in ``template.html`` that will ``POST`` to this view:
  152. .. code-block:: html+django
  153. {% load tz %}{% load url from future %}
  154. <form action="{% url 'set_timezone' %}" method="POST">
  155. {% csrf_token %}
  156. <label for="timezone">Time zone:</label>
  157. <select name="timezone">
  158. {% for tz in timezones %}
  159. <option value="{{ tz }}"{% if tz == TIME_ZONE %} selected="selected"{% endif %}>{{ tz }}</option>
  160. {% endfor %}
  161. </select>
  162. <input type="submit" value="Set" />
  163. </form>
  164. .. _time-zones-in-forms:
  165. Time zone aware input in forms
  166. ==============================
  167. When you enable time zone support, Django interprets datetimes entered in
  168. forms in the :ref:`current time zone <default-current-time-zone>` and returns
  169. aware datetime objects in ``cleaned_data``.
  170. If the current time zone raises an exception for datetimes that don't exist or
  171. are ambiguous because they fall in a DST transition (the timezones provided by
  172. pytz_ do this), such datetimes will be reported as invalid values.
  173. .. _time-zones-in-templates:
  174. Time zone aware output in templates
  175. ===================================
  176. When you enable time zone support, Django converts aware datetime objects to
  177. the :ref:`current time zone <default-current-time-zone>` when they're rendered
  178. in templates. This behaves very much like :doc:`format localization
  179. </topics/i18n/formatting>`.
  180. .. warning::
  181. Django doesn't convert naive datetime objects, because they could be
  182. ambiguous, and because your code should never produce naive datetimes when
  183. time zone support is enabled. However, you can force conversion with the
  184. template filters described below.
  185. Conversion to local time isn't always appropriate -- you may be generating
  186. output for computers rather than for humans. The following filters and tags,
  187. provided by the ``tz`` template tag library, allow you to control the time zone
  188. conversions.
  189. Template tags
  190. -------------
  191. .. templatetag:: localtime
  192. localtime
  193. ~~~~~~~~~
  194. Enables or disables conversion of aware datetime objects to the current time
  195. zone in the contained block.
  196. This tag has exactly the same effects as the :setting:`USE_TZ` setting as far
  197. as the template engine is concerned. It allows a more fine grained control of
  198. conversion.
  199. To activate or deactivate conversion for a template block, use::
  200. {% load tz %}
  201. {% localtime on %}
  202. {{ value }}
  203. {% endlocaltime %}
  204. {% localtime off %}
  205. {{ value }}
  206. {% endlocaltime %}
  207. .. note::
  208. The value of :setting:`USE_TZ` isn't respected inside of a
  209. ``{% localtime %}`` block.
  210. .. templatetag:: timezone
  211. timezone
  212. ~~~~~~~~
  213. Sets or unsets the current time zone in the contained block. When the current
  214. time zone is unset, the default time zone applies.
  215. ::
  216. {% load tz %}
  217. {% timezone "Europe/Paris" %}
  218. Paris time: {{ value }}
  219. {% endtimezone %}
  220. {% timezone None %}
  221. Server time: {{ value }}
  222. {% endtimezone %}
  223. .. templatetag:: get_current_timezone
  224. get_current_timezone
  225. ~~~~~~~~~~~~~~~~~~~~
  226. When the :func:`django.core.context_processors.tz` context processor is
  227. enabled -- by default, it is -- each :class:`~django.template.RequestContext`
  228. contains a ``TIME_ZONE`` variable that provides the name of the current time
  229. zone.
  230. If you don't use a :class:`~django.template.RequestContext`, you can obtain
  231. this value with the ``get_current_timezone`` tag::
  232. {% get_current_timezone as TIME_ZONE %}
  233. Template filters
  234. ----------------
  235. These filters accept both aware and naive datetimes. For conversion purposes,
  236. they assume that naive datetimes are in the default time zone. They always
  237. return aware datetimes.
  238. .. templatefilter:: localtime
  239. localtime
  240. ~~~~~~~~~
  241. Forces conversion of a single value to the current time zone.
  242. For example::
  243. {% load tz %}
  244. {{ value|localtime }}
  245. .. templatefilter:: utc
  246. utc
  247. ~~~
  248. Forces conversion of a single value to UTC.
  249. For example::
  250. {% load tz %}
  251. {{ value|utc }}
  252. .. templatefilter:: timezone
  253. timezone
  254. ~~~~~~~~
  255. Forces conversion of a single value to an arbitrary timezone.
  256. The argument must be an instance of a :class:`~datetime.tzinfo` subclass or a
  257. time zone name. If it is a time zone name, pytz_ is required.
  258. For example::
  259. {% load tz %}
  260. {{ value|timezone:"Europe/Paris" }}
  261. .. _time-zones-migration-guide:
  262. Migration guide
  263. ===============
  264. Here's how to migrate a project that was started before Django supported time
  265. zones.
  266. Database
  267. --------
  268. PostgreSQL
  269. ~~~~~~~~~~
  270. The PostgreSQL backend stores datetimes as ``timestamp with time zone``. In
  271. practice, this means it converts datetimes from the connection's time zone to
  272. UTC on storage, and from UTC to the connection's time zone on retrieval.
  273. As a consequence, if you're using PostgreSQL, you can switch between ``USE_TZ
  274. = False`` and ``USE_TZ = True`` freely. The database connection's time zone
  275. will be set to :setting:`TIME_ZONE` or ``UTC`` respectively, so that Django
  276. obtains correct datetimes in all cases. You don't need to perform any data
  277. conversions.
  278. Other databases
  279. ~~~~~~~~~~~~~~~
  280. Other backends store datetimes without time zone information. If you switch
  281. from ``USE_TZ = False`` to ``USE_TZ = True``, you must convert your data from
  282. local time to UTC -- which isn't deterministic if your local time has DST.
  283. Code
  284. ----
  285. The first step is to add :setting:`USE_TZ = True <USE_TZ>` to your settings
  286. file and install pytz_ (if possible). At this point, things should mostly
  287. work. If you create naive datetime objects in your code, Django makes them
  288. aware when necessary.
  289. However, these conversions may fail around DST transitions, which means you
  290. aren't getting the full benefits of time zone support yet. Also, you're likely
  291. to run into a few problems because it's impossible to compare a naive datetime
  292. with an aware datetime. Since Django now gives you aware datetimes, you'll get
  293. exceptions wherever you compare a datetime that comes from a model or a form
  294. with a naive datetime that you've created in your code.
  295. So the second step is to refactor your code wherever you instantiate datetime
  296. objects to make them aware. This can be done incrementally.
  297. :mod:`django.utils.timezone` defines some handy helpers for compatibility
  298. code: :func:`~django.utils.timezone.now`,
  299. :func:`~django.utils.timezone.is_aware`,
  300. :func:`~django.utils.timezone.is_naive`,
  301. :func:`~django.utils.timezone.make_aware`, and
  302. :func:`~django.utils.timezone.make_naive`.
  303. Finally, in order to help you locate code that needs upgrading, Django raises
  304. a warning when you attempt to save a naive datetime to the database::
  305. RuntimeWarning: DateTimeField received a naive datetime (2012-01-01 00:00:00) while time zone support is active.
  306. During development, you can turn such warnings into exceptions and get a
  307. traceback by adding the following to your settings file::
  308. import warnings
  309. warnings.filterwarnings(
  310. 'error', r"DateTimeField received a naive datetime",
  311. RuntimeWarning, r'django\.db\.models\.fields')
  312. Fixtures
  313. --------
  314. When serializing an aware datetime, the UTC offset is included, like this::
  315. "2011-09-01T13:20:30+03:00"
  316. For a naive datetime, it obviously isn't::
  317. "2011-09-01T13:20:30"
  318. For models with :class:`~django.db.models.DateTimeField`\ s, this difference
  319. makes it impossible to write a fixture that works both with and without time
  320. zone support.
  321. Fixtures generated with ``USE_TZ = False``, or before Django 1.4, use the
  322. "naive" format. If your project contains such fixtures, after you enable time
  323. zone support, you'll see :exc:`RuntimeWarning`\ s when you load them. To get
  324. rid of the warnings, you must convert your fixtures to the "aware" format.
  325. You can regenerate fixtures with :djadmin:`loaddata` then :djadmin:`dumpdata`.
  326. Or, if they're small enough, you can simply edit them to add the UTC offset
  327. that matches your :setting:`TIME_ZONE` to each serialized datetime.
  328. .. _time-zones-faq:
  329. FAQ
  330. ===
  331. Setup
  332. -----
  333. 1. **I don't need multiple time zones. Should I enable time zone support?**
  334. Yes. When time zone support is enabled, Django uses a more accurate model
  335. of local time. This shields you from subtle and unreproducible bugs around
  336. Daylight Saving Time (DST) transitions.
  337. In this regard, time zones are comparable to ``unicode`` in Python. At first
  338. it's hard. You get encoding and decoding errors. Then you learn the rules.
  339. And some problems disappear -- you never get mangled output again when your
  340. application receives non-ASCII input.
  341. When you enable time zone support, you'll encounter some errors because
  342. you're using naive datetimes where Django expects aware datetimes. Such
  343. errors show up when running tests and they're easy to fix. You'll quickly
  344. learn how to avoid invalid operations.
  345. On the other hand, bugs caused by the lack of time zone support are much
  346. harder to prevent, diagnose and fix. Anything that involves scheduled tasks
  347. or datetime arithmetic is a candidate for subtle bugs that will bite you
  348. only once or twice a year.
  349. For these reasons, time zone support is enabled by default in new projects,
  350. and you should keep it unless you have a very good reason not to.
  351. 2. **I've enabled time zone support. Am I safe?**
  352. Maybe. You're better protected from DST-related bugs, but you can still
  353. shoot yourself in the foot by carelessly turning naive datetimes into aware
  354. datetimes, and vice-versa.
  355. If your application connects to other systems -- for instance, if it queries
  356. a Web service -- make sure datetimes are properly specified. To transmit
  357. datetimes safely, their representation should include the UTC offset, or
  358. their values should be in UTC (or both!).
  359. Finally, our calendar system contains interesting traps for computers::
  360. >>> import datetime
  361. >>> def substract_one_year(value): # DON'T DO THAT!
  362. ... return value.replace(year=value.year - 1)
  363. >>> one_year_before(datetime.datetime(2012, 3, 1, 10, 0))
  364. datetime.datetime(2011, 3, 1, 10, 0)
  365. >>> one_year_before(datetime.datetime(2012, 2, 29, 10, 0))
  366. Traceback (most recent call last):
  367. ...
  368. ValueError: day is out of range for month
  369. (To implement this function, you must decide whether 2012-02-29 minus
  370. one year is 2011-02-28 or 2011-03-01, which depends on your business
  371. requirements.)
  372. 3. **Should I install pytz?**
  373. Yes. Django has a policy of not requiring external dependencies, and for
  374. this reason pytz_ is optional. However, it's much safer to install it.
  375. As soon as you activate time zone support, Django needs a definition of the
  376. default time zone. When pytz is available, Django loads this definition
  377. from the `tz database`_. This is the most accurate solution. Otherwise, it
  378. relies on the difference between local time and UTC, as reported by the
  379. operating system, to compute conversions. This is less reliable, especially
  380. around DST transitions.
  381. Furthermore, if you want to support users in more than one time zone, pytz
  382. is the reference for time zone definitions.
  383. Troubleshooting
  384. ---------------
  385. 1. **My application crashes with** ``TypeError: can't compare offset-naive``
  386. ``and offset-aware datetimes`` **-- what's wrong?**
  387. Let's reproduce this error by comparing a naive and an aware datetime::
  388. >>> import datetime
  389. >>> from django.utils import timezone
  390. >>> naive = datetime.datetime.utcnow()
  391. >>> aware = naive.replace(tzinfo=timezone.utc)
  392. >>> naive == aware
  393. Traceback (most recent call last):
  394. ...
  395. TypeError: can't compare offset-naive and offset-aware datetimes
  396. If you encounter this error, most likely your code is comparing these two
  397. things:
  398. - a datetime provided by Django -- for instance, a value read from a form or
  399. a model field. Since you enabled time zone support, it's aware.
  400. - a datetime generated by your code, which is naive (or you wouldn't be
  401. reading this).
  402. Generally, the correct solution is to change your code to use an aware
  403. datetime instead.
  404. If you're writing a pluggable application that's expected to work
  405. independently of the value of :setting:`USE_TZ`, you may find
  406. :func:`django.utils.timezone.now` useful. This function returns the current
  407. date and time as a naive datetime when ``USE_TZ = False`` and as an aware
  408. datetime when ``USE_TZ = True``. You can add or subtract
  409. :class:`datetime.timedelta` as needed.
  410. 2. **I see lots of** ``RuntimeWarning: DateTimeField received a naive
  411. datetime`` ``(YYYY-MM-DD HH:MM:SS)`` ``while time zone support is active``
  412. **-- is that bad?**
  413. When time zone support is enabled, the database layer expects to receive
  414. only aware datetimes from your code. This warning occurs when it receives a
  415. naive datetime. This indicates that you haven't finished porting your code
  416. for time zone support. Please refer to the :ref:`migration guide
  417. <time-zones-migration-guide>` for tips on this process.
  418. In the meantime, for backwards compatibility, the datetime is considered to
  419. be in the default time zone, which is generally what you expect.
  420. 3. ``now.date()`` **is yesterday! (or tomorrow)**
  421. If you've always used naive datetimes, you probably believe that you can
  422. convert a datetime to a date by calling its :meth:`~datetime.datetime.date`
  423. method. You also consider that a :class:`~datetime.date` is a lot like a
  424. :class:`~datetime.datetime`, except that it's less accurate.
  425. None of this is true in a time zone aware environment::
  426. >>> import datetime
  427. >>> import pytz
  428. >>> paris_tz = pytz.timezone("Europe/Paris")
  429. >>> new_york_tz = pytz.timezone("America/New_York")
  430. >>> paris = paris_tz.localize(datetime.datetime(2012, 3, 3, 1, 30))
  431. # This is the correct way to convert between time zones with pytz.
  432. >>> new_york = new_york_tz.normalize(paris.astimezone(new_york_tz))
  433. >>> paris == new_york, paris.date() == new_york.date()
  434. (True, False)
  435. >>> paris - new_york, paris.date() - new_york.date()
  436. (datetime.timedelta(0), datetime.timedelta(1))
  437. >>> paris
  438. datetime.datetime(2012, 3, 3, 1, 30, tzinfo=<DstTzInfo 'Europe/Paris' CET+1:00:00 STD>)
  439. >>> new_york
  440. datetime.datetime(2012, 3, 2, 19, 30, tzinfo=<DstTzInfo 'America/New_York' EST-1 day, 19:00:00 STD>)
  441. As this example shows, the same datetime has a different date, depending on
  442. the time zone in which it is represented. But the real problem is more
  443. fundamental.
  444. A datetime represents a **point in time**. It's absolute: it doesn't depend
  445. on anything. On the contrary, a date is a **calendaring concept**. It's a
  446. period of time whose bounds depend on the time zone in which the date is
  447. considered. As you can see, these two concepts are fundamentally different,
  448. and converting a datetime to a date isn't a deterministic operation.
  449. What does this mean in practice?
  450. Generally, you should avoid converting a :class:`~datetime.datetime` to
  451. :class:`~datetime.date`. For instance, you can use the :tfilter:`date`
  452. template filter to only show the date part of a datetime. This filter will
  453. convert the datetime into the current time zone before formatting it,
  454. ensuring the results appear correctly.
  455. If you really need to do the conversion yourself, you must ensure the
  456. datetime is converted to the appropriate time zone first. Usually, this
  457. will be the current timezone::
  458. >>> from django.utils import timezone
  459. >>> timezone.activate(pytz.timezone("Asia/Singapore"))
  460. # For this example, we just set the time zone to Singapore, but here's how
  461. # you would obtain the current time zone in the general case.
  462. >>> current_tz = timezone.get_current_timezone()
  463. # Again, this is the correct way to convert between time zones with pytz.
  464. >>> local = current_tz.normalize(paris.astimezone(current_tz))
  465. >>> local
  466. datetime.datetime(2012, 3, 3, 8, 30, tzinfo=<DstTzInfo 'Asia/Singapore' SGT+8:00:00 STD>)
  467. >>> local.date()
  468. datetime.date(2012, 3, 3)
  469. Usage
  470. -----
  471. 1. **I have a string** ``"2012-02-21 10:28:45"`` **and I know it's in the**
  472. ``"Europe/Helsinki"`` **time zone. How do I turn that into an aware
  473. datetime?**
  474. This is exactly what pytz_ is for.
  475. >>> from django.utils.dateparse import parse_datetime
  476. >>> naive = parse_datetime("2012-02-21 10:28:45")
  477. >>> import pytz
  478. >>> pytz.timezone("Europe/Helsinki").localize(naive)
  479. datetime.datetime(2012, 2, 21, 10, 28, 45, tzinfo=<DstTzInfo 'Europe/Helsinki' EET+2:00:00 STD>)
  480. Note that ``localize`` is a pytz extension to the :class:`~datetime.tzinfo`
  481. API. Also, you may want to catch :exc:`~pytz.InvalidTimeError`. The
  482. documentation of pytz contains `more examples`_. You should review it
  483. before attempting to manipulate aware datetimes.
  484. 2. **How can I obtain the current time in the local time zone?**
  485. Well, the first question is, do you really need to?
  486. You should only use local time when you're interacting with humans, and the
  487. template layer provides :ref:`filters and tags <time-zones-in-templates>`
  488. to convert datetimes to the time zone of your choice.
  489. Furthermore, Python knows how to compare aware datetimes, taking into
  490. account UTC offsets when necessary. It's much easier (and possibly faster)
  491. to write all your model and view code in UTC. So, in most circumstances,
  492. the datetime in UTC returned by :func:`django.utils.timezone.now` will be
  493. sufficient.
  494. For the sake of completeness, though, if you really wanted the current time
  495. in the local time zone, here's how you would obtain it::
  496. >>> import datetime
  497. >>> from django.utils import timezone
  498. >>> datetime.datetime.now(tz=timezone.get_default_timezone())
  499. datetime.datetime(2012, 3, 3, 20, 10, 53, 873365, tzinfo=<DstTzInfo 'Europe/Paris' CET+1:00:00 STD>)
  500. In this example, pytz_ is installed and :setting:`TIME_ZONE` is
  501. ``"Europe/Paris"``.
  502. 3. **How can I see all available time zones?**
  503. pytz_ provides helpers_, including a list of current time zones and a list
  504. of all available time zones -- some of which are only of historical
  505. interest.
  506. .. _pytz: http://pytz.sourceforge.net/
  507. .. _more examples: http://pytz.sourceforge.net/#example-usage
  508. .. _these issues: http://pytz.sourceforge.net/#problems-with-localtime
  509. .. _helpers: http://pytz.sourceforge.net/#helpers
  510. .. _tz database: http://en.wikipedia.org/wiki/Tz_database