web-dev-qa-db-fra.com

Convertir des horodatages avec offset en date-heure obj à l'aide de strptime

J'essaye de convertir des horodatages du format "2012-07-24T23: 14: 29-07: 00" en objets datetime en python en utilisant la méthode strptime. Le problème vient du décalage temporel à la fin (-07: 00). Sans l'offset, je peux réussir

time_str = "2012-07-24T23:14:29"

time_obj=datetime.datetime.strptime(time_str,'%Y-%m-%dT%H:%M:%S')

Mais avec l'offset j'ai essayé

time_str = "2012-07-24T23:14:29-07:00"

time_obj=datetime.datetime.strptime(time_str,'%Y-%m-%dT%H:%M:%S-%z').

Mais cela donne une erreur de valeur disant que "z" est une mauvaise directive.

Des idées pour un travail autour?

44
auny

En effet, la fonction strptime() de Python 2 ne prend pas en charge le format %z pour les fuseaux horaires (car la fonction sous-jacente time.strptime() ne la prend pas en charge). Vous avez deux options:

  • Ignorer le fuseau horaire lors de l'analyse avec strptime:

    time_obj = datetime.datetime.strptime(time_str[:19], '%Y-%m-%dT%H:%M:%S')
    
  • utilisez le module dateutil , sa fonction d'analyse ne traite des fuseaux horaires:

    from dateutil.parser import parse
    time_obj = parse(time_str)
    

Démo rapide sur l'invite de commande:

>>> from dateutil.parser import parse
>>> parse("2012-07-24T23:14:29-07:00")
datetime.datetime(2012, 7, 24, 23, 14, 29, tzinfo=tzoffset(None, -25200))

Vous pouvez également effectuer une mise à niveau vers Python 3.2 ou version ultérieure, où la prise en charge des fuseaux horaires a été améliorée au point que %z fonctionnerait, à condition de supprimer le dernier : de l'entrée et le - d'avant le %z:

>>> import datetime
>>> time_str = "2012-07-24T23:14:29-07:00"
>>> datetime.datetime.strptime(time_str, '%Y-%m-%dT%H:%M:%S%z')
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "/Users/mj/Development/Library/buildout.python/parts/opt/lib/python3.4/_strptime.py", line 500, in _strptime_datetime
    tt, fraction = _strptime(data_string, format)
  File "/Users/mj/Development/Library/buildout.python/parts/opt/lib/python3.4/_strptime.py", line 337, in _strptime
    (data_string, format))
ValueError: time data '2012-07-24T23:14:29-07:00' does not match format '%Y-%m-%dT%H:%M:%S%z'
>>> ''.join(time_str.rsplit(':', 1))
'2012-07-24T23:14:29-0700'
>>> datetime.datetime.strptime(''.join(time_str.rsplit(':', 1)), '%Y-%m-%dT%H:%M:%S%z')
datetime.datetime(2012, 7, 24, 23, 14, 29, tzinfo=datetime.timezone(datetime.timedelta(-1, 61200)))
56
Martijn Pieters

En Python 3.7+:

from datetime import datetime

time_str = "2012-07-24T23:14:29-07:00"
dt_aware = datetime.fromisoformat(time_str)
print(dt_aware.isoformat('T'))
# -> 2012-07-24T23:14:29-07:00

En Python 3.2+:

from datetime import datetime

time_str = "2012-07-24T23:14:29-0700"
dt_aware = datetime.strptime(time_str, '%Y-%m-%dT%H:%M:%S%z')
print(dt_aware.isoformat('T'))
# -> 2012-07-24T23:14:29-07:00

Remarque: Avant Python 3.7, cette variante ne supportait pas : dans la partie -0700 (les deux formats sont autorisés par rfc 3339 ). Voir date/heure: possibilité d’analyser les dates et heures de la RFC 3339 .

Sur les anciennes versions de Python telles que Python 2.7, vous pouvez analyser manuellement le décalage utc:

from datetime import datetime

time_str = "2012-07-24T23:14:29-0700"
# split the utc offset part
naive_time_str, offset_str = time_str[:-5], time_str[-5:]
# parse the naive date/time part
naive_dt = datetime.strptime(naive_time_str, '%Y-%m-%dT%H:%M:%S')
# parse the utc offset
offset = int(offset_str[-4:-2])*60 + int(offset_str[-2:])
if offset_str[0] == "-":
   offset = -offset
dt = naive_dt.replace(tzinfo=FixedOffset(offset))
print(dt.isoformat('T'))

FixedOffset class est défini ici .

14
jfs

ValueError: 'z' is a bad directive in format...

(note: je dois m'en tenir à python 2.7 dans mon cas)

J'ai eu un problème similaire lors de l'analyse des dates de validation de la sortie de git log --date=iso8601 qui n'est en réalité pas le format ISO8601 (d'où l'ajout de --date=iso8601-strict dans une version ultérieure).

Depuis que je me sers de Django, je peux utiliser les utilitaires là-bas.

https://github.com/Django/django/blob/master/Django/utils/dateparse.py

>>> from Django.utils.dateparse import parse_datetime
>>> parse_datetime('2013-07-23T15:10:59.342107+01:00')
datetime.datetime(2013, 7, 23, 15, 10, 59, 342107, tzinfo=+0100)

Au lieu de strptime, vous pouvez utiliser votre propre expression régulière.

4
dnozay

Avec python 3.5.2
Pour convertir 26 Sep 2000 05:11:00 -0700

from datetime import datetime    
dt_obj = datetime.strptime("26 Sep 2000 05:11:00 -0700", '%d %b %Y %H:%M:%S %z')

Pour convertir 2012-07-24T23:14:29 -0700

dt_obj = datetime.strptime('2012-07-24T23:14:29 -0700', '%Y-%m-%dT%H:%M:%S %z')

Python 3.5.2 ne supporte pas -07: 00 le décalage horaire ':' doit être supprimé

0
H S Rathore