Browse Source

Merge branch '1720-corrigir-migracao-registro-votacao'

Fix #1720
pull/1728/head
Marcio Mazza 7 years ago
parent
commit
68c2099905
  1. 1
      sapl/legacy/migracao_usuarios.py
  2. 190
      sapl/legacy/migration.py
  3. 2
      sapl/legacy/scripts/exporta_zope/exporta_zope.py
  4. 14
      sapl/sessao/models.py
  5. 28
      sapl/sessao/tests/test_sessao.py

1
sapl/legacy/migracao_usuarios.py

@ -92,3 +92,4 @@ def migra_usuarios():
usuario.groups.add(PERFIL_LEGADO_PARA_NOVO[perfil]) usuario.groups.add(PERFIL_LEGADO_PARA_NOVO[perfil])
# apaga arquivo (importante pois contém senhas) # apaga arquivo (importante pois contém senhas)
ARQUIVO_USUARIOS.remove() ARQUIVO_USUARIOS.remove()
print('Usuários migrados com sucesso.')

190
sapl/legacy/migration.py

@ -121,7 +121,9 @@ def warn(msg):
class ForeignKeyFaltando(ObjectDoesNotExist): class ForeignKeyFaltando(ObjectDoesNotExist):
'Uma FK aponta para um registro inexistente' 'Uma FK aponta para um registro inexistente'
pass
def __init__(self, msg=''):
self.msg = msg
@lru_cache() @lru_cache()
@ -350,9 +352,47 @@ def anula_tipos_origem_externa_invalidos():
where tip_origem_externa not in {};''', tipos_validos) where tip_origem_externa not in {};''', tipos_validos)
def get_ids_registros_votacao_para(tabela):
sql = '''
select r.cod_votacao from {} o
inner join registro_votacao r on
o.cod_ordem = r.cod_ordem and o.cod_materia = r.cod_materia
where o.ind_excluido != 1 and r.ind_excluido != 1
order by o.cod_sessao_plen, num_ordem
'''.format(tabela)
return set(primeira_coluna(exec_legado(sql)))
def checa_registros_votacao_ambiguos_e_remove_nao_usados():
"""Interrompe a migração caso restem registros de votação
que apontam para uma ordem_dia e um expediente_materia ao mesmo tempo.
Remove do legado registros de votação que não têm
nem ordem_dia nem expediente_materia associados."""
ordem, expediente = [
get_ids_registros_votacao_para(tabela)
for tabela in ('ordem_dia', 'expediente_materia')]
# interrompe migração se houver registros ambíguos
ambiguos = ordem.intersection(expediente)
assert not ambiguos, '''Existe(m) RegistroVotacao ambíguo(s): {}
Corrija os dados originais antes de migrar!'''.format(
ambiguos)
# exclui registros não usados (zumbis)
todos = set(primeira_coluna(exec_legado(
'select cod_votacao from registro_votacao')))
nao_usados = todos - ordem.union(expediente)
exec_legado_em_subconjunto('''
update registro_votacao set ind_excluido = 1
where cod_votacao in {}''', nao_usados)
def uniformiza_banco(): def uniformiza_banco():
# desliga todas as checagens do mysql exec_legado('SET SESSION sql_mode = "";') # desliga checagens do mysql
exec_legado('SET SESSION sql_mode = "";')
checa_registros_votacao_ambiguos_e_remove_nao_usados()
garante_coluna_no_legado('proposicao', garante_coluna_no_legado('proposicao',
'num_proposicao int(11) NULL') 'num_proposicao int(11) NULL')
@ -437,10 +477,10 @@ relatoria | tip_fim_relatoria = NULL | tip_fim_relatoria = 0
anula_tipos_origem_externa_invalidos() anula_tipos_origem_externa_invalidos()
def iter_sql_records(sql, db): def iter_sql_records(sql):
class Record: class Record:
pass pass
cursor = exec_sql(sql, db) cursor = exec_legado(sql)
fieldnames = [name[0] for name in cursor.description] fieldnames = [name[0] for name in cursor.description]
for row in cursor.fetchall(): for row in cursor.fetchall():
record = Record() record = Record()
@ -534,24 +574,6 @@ def excluir_registrovotacao_duplicados():
assert 0 assert 0
def delete_old(legacy_model, cols_values):
# ajuste necessário por conta de cósigos html em txt_expediente
if legacy_model.__name__ == 'ExpedienteSessaoPlenaria':
cols_values.pop('txt_expediente')
def eq_clause(col, value):
if value is None:
return '{} IS NULL'.format(col)
else:
return '{}="{}"'.format(col, value)
delete_sql = 'delete from {} where {}'.format(
legacy_model._meta.db_table,
' and '.join([eq_clause(col, value)
for col, value in cols_values.items()]))
exec_sql(delete_sql, 'legacy')
def get_last_pk(model): def get_last_pk(model):
last_value = model.objects.all().aggregate(Max('pk')) last_value = model.objects.all().aggregate(Max('pk'))
return last_value['pk__max'] or 0 return last_value['pk__max'] or 0
@ -563,6 +585,12 @@ def reinicia_sequence(model, id):
sequence_name, id)) sequence_name, id))
def get_pk_legado(tabela):
res = exec_legado(
'show index from {} WHERE Key_name = "PRIMARY"'.format(tabela))
return [r[4] for r in res]
class DataMigrator: class DataMigrator:
def __init__(self): def __init__(self):
@ -648,8 +676,8 @@ class DataMigrator:
info('Começando migração: %s...' % obj) info('Começando migração: %s...' % obj)
self._do_migrate(obj) self._do_migrate(obj)
info('Excluindo possíveis duplicações em RegistroVotacao...') # info('Excluindo possíveis duplicações em RegistroVotacao...')
excluir_registrovotacao_duplicados() # excluir_registrovotacao_duplicados()
# recria tipos de autor padrão que não foram criados pela migração # recria tipos de autor padrão que não foram criados pela migração
cria_models_tipo_autor() cria_models_tipo_autor()
@ -677,45 +705,29 @@ class DataMigrator:
def migrate_model(self, model): def migrate_model(self, model):
print('Migrando %s...' % model.__name__) print('Migrando %s...' % model.__name__)
legacy_model_name = self.model_renames.get(model, model.__name__) nome_model = self.model_renames.get(model, model.__name__)
legacy_model = legacy_app.get_model(legacy_model_name) model_legado = legacy_app.get_model(nome_model)
legacy_pk_name = legacy_model._meta.pk.name tabela_legado = model_legado._meta.db_table
campos_pk = get_pk_legado(tabela_legado)
# setup migration strategy for tables with or without a pk
if legacy_pk_name == 'id':
deve_ajustar_sequence_ao_final = False
# There is no pk in the legacy table
def save(new, old):
with reversion.create_revision():
new.save()
reversion.set_comment('Objeto criado pela migração')
# apaga registro do legado if len(campos_pk) == 1:
delete_old(legacy_model, old.__dict__) # a pk no legado tem um único campo
nome_pk = model_legado._meta.pk.name
old_records = model_legado.objects.all().order_by(nome_pk)
old_records = iter_sql_records( def get_id_do_legado(old):
'select * from ' + legacy_model._meta.db_table, 'legacy') return getattr(old, nome_pk)
else: else:
deve_ajustar_sequence_ao_final = True # a pk no legado tem mais de um campo
old_records = iter_sql_records('select * from ' + tabela_legado)
def save(new, old): get_id_do_legado = None
with reversion.create_revision():
# salva new com id de old
new.id = getattr(old, legacy_pk_name)
new.save()
reversion.set_comment('Objeto criado pela migração')
# apaga registro do legado
delete_old(legacy_model, {legacy_pk_name: new.id})
old_records = legacy_model.objects.all().order_by(legacy_pk_name)
ajuste_antes_salvar = AJUSTE_ANTES_SALVAR.get(model) ajuste_antes_salvar = AJUSTE_ANTES_SALVAR.get(model)
ajuste_depois_salvar = AJUSTE_DEPOIS_SALVAR.get(model) ajuste_depois_salvar = AJUSTE_DEPOIS_SALVAR.get(model)
# convert old records to new ones # convert old records to new ones
with transaction.atomic(): with transaction.atomic():
sql_delete_legado = ''
for old in old_records: for old in old_records:
if getattr(old, 'ind_excluido', False): if getattr(old, 'ind_excluido', False):
# não migramos registros marcados como excluídos # não migramos registros marcados como excluídos
@ -725,21 +737,42 @@ class DataMigrator:
self.populate_renamed_fields(new, old) self.populate_renamed_fields(new, old)
if ajuste_antes_salvar: if ajuste_antes_salvar:
ajuste_antes_salvar(new, old) ajuste_antes_salvar(new, old)
except ForeignKeyFaltando: except ForeignKeyFaltando as e:
# tentamos preencher uma FK e o ojeto relacionado # tentamos preencher uma FK e o ojeto relacionado
# não existe # não existe
# então este é um objeo órfão: simplesmente ignoramos # então este é um objeo órfão: simplesmente ignoramos
warn(e.msg)
continue continue
else: else:
save(new, old) if get_id_do_legado:
new.id = get_id_do_legado(old)
# validação do model
new.clean()
# salva novo registro
with reversion.create_revision():
new.save()
reversion.set_comment('Objeto criado pela migração')
# acumula deleção do registro no legado
sql_delete_legado += 'delete from {} where {};\n'.format(
tabela_legado,
' and '.join(
'{} = "{}"'.format(campo,
getattr(old, campo))
for campo in campos_pk))
if ajuste_depois_salvar: if ajuste_depois_salvar:
ajuste_depois_salvar(new, old) ajuste_depois_salvar(new, old)
# reinicia sequence # se configuramos ids explicitamente devemos reiniciar a sequence
if deve_ajustar_sequence_ao_final: if get_id_do_legado:
last_pk = get_last_pk(model) last_pk = get_last_pk(model)
reinicia_sequence(model, last_pk + 1) reinicia_sequence(model, last_pk + 1)
# apaga registros migrados do legado
if sql_delete_legado:
exec_legado(sql_delete_legado)
def migrate(obj=appconfs, interativo=True): def migrate(obj=appconfs, interativo=True):
dm = DataMigrator() dm = DataMigrator()
@ -757,19 +790,21 @@ def adjust_documentoadministrativo(new, old):
protocolo = Protocolo.objects.filter( protocolo = Protocolo.objects.filter(
numero=old.num_protocolo, ano=new.ano) numero=old.num_protocolo, ano=new.ano)
if not protocolo: if not protocolo:
# tentamos encontrar o protocolo no ano seguinte
protocolo = Protocolo.objects.filter( protocolo = Protocolo.objects.filter(
numero=old.num_protocolo, ano=new.ano + 1) numero=old.num_protocolo, ano=new.ano + 1)
if protocolo:
print('PROTOCOLO ENCONTRADO APENAS PARA O ANO SEGUINTE!!!!! ' print('PROTOCOLO ENCONTRADO APENAS PARA O ANO SEGUINTE!!!!! '
'DocumentoAdministrativo: {}, numero_protocolo: {}, ' 'DocumentoAdministrativo: {}, numero_protocolo: {}, '
'ano doc adm: {}'.format( 'ano doc adm: {}'.format(
old.cod_documento, old.num_protocolo, new.ano)) old.cod_documento, old.num_protocolo, new.ano))
if not protocolo: else:
raise ForeignKeyFaltando( warn('Protocolo {} faltando '
'Protocolo {} faltando '
'(referenciado no documento administrativo {}'.format( '(referenciado no documento administrativo {}'.format(
old.num_protocolo, old.cod_documento)) old.num_protocolo, old.cod_documento))
assert len(protocolo) == 1 if protocolo:
new.protocolo = protocolo[0] assert len(protocolo) == 1, 'mais de um protocolo encontrado'
[new.protocolo] = protocolo
def adjust_mandato(new, old): def adjust_mandato(new, old):
@ -882,19 +917,6 @@ def adjust_registrovotacao_antes_salvar(new, old):
new.expediente = expediente_materia[0] new.expediente = expediente_materia[0]
def adjust_registrovotacao_depois_salvar(new, old):
if not new.ordem and not new.expediente:
with reversion.create_revision():
problema = 'RegistroVotacao de PK %s não possui nenhuma OrdemDia'\
' ou ExpedienteMateria.' % old.pk
descricao = 'RevistroVotacao deve ter no mínimo uma ordem do dia'\
' ou expediente vinculado.'
warn(problema + ' => ' + descricao)
save_relation(obj=new, problema=problema,
descricao=descricao, eh_stub=False)
reversion.set_comment('RegistroVotacao sem ordem ou expediente')
def adjust_tipoafastamento(new, old): def adjust_tipoafastamento(new, old):
if old.ind_afastamento == 1: if old.ind_afastamento == 1:
new.indicador = 'A' new.indicador = 'A'
@ -975,8 +997,9 @@ def vincula_autor(new, old, model_relacionado, campo_relacionado, campo_nome):
new.autor_related = model_relacionado.objects.get(pk=pk_rel) new.autor_related = model_relacionado.objects.get(pk=pk_rel)
except ObjectDoesNotExist: except ObjectDoesNotExist:
# ignoramos o autor órfão # ignoramos o autor órfão
raise ForeignKeyFaltando('{} inexiste para autor'.format( raise ForeignKeyFaltando(
model_relacionado._meta.verbose_name)) '{} [pk={}] inexistente para autor'.format(
model_relacionado._meta.verbose_name, pk_rel))
else: else:
new.nome = getattr(new.autor_related, campo_nome) new.nome = getattr(new.autor_related, campo_nome)
return True return True
@ -1010,8 +1033,8 @@ def adjust_autor(new, old):
def adjust_comissao(new, old): def adjust_comissao(new, old):
if not old.dat_extincao and not old.dat_fim_comissao: if not old.dat_extincao and not old.dat_fim_comissao:
new.ativa = True new.ativa = True
elif old.dat_extincao and date.today() < new.data_extincao or \ elif (old.dat_extincao and date.today() < new.data_extincao or
old.dat_fim_comissao and date.today() < new.data_fim_comissao: old.dat_fim_comissao and date.today() < new.data_fim_comissao):
new.ativa = True new.ativa = True
else: else:
new.ativa = False new.ativa = False
@ -1043,15 +1066,14 @@ AJUSTE_DEPOIS_SALVAR = {
NormaJuridica: adjust_normajuridica_depois_salvar, NormaJuridica: adjust_normajuridica_depois_salvar,
OrdemDia: adjust_ordemdia_depois_salvar, OrdemDia: adjust_ordemdia_depois_salvar,
Protocolo: adjust_protocolo_depois_salvar, Protocolo: adjust_protocolo_depois_salvar,
RegistroVotacao: adjust_registrovotacao_depois_salvar,
} }
# CHECKS #################################################################### # CHECKS ####################################################################
def get_ind_excluido(new): def get_ind_excluido(new):
legacy_model = legacy_app.get_model(type(new).__name__) model_legado = legacy_app.get_model(type(new).__name__)
old = legacy_model.objects.get(**{legacy_model._meta.pk.name: new.id}) old = model_legado.objects.get(**{model_legado._meta.pk.name: new.id})
return getattr(old, 'ind_excluido', False) return getattr(old, 'ind_excluido', False)

