views:

55

answers:

2

I want to convert a Python datetime to a an RFC 2822 datetime. I've tried these methods to no avail:

>>> from email.Utils import formatdate
>>> import datetime
>>> formatdate(datetime.datetime.now())
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "/System/Library/Frameworks/Python.framework/Versions/2.6/lib/python2.6/email    /utils.py", line 159, in formatdate
    now = time.gmtime(timeval)
TypeError: a float is required
A: 

If you indeed want the current time, just call formatdate with no arguments:

>>> from email.Utils import formatdate
>>> formatdate()
'Tue, 10 Aug 2010 20:40:23 -0000'

But, if you must pass it an argument, you want the output of time.time (a number of seconds since 01/01/1970):

>>> import time
>>> formatdate(time.time())
'Tue, 10 Aug 2010 20:41:43 -0000'

FWIW, datetime.datetime.now() returns a datetime object, which is not what formatdate expects.

Edited to add: if you already have a datetime object, you can format it appropriately for formatdate:

>>> import datetime
>>> dt = datetime.datetime.now()
>>> formatdate(float(dt.strftime('%s')))
'Tue, 10 Aug 2010 20:46:16 -0000'

Edited: Alex Martelli noted that the '%s' format string for strftime may not be portable across platforms. A possible alternative would be, as he himself suggested,

>>> formatdate(time.mktime(dt.timetuple()))
'Tue, 10 Aug 2010 20:46:16 -0000'
rbp
Interesting -- the `'%s'` format to `strftime` appears to be undocumented, and thus one of those which "happen to work on some platforms" (but not on all of them) -- unless you have docs that show it _will_ work cross-platform?
Alex Martelli
Actually, that's the format string that the Unix "date" command takes to output epoch-based time, I used it without thinking and it worked :) I'll check and get back to you (or change my answer).
rbp
Well, http://docs.python.org/library/datetime.html#strftime-strptime-behavior states that "The full set of format codes supported varies across platforms, because Python calls the platform C library’s strftime() function, and platform variations are common." And, on Linux, man 3 strftime lists '%s' as a possible format value. So I suppose you're right, it's likely not to be portable. Thanks :)
rbp
+2  A: 

Here's some working code, broken down into simple pieces just for clarity:

>>> import datetime
>>> import time
>>> from email import utils
>>> nowdt = datetime.datetime.now()
>>> nowtuple = nowdt.timetuple()
>>> nowtimestamp = time.mktime(nowtuple)
>>> utils.formatdate(nowtimestamp)
'Tue, 10 Aug 2010 20:43:53 -0000'

Explanation: email.utils.formatdate wants a timestamp -- i.e., a float with seconds (and fraction thereof) since the epoch. A datetime instance doesn't give you a timestamp directly -- but, it can give you a time-tuple with the timetuple method, and time.mktime of course can then make a timestamp from such a tuple.

Alex Martelli