Browse Source

Merge branch '474-criar-secao-doc-adm-v2' into 504-autorizacao-autor-proposicao

pull/508/head
Eduardo Calil 9 years ago
parent
commit
990e71437e
  1. 9
      requirements/dev-requirements.txt
  2. 26
      requirements/requirements.txt
  3. 12
      requirements/test-requirements.txt
  4. 19
      sapl/base/migrations/0016_auto_20160701_0940.py
  5. 6
      sapl/base/models.py
  6. 2
      sapl/base/templatetags/menus.py
  7. 13
      sapl/base/views.py
  8. 1
      sapl/crispy_layout_mixin.py
  9. 98
      sapl/parlamentares/forms.py
  10. 21
      sapl/parlamentares/migrations/0022_auto_20160702_1519.py
  11. 4
      sapl/templates/parlamentares/layouts.yaml
  12. 33
      sapl/test_crispy_layout_mixin.py
  13. 14
      sapl/test_general.py
  14. 38
      scripts/atualizar_requirements.py
  15. 9
      scripts/inicializa_grupos_autorizacoes.py

9
requirements/dev-requirements.txt

@ -1,7 +1,8 @@
-r test-requirements.txt -r test-requirements.txt
autopep8==1.2.2 autopep8==1.2.4
beautifulsoup4==4.4.1 beautifulsoup4==4.4.1
django-debug-toolbar==1.4 django-debug-toolbar==1.4
ipdb==0.9.0 ipdb==0.10.1
pygraphviz==1.3rc2 pip-review==0.4
pytest-ipdb==0.1-prerelease2 pygraphviz==1.3.1
pytest-ipdb==0.1-prerelease2

26
requirements/requirements.txt

@ -2,25 +2,25 @@ dj-database-url==0.4.1
django-admin-bootstrapped==2.5.7 django-admin-bootstrapped==2.5.7
django-bootstrap3==7.0.1 django-bootstrap3==7.0.1
django-bower==5.1.0 django-bower==5.1.0
django-braces==1.8.1 django-braces==1.9.0
django-compressor==2.0 django-compressor==2.0
django-crispy-forms==1.6.0 django-crispy-forms==1.6.0
django-extensions==1.6.1 django-extensions==1.6.7
django-extra-views==0.7.1 django-extra-views==0.8.0
django-filter==0.13.0 django-filter==0.13.0
django-floppyforms==1.6.1 django-floppyforms==1.6.2
django-model-utils==2.4 django-model-utils==2.5
django-sass-processor==0.3.4 django-sass-processor==0.4.0
django==1.9.5 django==1.9.7
djangorestframework djangorestframework
easy-thumbnails==2.3 easy-thumbnails==2.3
git+git://github.com/interlegis/trml2pdf.git git+git://github.com/interlegis/trml2pdf.git
libsass==0.11.0 libsass==0.11.1
psycopg2==2.6.1 psycopg2==2.6.2
python-decouple==3.0 python-decouple==3.0
pytz==2016.3 pytz==2016.4
pyyaml==3.11 pyyaml==3.11
rtyaml==0.0.2 rtyaml==0.0.3
unipath==1.1 unipath==1.1
python-magic==0.4.10 python-magic==0.4.12
gunicorn==19.4.5 gunicorn==19.6.0

12
requirements/test-requirements.txt

@ -1,11 +1,11 @@
-r requirements.txt -r requirements.txt
coverage==4.0.3 coverage==4.1
django-webtest django-webtest
flake8==2.5.4 flake8==2.6.2
isort==4.2.5 isort==4.2.5
model_mommy==1.2.6 model-mommy==1.2.6
pep8==1.7.0 pep8==1.7.0
pytest==2.7.2 pytest==2.9.2
pytest-cov==2.2.1 pytest-cov==2.3.0
pytest-django==2.9.1 pytest-django==2.9.1
webtest==2.0.21 webtest==2.0.21

19
sapl/base/migrations/0016_auto_20160701_0940.py

@ -0,0 +1,19 @@
# -*- coding: utf-8 -*-
# Generated by Django 1.9.5 on 2016-07-01 12:40
from __future__ import unicode_literals
from django.db import migrations
class Migration(migrations.Migration):
dependencies = [
('base', '0015_problemamigracao_nome_campo'),
]
operations = [
migrations.AlterModelOptions(
name='casalegislativa',
options={'verbose_name': 'Casa Legislativa', 'verbose_name_plural': 'Casa Legislativa'},
),
]

