Как преобразовать дату и время из одного произвольного часового пояса в другой произвольный часовой пояс

Допустим, я получаю произвольный объект даты-времени в запросе, подобном этому, который может поступать из любого возможного часового пояса - я не знаю, какой именно. Ради примера, притворимся, что это с Восточного побережья

import pytz
from colander import iso8601
...
    ests1 = iso8601.parse_date('2016-04-01T00:00:00.0000-04:00')

притворяться, ests1 является объектом, который входит в

Используя pytz, я могу немного узнать о его часовом поясе.

    etz = ests1.timetz()  # 00:00:00-04:00
    etzi = ests1.tzinfo   # <FixedOffset '-04:00' datetime.timedelta(-1, 72000)>
    etzn = ests1.tzname() # '-04:00'  # EST
    edst = ests1.dst()    #  '0:00:00'

Обратите внимание, что независимо от того, какая дата ests1 - dst() всегда возвращает одно и то же значение

Я хотел бы сделать что-то вроде этого:

    psts1 = pytz.timezone(etzn
                     ).localize(dtime.datetime.fromtimestamp(ests1)
                     ).astimezone(pytz.timezone('US/Pacific'))

но это только приводит к

UnknownTimeZoneError: '-04:00'

Я также попробовал это:

def convert_dt_to_pstz(dt):
    pstoffset = -25200 # -7 * 60 * 60   - but what about DST?  how do we tell which offset to use? -8 or -7
    return dt.astimezone(tzoffset(None,pstoffset)) # convert to PST

Но проблема в том, что я не знаю, как определить, нужно ли корректировать дату на летнее время.

Я также попробовал это:

    utc_date = ests1.astimezone(pytz.utc)   # 2016-04-01T04:00:00+00:00
    ptz = pytz.timezone('US/Pacific')
    new_date = utc_date.replace(tzinfo=ptz) # 2016-04-01T04:00:00-07:53

Но посмотрите на странный результат для new_date: -07:53??

Обратите внимание, что я не могу использовать любую информацию, соответствующую "сейчас" или расположению сервера, так как это может быть где угодно

2 ответа

Решение

Чтобы преобразовать с учетом часового пояса datetime объект в другой часовой пояс, используйте datetime.astimezone() метод:

pacific_time = ests1.astimezone(pytz.timezone('US/Pacific'))

В общем, pytz документы рекомендую звонить pytz.timezone('US/Pacific').normalize() на результат, но вам здесь не нужно, потому что ests1 часовой пояс имеет фиксированное смещение utc.

Я думаю, что я правильно понял на этот раз:

from datetime import datetime as dtime
import pytz
from colander import iso8601

ests1 = iso8601.parse_date('2016-04-01T00:00:00.0000-04:00')
epoch = dtime(1970, 1, 1, tzinfo=pytz.UTC)          # 1970-01-01T00:00:00+00:00
ept = (ests1 - epoch).total_seconds()               # 1459483200.0

utc_date = dtime.fromtimestamp(ept, tz=pytz.UTC)    # 2016-04-01T04:00:00+00:00
ptz = pytz.timezone('US/Pacific')
pst_date = dtime.fromtimestamp(ept, tz=ptz)         # 2016-03-31T21:00:00-07:00

ests2 = iso8601.parse_date('2016-02-01T00:00:00.0000-04:00')
ept2 = (ests2 - epoch).total_seconds()              # 1454299200.00

utc_date2 = dtime.fromtimestamp(ept2, tz=pytz.UTC)  # 2016-02-01T04:00:00+00:00
pst_date2 = dtime.fromtimestamp(ept2, tz=ptz)       # 2016-01-31T20:00:00-08:00

так что, кажется, так же просто, как

def convert_to_tz(dt,tz_new):
   seconds = (dt - epoch).total_seconds()
   return dtime.fromtimestamp(seconds, tz=tz_new)
Другие вопросы по тегам