Browse Source

Merge branch '1693-migrar-tempo-com-timezones'

Fix #1693
pull/1704/head
Marcio Mazza 7 years ago
parent
commit
70f86dd4b9
  1. 25
      sapl/legacy/migration.py
  2. 197
      sapl/legacy/timezonesbrasil.py

25
sapl/legacy/migration.py

@ -1,3 +1,4 @@
import os
import re import re
from datetime import date from datetime import date
from functools import lru_cache, partial from functools import lru_cache, partial
@ -5,8 +6,9 @@ from itertools import groupby
from subprocess import PIPE, call from subprocess import PIPE, call
import pkg_resources import pkg_resources
import reversion
import yaml import yaml
import reversion
from django.apps import apps from django.apps import apps
from django.apps.config import AppConfig from django.apps.config import AppConfig
from django.contrib.auth import get_user_model from django.contrib.auth import get_user_model
@ -16,7 +18,6 @@ from django.core.exceptions import ObjectDoesNotExist
from django.db import connections, transaction from django.db import connections, transaction
from django.db.models import Count, Max from django.db.models import Count, Max
from django.db.models.base import ModelBase from django.db.models.base import ModelBase
from sapl.base.models import AppConfig as AppConf from sapl.base.models import AppConfig as AppConf
from sapl.base.models import (Autor, ProblemaMigracao, TipoAutor, from sapl.base.models import (Autor, ProblemaMigracao, TipoAutor,
cria_models_tipo_autor) cria_models_tipo_autor)
@ -33,9 +34,11 @@ from sapl.parlamentares.models import (Legislatura, Mandato, Parlamentar,
from sapl.protocoloadm.models import (DocumentoAdministrativo, Protocolo, from sapl.protocoloadm.models import (DocumentoAdministrativo, Protocolo,
StatusTramitacaoAdministrativo) StatusTramitacaoAdministrativo)
from sapl.sessao.models import ExpedienteMateria, OrdemDia, RegistroVotacao from sapl.sessao.models import ExpedienteMateria, OrdemDia, RegistroVotacao
from sapl.settings import PROJECT_DIR from sapl.settings import DATABASES, PROJECT_DIR
from sapl.utils import normalize from sapl.utils import normalize
from .timezonesbrasil import get_timezone
# BASE ###################################################################### # BASE ######################################################################
# apps to be migrated, in app dependency order (very important) # apps to be migrated, in app dependency order (very important)
appconfs = [apps.get_app_config(n) for n in [ appconfs = [apps.get_app_config(n) for n in [
@ -563,6 +566,15 @@ class DataMigrator:
self.data_mudada = {} self.data_mudada = {}
self.choice_valida = {} self.choice_valida = {}
# configura timezone de migração
nome_legado = DATABASES['legacy']['NAME']
match = re.match('sapl_cm_(.*)', nome_legado)
sigla_casa = match.group(1)
with open(os.path.expanduser('~/sapl_dumps/tabela_timezones.yaml'), 'r') as arq:
tabela_timezones = yaml.load(arq)
municipio, uf = tabela_timezones[sigla_casa]
self.timezone = get_timezone(municipio, uf)
def populate_renamed_fields(self, new, old): def populate_renamed_fields(self, new, old):
renames = self.field_renames[type(new)] renames = self.field_renames[type(new)]
@ -603,6 +615,13 @@ class DataMigrator:
if (field_type in ['CharField', 'TextField'] if (field_type in ['CharField', 'TextField']
and value in [None, 'None']): and value in [None, 'None']):
value = '' value = ''
# adiciona timezone faltante aos campos com tempo
# os campos TIMESTAMP do mysql são gravados em UTC
# os DATETIME e TIME não têm timezone
if (field_type in ['DateTimeField', 'TimeField']
and value and not value.tzinfo):
value = self.timezone.localize(value)
setattr(new, field.name, value) setattr(new, field.name, value)
def migrate(self, obj=appconfs, interativo=True): def migrate(self, obj=appconfs, interativo=True):

197
sapl/legacy/timezonesbrasil.py

@ -0,0 +1,197 @@
import unicodedata
from pytz import timezone
UF_PARA_TIMEZONE = '''
AC America/Rio_Branco
AL America/Maceio
AP America/Belem
AM America/Manaus
BA America/Bahia
CE America/Fortaleza
DF America/Sao_Paulo
ES America/Sao_Paulo
GO America/Sao_Paulo
MA America/Fortaleza
MT America/Cuiaba
MS America/Campo_Grande
MG America/Sao_Paulo
PR America/Sao_Paulo
PB America/Fortaleza
PA America/Belem
PE America/Recife
PI America/Fortaleza
RJ America/Sao_Paulo
RN America/Fortaleza
RS America/Sao_Paulo
RO America/Porto_Velho
RR America/Boa_Vista
SC America/Sao_Paulo
SE America/Maceio
SP America/Sao_Paulo
TO America/Araguaina
'''
UF_PARA_TIMEZONE = dict(line.split()
for line in UF_PARA_TIMEZONE.strip().splitlines())
def normalizar_texto(texto):
# baseado em https://gist.github.com/j4mie/557354
norm = unicodedata.normalize('NFKD', texto.lower())
return norm.encode('ASCII', 'ignore').decode('ascii')
# Exceções (Anazonas e Pará):
# leste do Amazonas: America/Manaus
# oeste do Amazonas: America/Eirunepe
# leste do Pará: America/Belem
# oeste do Pará: America/Santarem
# fontes:
# https://en.wikipedia.org/wiki/Time_in_Brazil
# https://www.zeitverschiebung.net/en/timezone/america--belem
# https://www.zeitverschiebung.net/en/timezone/america--santarem
# https://www.zeitverschiebung.net/en/timezone/america--manaus
# https://www.zeitverschiebung.net/en/timezone/america--eirunepe
TZ_CIDADES_AMAZONAS_E_PARA = [
('America/Manaus', '''
Manaus
Itacoatiara
Parintins
Manacapuru
Coari
Tefé
Humaitá
Tabatinga
Rio Preto da Eva
Maués
Carauari
Fonte Boa
São Gabriel da Cachoeira
Boca do Acre
Manicoré
Nova Olinda do Norte
Borba
São Paulo de Olivença
Barreirinha
Codajás
Iranduba
Novo Aripuanã
Urucurituba
Manaquiri
Guajará
Autazes
Santo Antônio do Içá
Urucará
Anori
Pauini
Barcelos
Careiro da Várzea
Canutama
Jutaí
Alvarães
'''),
('America/Eirunepe', '''
Eirunepé
Benjamin Constant
Envira
'''),
('America/Belem', '''
Belém
Ananindeua
Macapá
Marabá
Castanhal
Santana
Abaetetuba
Tucuruí
Paragominas
Bragança
Benevides
Capanema
Breves
Cametá
Salinópolis
Tomé Açu
Capitão Poço
Barcarena
Vigia
São Miguel do Guamá
Conceição do Araguaia
Igarapé Miri
Igarapé Açu
Moju
Portel
Itupiranga
Viseu
Soure
Mocajuba
São Félix do Xingu
Augusto Corrêa
Tucumã
Santa Maria do Pará
Acará
Maracanã
Baião
Curuçá
Marapanim
Oeiras do Pará
São João de Pirabas
Santo Antônio do Tauá
São Caetano de Odivelas
Ourém
Muaná
Afuá
Mazagão
Gurupá
Bujaru
Senador José Porfírio
Irituia
'''),
('America/Santarem', '''
Santarém
Altamira
Itaituba
Oriximiná
Alenquer
Ábidos
Monte Alegre
Almeirim
Terra Santa
Juruti
Porto de Moz
Nhamundá
Prainha
'''),
]
TZ_CIDADES_AMAZONAS_E_PARA = {normalizar_texto(cidade.strip()): tz
for tz, linhas in TZ_CIDADES_AMAZONAS_E_PARA
for cidade in linhas.strip().splitlines()}
def get_nome_timezone(cidade, uf):
tz = UF_PARA_TIMEZONE[uf]
if uf in ['PA', 'AM']:
cidade = normalizar_texto(cidade)
return TZ_CIDADES_AMAZONAS_E_PARA[cidade]
else:
return tz
def get_timezone(cidade, uf):
return timezone(get_nome_timezone(cidade, uf))
def test_get_nome_timezone():
for cidade, uf, tz in [
('Fortaleza', 'CE', 'America/Fortaleza'),
('Salvador', 'BA', 'America/Bahia'),
('Belem', 'PA', 'America/Belem'), # sem acento
('Belém', 'PA', 'America/Belem'), # com acento
('Santarem', 'PA', 'America/Santarem'), # sem acento
('Santarém', 'PA', 'America/Santarem'), # com acento
('Manaus', 'AM', 'America/Manaus'),
('Eirunepe', 'AM', 'America/Eirunepe'), # sem acento
('Eirunepé', 'AM', 'America/Eirunepe'), # com acento
]:
assert get_nome_timezone(cidade, uf) == tz, (cidade, uf, tz)
Loading…
Cancel
Save