2008-10-22 15 views
54

Gibt es eine ziemlich einfache Art und Weise ein Datetime-Objekt in eine RFC 1123 (HTTP/1.1) Datum/Zeit-String, dh eine Zeichenkette mit dem FormatRFC 1123 Datumsdarstellung in Python?

Sun, 06 Nov 1994 08:49:37 GMT 

Mit strftime nicht funktioniert, da die Zeichenfolge zu konvertieren sind Gebietsschema-abhängig. Muss ich die Schnur von Hand bauen?

Antwort

75

Sie können wsgiref.handlers.format_date_time vom stdlib verwenden, die auf lokalen Einstellungen beruhen nicht

from wsgiref.handlers import format_date_time 
from datetime import datetime 
from time import mktime 

now = datetime.now() 
stamp = mktime(now.timetuple()) 
print format_date_time(stamp) #--> Wed, 22 Oct 2008 10:52:40 GMT 

Sie können email.utils.formatdate vom stdlib verwenden, die auf lokalen Einstellungen beruht nicht

from email.utils import formatdate 
from datetime import datetime 
from time import mktime 

now = datetime.now() 
stamp = mktime(now.timetuple()) 
print formatdate(
    timeval  = stamp, 
    localtime = False, 
    usegmt  = True 
) #--> Wed, 22 Oct 2008 10:55:46 GMT 

Wenn Sie das Gebietsschema Prozess einstellen breit, dann können Sie tun:

import locale, datetime 

locale.setlocale(locale.LC_TIME, 'en_US') 
datetime.datetime.utcnow().strftime('%a, %d %b %Y %H:%M:%S GMT') 

Wenn Sie Sie Babeldate formating

from datetime import datetime 
from babel.dates import format_datetime 

now = datetime.utcnow() 
format = 'EEE, dd LLL yyyy hh:mm:ss' 
print format_datetime(now, format, locale='en') + ' GMT' 

eine manuelle Art und Weise zu Format verwenden konnte nicht breit die locale Prozess einstellen möchten es, die mit wsgiref.handlers.format_date_time identisch ist:

def httpdate(dt): 
    """Return a string representation of a date according to RFC 1123 
    (HTTP/1.1). 

    The supplied date must be in UTC. 

    """ 
    weekday = ["Mon", "Tue", "Wed", "Thu", "Fri", "Sat", "Sun"][dt.weekday()] 
    month = ["Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", 
      "Oct", "Nov", "Dec"][dt.month - 1] 
    return "%s, %02d %s %04d %02d:%02d:%02d GMT" % (weekday, dt.day, month, 
     dt.year, dt.hour, dt.minute, dt.second) 
+1

Diese Lösung wird aus dem in der Frage beschriebenen Grund nicht funktionieren: stftime ist locale-abhängig. –

+0

das ist warum Sie ocale.setlocale (locale.LC_TIME, 'en_US') –

+1

Und deshalb benötigt SO einen Indikator, wann Antworten bearbeitet wurden. –

1

können Sie einstellen, LC_TIME stftime zu zwingen() ein bestimmtes Gebietsschema zu verwenden:

>>> locale.setlocale(locale.LC_TIME, 'en_US') 
'en_US' 
>>> datetime.datetime.now().strftime(locale.nl_langinfo(locale.D_T_FMT)) 
'Wed 22 Oct 2008 06:05:39 AM ' 
+0

Nun, Einstellung (und Zurücksetzen) locale.LC_TIME arbeitet, natürlich, aber das hat Nachteile, wie die meisten wahrscheinlich nicht Thread-sicher zu sein, und je nach dem Katalog von String-Übersetzungen. Außerdem gibt locale.D_T_FMT das richtige Format nicht zurück, wenn LC_TIME auf en_US gesetzt ist. –

+0

locale.D_T_FMT war nur ein Beispiel. Sie können jedes beliebige Format an strftime() übergeben. –

30

Sie die Format() Funktion aus dem Standard-emai Python verwenden können l Modul:

from email.utils import formatdate 
print formatdate(timeval=None, localtime=False, usegmt=True) 

Gibt die aktuelle Uhrzeit im gewünschten Format:

Wed, 22 Oct 2008 10:32:33 GMT 

In der Tat, diese Funktion macht es "von Hand" ohne strftime() verwendet

1

Nun, hier ist eine manuelle Funktion zu formatieren:

def httpdate(dt): 
    """Return a string representation of a date according to RFC 1123 
    (HTTP/1.1). 

    The supplied date must be in UTC. 

    """ 
    weekday = ["Mon", "Tue", "Wed", "Thu", "Fri", "Sat", "Sun"][dt.weekday()] 
    month = ["Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", 
      "Oct", "Nov", "Dec"][dt.month - 1] 
    return "%s, %02d %s %04d %02d:%02d:%02d GMT" % (weekday, dt.day, month, 
     dt.year, dt.hour, dt.minute, dt.second) 
6

Wenn dies jemand liest arbeitet an einem Dja ngo Projekt, Django bietet eine Funktion django.utils.http.http_date(epoch_seconds).

from django.utils.http import http_date 

some_datetime = some_object.last_update 
response['Last-Modified'] = http_date(some_datetime.timestamp())