2
sapl/legacy/scripts/exporta_zope/exporta_zope.py

@ -22,6 +22,7 @@ EXTENSOES = {
'application/msword': '.doc', 'application/msword': '.doc',
'application/pdf': '.pdf', 'application/pdf': '.pdf',
'application/vnd.oasis.opendocument.text': '.odt', 'application/vnd.oasis.opendocument.text': '.odt',
'application/vnd.ms-excel': '.xls',
'application/vnd.openxmlformats-officedocument.wordprocessingml.document': '.docx', # noqa 'application/vnd.openxmlformats-officedocument.wordprocessingml.document': '.docx', # noqa
'application/xml': '.xml', 'application/xml': '.xml',
'text/xml': '.xml', 'text/xml': '.xml',
@ -219,6 +220,7 @@ DUMP_FUNCTIONS = {
'Folder': partial(dump_folder, enum=enumerate_folder), 'Folder': partial(dump_folder, enum=enumerate_folder),
'BTreeFolder2': partial(dump_folder, enum=enumerate_btree), 'BTreeFolder2': partial(dump_folder, enum=enumerate_btree),
'SDE-Document': partial(dump_sde, tipo='sde.document'), 'SDE-Document': partial(dump_sde, tipo='sde.document'),
'StrDoc': partial(dump_sde, tipo='sde.document'),
'SDE-Template': partial(dump_sde, tipo='sde.template'), 'SDE-Template': partial(dump_sde, tipo='sde.template'),
# explicitamente ignorados # explicitamente ignorados

14
sapl/sessao/models.py

@ -1,7 +1,11 @@
from operator import xor
import reversion import reversion
from django.core.exceptions import ValidationError
from django.db import models from django.db import models
from django.utils.translation import ugettext_lazy as _ from django.utils.translation import ugettext_lazy as _
from model_utils import Choices from model_utils import Choices
from sapl.base.models import Autor from sapl.base.models import Autor
from sapl.materia.models import MateriaLegislativa from sapl.materia.models import MateriaLegislativa
from sapl.parlamentares.models import (CargoMesa, Legislatura, Parlamentar, from sapl.parlamentares.models import (CargoMesa, Legislatura, Parlamentar,
@ -429,6 +433,16 @@ class RegistroVotacao(models.Model):
'votacao': self.tipo_resultado_votacao, 'votacao': self.tipo_resultado_votacao,
'materia': self.materia} 'materia': self.materia}
def clean(self):
"""Exatamente um dos campos ordem ou expediente deve estar preenchido.
"""
# TODO remover esse método quando OrdemDia e ExpedienteMateria
# forem reestruturados e os campos ordem e expediente forem unificados
if not xor(bool(self.ordem), bool(self.expediente)):
raise ValidationError(
'RegistroVotacao deve ter exatamente um dos campos '
'ordem ou expediente deve estar preenchido')
@reversion.register() @reversion.register()
class VotoParlamentar(models.Model): # RegistroVotacaoParlamentar class VotoParlamentar(models.Model): # RegistroVotacaoParlamentar

28
sapl/sessao/tests/test_sessao.py

@ -1,11 +1,13 @@
import pytest import pytest
from django.core.exceptions import ValidationError
from django.utils.translation import ugettext_lazy as _ from django.utils.translation import ugettext_lazy as _
from model_mommy import mommy from model_mommy import mommy
from sapl.materia.models import MateriaLegislativa, TipoMateriaLegislativa from sapl.materia.models import MateriaLegislativa, TipoMateriaLegislativa
from sapl.parlamentares.models import Legislatura, Partido, SessaoLegislativa from sapl.parlamentares.models import Legislatura, Partido, SessaoLegislativa
from sapl.sessao import forms from sapl.sessao import forms
from sapl.sessao.models import (ExpedienteMateria, SessaoPlenaria, from sapl.sessao.models import (ExpedienteMateria, OrdemDia, RegistroVotacao,
TipoSessaoPlenaria) SessaoPlenaria, TipoSessaoPlenaria)
def test_valida_campos_obrigatorios_sessao_plenaria_form(): def test_valida_campos_obrigatorios_sessao_plenaria_form():
@ -138,3 +140,25 @@ def test_expediente_materia_form_valido():
}, },
instance=instance) instance=instance)
assert form.is_valid() assert form.is_valid()
@pytest.mark.django_db(transaction=False)
def test_registro_votacao_tem_ordem_xor_expediente():
def registro_votacao_com(ordem, expediente):
return mommy.make(RegistroVotacao, ordem=ordem, expediente=expediente)
ordem = mommy.make(OrdemDia)
expediente = mommy.make(ExpedienteMateria)
# a validação funciona com exatamente um dos campos preenchido
registro_votacao_com(ordem, None).full_clean()
registro_votacao_com(None, expediente).full_clean()
# a validação NÃO funciona quando nenhum deles é preenchido
with pytest.raises(ValidationError):
registro_votacao_com(None, None).full_clean()
# a validação NÃO funciona quando ambos são preenchidos
with pytest.raises(ValidationError):
registro_votacao_com(ordem, expediente).full_clean()

Loading…
Cancel
Save