6
sapl/base/models.py

@ -46,7 +46,11 @@ class CasaLegislativa(models.Model):
class Meta: class Meta:
verbose_name = _('Casa Legislativa') verbose_name = _('Casa Legislativa')
verbose_name_plural = _('Casas Legislativas') verbose_name_plural = _('Casa Legislativa')
def __str__(self):
return _('Casa Legislativa de %(municipio)s') % {
'municipio': self.municipio}
class ProblemaMigracao(models.Model): class ProblemaMigracao(models.Model):

2
sapl/base/templatetags/menus.py

@ -41,7 +41,7 @@ def subnav(context, path=None):
if path: if path:
yaml_path = path yaml_path = path
elif 'subnav_template_name' in context: elif 'subnav_template_name' in context:
yaml_path = context['subnav_template_name'] yaml_path = context['subnav_template_name']
else: else:
yaml_path = '%s/%s' % (app_template, 'subnav.yaml') yaml_path = '%s/%s' % (app_template, 'subnav.yaml')

13
sapl/base/views.py

@ -1,7 +1,10 @@
from django.contrib.auth.mixins import PermissionRequiredMixin from django.contrib.auth.mixins import PermissionRequiredMixin
from django.core.urlresolvers import reverse
from django.http import HttpResponseRedirect
from django.views.generic.base import TemplateView from django.views.generic.base import TemplateView
from sapl.crud.base import Crud, CrudBaseMixin, CrudCreateView, CrudUpdateView from sapl.crud.base import Crud, CrudBaseMixin, CrudCreateView,\
CrudDetailView, CrudUpdateView
from sapl.utils import permissao_tb_aux from sapl.utils import permissao_tb_aux
from .forms import CasaLegislativaForm from .forms import CasaLegislativaForm
@ -33,6 +36,14 @@ class CasaLegislativaCrud(Crud):
permission_required = {'base.change_casalegislativa'} permission_required = {'base.change_casalegislativa'}
form_class = CasaLegislativaForm form_class = CasaLegislativaForm
class DetailView(CrudDetailView):
form_class = CasaLegislativaForm
def get(self, request, *args, **kwargs):
return HttpResponseRedirect(
reverse('sapl.base:casalegislativa_update',
kwargs={'pk': self.kwargs['pk']}))
class HelpView(PermissionRequiredMixin, TemplateView): class HelpView(PermissionRequiredMixin, TemplateView):
# XXX treat non existing template as a 404!!!! # XXX treat non existing template as a 404!!!!

1
sapl/crispy_layout_mixin.py

@ -145,7 +145,6 @@ class CrispyLayoutFormMixin:
def read_yaml_from_file(yaml_layout): def read_yaml_from_file(yaml_layout):
# TODO cache this at application level # TODO cache this at application level
t = template.loader.get_template(yaml_layout) t = template.loader.get_template(yaml_layout)
rendered = t.render() rendered = t.render()

98
sapl/parlamentares/forms.py

@ -1,14 +1,13 @@
from datetime import date from datetime import date, timedelta
from django import forms from django import forms
from django.core.exceptions import ValidationError from django.core.exceptions import ValidationError
from django.db import transaction from django.db import transaction
from django.db.models import Q
from django.forms import ModelForm from django.forms import ModelForm
from django.utils.translation import ugettext_lazy as _ from django.utils.translation import ugettext_lazy as _
from floppyforms.widgets import ClearableFileInput from floppyforms.widgets import ClearableFileInput
from sapl.utils import intervalos_tem_intersecao
from .models import (ComposicaoColigacao, Filiacao, Legislatura, Mandato, from .models import (ComposicaoColigacao, Filiacao, Legislatura, Mandato,
Parlamentar) Parlamentar)
@ -74,59 +73,60 @@ class ParlamentarCreateForm(ParlamentarForm):
def validar_datas(data_filiacao, data_desfiliacao, parlamentar, filiacao): def validar_datas(data_filiacao, data_desfiliacao, parlamentar, filiacao):
# Verifica se data de desfiliacao é anterior a data de filiacao # Verifica se data de desfiliacao é anterior a data de filiacao
if data_desfiliacao and data_desfiliacao < data_filiacao: if data_desfiliacao and data_desfiliacao < data_filiacao:
error_msg = _("A data de desfiliação não pode anterior \ error_msg = _("A data de desfiliação não pode anterior \
à data de filiação") à data de filiação")
return [False, error_msg] return [False, error_msg]
filiacao_atual_id = filiacao.pk filiacoes = parlamentar.filiacao_set.order_by('data')
# recupera filiacoes em ordem crescente de data if not filiacoes.exists():
todas_filiacoes = parlamentar.filiacao_set.all().order_by('data') return [True, '']
filiacoes_id = [parlamentar.pk for parlamentar in todas_filiacoes]
# data ficticia de desfiliacao
# Novo registro inserido com filiacoes ja existentes df_desfiliacao = data_desfiliacao if data_desfiliacao else date.today()
if filiacao_atual_id not in filiacoes_id and len(filiacoes_id) > 0:
ultima_filiacao = todas_filiacoes.last() # se não puder haver filiação no mesmo dia de desfiliação, basta
# Se ultima filiacao aberta e insercao posterior a esta filiacao # retirar os timedelta abaixo
if (not ultima_filiacao.data_desfiliacao and range_livre_exigido = Q(
data_filiacao >= ultima_filiacao.data): data__range=[data_filiacao + timedelta(days=1),
error_msg = _("O parlamentar não pode se filiar \ df_desfiliacao - timedelta(days=1)]) | Q(
a novo partido sem antes se \ data_desfiliacao__range=[data_filiacao + timedelta(days=1),
desfiliar do partido anterior") df_desfiliacao - timedelta(days=1)])
return [False, error_msg]
filiacao_em_edicao_id = filiacao.pk
# checa intervalos de interseccao
error_msg = None error_msg = None
for filiacoes in todas_filiacoes: # filiação em edição não é a última e está sem data de desfiliação
# nao comparar o registro com ele mesmo if not data_desfiliacao and filiacao_em_edicao_id and\
if filiacoes.id != filiacao_atual_id: filiacao_em_edicao_id != filiacoes.last().pk:
error_msg = _("Data de desfiliação do parlamentar não pode ser\
# Se a atualizacao eh para remover a data de desfiliacao ausente, se existirem datas de filiação posteriores.")
if not data_desfiliacao:
# so permite na ultima data (ou a unica) # a filiação que está sendo inclusa não tem data de desfiliação mas
if filiacao_atual_id != filiacoes_id[-1]: # já existe outra sem data de desfiliação
error_msg = _("Data de desfiliação do parlamentar não \ elif not data_desfiliacao and not filiacao_em_edicao_id and\
pode ser ausente, se existirem datas de \ not filiacoes.last().data_desfiliacao:
filiação posteriores") error_msg = _("O parlamentar não pode se filiar a novo partido sem\
return [False, error_msg] antes se desfiliar do partido anterior.")
else:
data_inicio = filiacoes.data if not error_msg:
data_fim = filiacoes.data_desfiliacao # se a filiação é uma edição, a exclui das possibilidades
if filiacao_em_edicao_id:
# Se filiacao ainda em aberto, preenche uma desfiliacao filiacoes = filiacoes.exclude(pk=filiacao_em_edicao_id)
# ficticia para fins de checagem de interseccao
if not data_fim: # testa a intercessão de intervalo com outra filiação
data_fim = date.today() if filiacoes.filter(range_livre_exigido).exists():
error_msg = _("A data de filiação e desfiliação não podem estar\
# finalmente verifica intersecao no intervalo de outro período de filiação.")
if intervalos_tem_intersecao(data_inicio, data_fim,
data_filiacao, data_desfiliacao): if not error_msg:
error_msg = _("A data de filiação e \ # passou pelo teste de intervalo mas a data de filiação é maior que
desfiliação não podem estar no intervalo \ # a ultima que está em aberto
de outro período de filiação") if filiacoes.filter(data_desfiliacao__isnull=True,
break data__lte=data_filiacao).exists():
error_msg = _("Não pode haver um registro de filiação com data de \
filiação igual ou superior a data de filiação em aberto.")
if error_msg: if error_msg:
return [False, error_msg] return [False, error_msg]

21
sapl/parlamentares/migrations/0022_auto_20160702_1519.py

@ -0,0 +1,21 @@
# -*- coding: utf-8 -*-
# Generated by Django 1.9.5 on 2016-07-02 18:19
from __future__ import unicode_literals
from django.db import migrations, models
import django.db.models.deletion
class Migration(migrations.Migration):
dependencies = [
('parlamentares', '0021_merge'),
]
operations = [
migrations.AlterField(
model_name='sessaolegislativa',
name='legislatura',
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='parlamentares.Legislatura', verbose_name='Legislatura'),
),
]

4
sapl/templates/parlamentares/layouts.yaml

@ -62,7 +62,7 @@ ParlamentarCreate:
- biografia - biografia
Filiacao: Filiacao:
{% trans ''Filiações Partidárias '' %}: {% trans 'Filiações Partidárias' %}:
- partido data data_desfiliacao - partido data data_desfiliacao
Mandato: Mandato:
@ -89,5 +89,5 @@ SituacaoMilitar:
- descricao - descricao
ComposicaoColigacao: ComposicaoColigacao:
{% trans ''Nome do Partido'' %}: {% trans 'Nome do Partido' %}:
- partido - partido

33
sapl/test_crispy_layout_mixin.py

@ -1,9 +1,13 @@
from unittest import mock
import rtyaml
from sapl.crispy_layout_mixin import read_layout_from_yaml from sapl.crispy_layout_mixin import read_layout_from_yaml
def test_read_layout_from_yaml(tmpdir): def test_read_layout_from_yaml(tmpdir):
contents = ''' stub_content = '''
ModelName: ModelName:
Cool Legend: Cool Legend:
- name:9 place tiny - name:9 place tiny
@ -12,18 +16,17 @@ ModelName:
More data: More data:
- equalA equalB equalC - equalA equalB equalC
- highlander ''' - highlander '''
file = tmpdir.join('zzz.yaml')
file.write(contents)
expected = [ with mock.patch('sapl.crispy_layout_mixin.read_yaml_from_file') as ryff:
['Cool Legend', ryff.return_value = rtyaml.load(stub_content)
[('name', 9), ('place', 2), ('tiny', 1)], assert read_layout_from_yaml('....', 'ModelName') == [
[('field', 10), ('nature', 2)], ['Cool Legend',
[('kind', 1), ('date', 3), ('unit', 5), ('status', 3)], [('name', 9), ('place', 2), ('tiny', 1)],
], [('field', 10), ('nature', 2)],
['More data', [('kind', 1), ('date', 3), ('unit', 5), ('status', 3)],
[('equalA', 4), ('equalB', 4), ('equalC', 4)], ],
[('highlander', 12)], ['More data',
], [('equalA', 4), ('equalB', 4), ('equalC', 4)],
] [('highlander', 12)],
assert read_layout_from_yaml(file.strpath, 'ModelName') == expected ],
]

14
sapl/test_general.py

@ -7,21 +7,19 @@ from .settings import SAPL_APPS
pytestmark = pytest.mark.django_db pytestmark = pytest.mark.django_db
sapl_appconfs = [apps.get_app_config(n) for n in SAPL_APPS] sapl_appconfs = [apps.get_app_config(n[5:]) for n in SAPL_APPS]
def test_charfiled_textfield(): def test_charfield_textfield():
for app in sapl_appconfs: for app in sapl_appconfs:
for model in app.get_models(): for model in app.get_models():
fields = model._meta.local_fields fields = model._meta.local_fields
for field in fields: for field in fields:
if isinstance(field, (CharField, TextField)): if isinstance(field, (CharField, TextField)):
msg = 'Model = %s || Field = %s - %s - %s' % ( assert not field.null, 'This %s is null: %s.%s' % (
model.__name__, type(field).__name__,
field.attname, model.__name__,
type(field).__name__, field.attname)
field.null)
assert not field.null, msg
def test_str_sanity(): def test_str_sanity():

38
scripts/atualizar_requirements.py

@ -0,0 +1,38 @@
#!/usr/bin/env python
# Este script altera os arquivos requirements/*requirements.txt
# atualizando as versões fixadas neles para coincidirem com as do venv.
#
# Rode esse script após atualizar as dependências do venv usando, p. ex.:
# pip-review
#
# Após usá-lo confira sempre o resultado com `git diff` e teste as mudanças
import glob
import re
import subprocess
freeze_output = subprocess.Popen(
'pip freeze', shell=True,
stdout=subprocess.PIPE).stdout.read().decode('ascii')
freeze = freeze_output.strip().split('\n')
freeze = {name.lower(): version
for name, version in [re.split('==+', s) for s in freeze]}
req_files = glob.glob('requirements/*requirements.txt')
requirements = [(f, open(f).read().strip().split('\n'))
for f in req_files]
def novas_linhas(linhas):
for linha in linhas:
split = re.split('==', linha)
if len(split) == 1:
yield split[0]
else:
nome, versao = split
nome = nome.lower()
yield '%s==%s' % (nome, freeze[nome])
for arq, linhas in requirements:
with open(arq, 'w') as f:
f.writelines(l + '\n' for l in novas_linhas(linhas))

9
scripts/inicializa_grupos_autorizacoes.py

@ -141,38 +141,47 @@ def cria_grupos_permissoes():
# Cria usuarios # Cria usuarios
op_geral_user = User.objects.get_or_create(username='op_geral')[0] op_geral_user = User.objects.get_or_create(username='op_geral')[0]
op_geral_user.set_password('interlegis') op_geral_user.set_password('interlegis')
op_geral_user.save()
op_geral.user_set.add(op_geral_user) op_geral.user_set.add(op_geral_user)
op_materia_user = User.objects.get_or_create(username='op_materia')[0] op_materia_user = User.objects.get_or_create(username='op_materia')[0]
op_materia_user.set_password('interlegis') op_materia_user.set_password('interlegis')
op_materia_user.save()
op_materia.user_set.add(op_materia_user) op_materia.user_set.add(op_materia_user)
op_prot_user = User.objects.get_or_create(username='op_protocolo')[0] op_prot_user = User.objects.get_or_create(username='op_protocolo')[0]
op_prot_user.set_password('interlegis') op_prot_user.set_password('interlegis')
op_prot_user.save()
op_prot.user_set.add(op_prot_user) op_prot.user_set.add(op_prot_user)
op_sessao_user = User.objects.get_or_create(username='op_sessao')[0] op_sessao_user = User.objects.get_or_create(username='op_sessao')[0]
op_sessao_user.set_password('interlegis') op_sessao_user.set_password('interlegis')
op_sessao_user.save()
op_sessao.user_set.add(op_sessao_user) op_sessao.user_set.add(op_sessao_user)
op_comissao_user = User.objects.get_or_create(username='op_comissao')[0] op_comissao_user = User.objects.get_or_create(username='op_comissao')[0]
op_comissao_user.set_password('interlegis') op_comissao_user.set_password('interlegis')
op_comissao_user.save()
op_comissao.user_set.add(op_comissao_user) op_comissao.user_set.add(op_comissao_user)
op_adm_user = User.objects.get_or_create(username='op_adm')[0] op_adm_user = User.objects.get_or_create(username='op_adm')[0]
op_adm_user.set_password('interlegis') op_adm_user.set_password('interlegis')
op_adm_user.save()
op_adm.user_set.add(op_adm_user) op_adm.user_set.add(op_adm_user)
op_norma_user = User.objects.get_or_create(username='op_norma')[0] op_norma_user = User.objects.get_or_create(username='op_norma')[0]
op_norma_user.set_password('interlegis') op_norma_user.set_password('interlegis')
op_norma_user.save()
op_norma.user_set.add(op_norma_user) op_norma.user_set.add(op_norma_user)
op_painel_user = User.objects.get_or_create(username='op_painel')[0] op_painel_user = User.objects.get_or_create(username='op_painel')[0]
op_painel_user.set_password('interlegis') op_painel_user.set_password('interlegis')
op_painel_user.save()
op_painel.user_set.add(op_norma_user) op_painel.user_set.add(op_norma_user)
op_autor_user = User.objects.get_or_create(username='op_autor')[0] op_autor_user = User.objects.get_or_create(username='op_autor')[0]
op_autor_user.set_password('interlegis') op_autor_user.set_password('interlegis')
op_autor_user.save()
op_autor.user_set.add(op_autor_user) op_autor.user_set.add(op_autor_user)
if __name__ == '__main__': if __name__ == '__main__':

Loading…
Cancel
Save