Browse Source

Documento Acessório Restrito em Documento Administrativo e em Matéria Legislativa

pull/3613/head
cristian-longhi 2 years ago
parent
commit
62d29f6846
  1. 15
      sapl/api/views_materia.py
  2. 5
      sapl/api/views_protocoloadm.py
  3. 8
      sapl/materia/forms.py
  4. 5
      sapl/materia/models.py
  5. 60
      sapl/materia/views.py
  6. 10
      sapl/protocoloadm/forms.py
  7. 5
      sapl/protocoloadm/models.py
  8. 47
      sapl/protocoloadm/views.py
  9. 71
      sapl/templates/materia/documentoacessorio_list.html
  10. 2
      sapl/templates/materia/layouts.yaml
  11. 2
      sapl/templates/protocoloadm/layouts.yaml

15
sapl/api/views_materia.py

@ -8,7 +8,7 @@ from drfautoapi.drfautoapi import ApiViewSetConstrutor, \
customize, wrapper_queryset_response_for_drf_action customize, wrapper_queryset_response_for_drf_action
from sapl.api.permissions import SaplModelPermissions from sapl.api.permissions import SaplModelPermissions
from sapl.materia.models import TipoMateriaLegislativa, Tramitacao,\ from sapl.materia.models import TipoMateriaLegislativa, Tramitacao,\
MateriaLegislativa, Proposicao MateriaLegislativa, Proposicao, DocumentoAcessorio
ApiViewSetConstrutor.build_class( ApiViewSetConstrutor.build_class(
@ -111,6 +111,19 @@ class _MateriaLegislativaViewSet:
return self.list(request, *args, **kwargs) return self.list(request, *args, **kwargs)
@customize(DocumentoAcessorio)
class _DocumentoAcessorioViewSet:
def get_queryset(self):
user = self.request.user
qs = super().get_queryset()
if user.is_anonymous or 'materia.change_documentoacessorio' not in user.get_all_permissions():
qs = qs.exclude(restrito=True)
return qs
@customize(TipoMateriaLegislativa) @customize(TipoMateriaLegislativa)
class _TipoMateriaLegislativaViewSet: class _TipoMateriaLegislativaViewSet:

5
sapl/api/views_protocoloadm.py

@ -60,9 +60,12 @@ class _DocumentoAcessorioAdministrativoViewSet:
def get_queryset(self): def get_queryset(self):
qs = super().get_queryset() qs = super().get_queryset()
user = self.request.user
if self.request.user.is_anonymous: if user.is_anonymous or 'protocoloadm.change_documentoacessorioadministrativo' not in user.get_all_permissions():
# if self.request.user.is_anonymous:
qs = qs.exclude(documento__restrito=True) qs = qs.exclude(documento__restrito=True)
qs = qs.exclude(restrito=True)
return qs return qs

8
sapl/materia/forms.py

@ -330,11 +330,17 @@ class AcompanhamentoMateriaForm(GoogleRecapthaMixin, ModelForm):
class DocumentoAcessorioForm(FileFieldCheckMixin, ModelForm): class DocumentoAcessorioForm(FileFieldCheckMixin, ModelForm):
data = forms.DateField(required=True) data = forms.DateField(required=True)
restrito = forms.ChoiceField(
label=_('Documento Restrito?'),
widget=forms.RadioSelect(),
choices=YES_NO_CHOICES,
initial=False)
class Meta: class Meta:
model = DocumentoAcessorio model = DocumentoAcessorio
fields = ['tipo', 'nome', 'data', 'autor', fields = ['tipo', 'nome', 'data', 'autor',
'ementa', 'indexacao', 'arquivo'] 'ementa', 'indexacao', 'arquivo',
'restrito', 'justificativa_restricao']
def clean(self): def clean(self):
super(DocumentoAcessorioForm, self).clean() super(DocumentoAcessorioForm, self).clean()

5
sapl/materia/models.py

@ -554,6 +554,11 @@ class DocumentoAcessorio(models.Model):
proposicao = GenericRelation('Proposicao', related_query_name='proposicao') proposicao = GenericRelation('Proposicao', related_query_name='proposicao')
data_ultima_atualizacao = models.DateTimeField( data_ultima_atualizacao = models.DateTimeField(
blank=True, null=True, auto_now=True, verbose_name=_('Data')) blank=True, null=True, auto_now=True, verbose_name=_('Data'))
restrito = models.BooleanField(default=False,
verbose_name=_('Restrito'),
blank=True)
justificativa_restricao = models.TextField(
blank=True, verbose_name=_('Justificativa de Restrição'))
class Meta: class Meta:
verbose_name = _('Documento Acessório') verbose_name = _('Documento Acessório')

60
sapl/materia/views.py

@ -1588,6 +1588,46 @@ class DocumentoAcessorioCrud(MasterDetailCrud):
context = super(UpdateView, self).get_context_data(**kwargs) context = super(UpdateView, self).get_context_data(**kwargs)
return context return context
class ListView(MasterDetailCrud.ListView):
def get_queryset(self):
qs = super(MasterDetailCrud.ListView, self).get_queryset()
kwargs = {self.crud.parent_field: self.kwargs['pk']}
exibir_restritos = 'materia.change_documentoacessorio' in self.request.user.get_all_permissions()
if 'o' in self.request.GET:
o = self.request.GET['o']
indice_field = abs(int(o)) - 1
if '-' in o:
order_by = '-' + self.list_field_names[indice_field]
else:
order_by = self.list_field_names[indice_field]
if exibir_restritos:
return qs.filter(**kwargs).order_by(order_by, '-data', '-id')
else:
return qs.filter(**kwargs).order_by('restrito', order_by, '-data', '-id')
if exibir_restritos:
return qs.filter(**kwargs).order_by('-data', '-id')
else:
return qs.filter(**kwargs).order_by('restrito', '-data', '-id')
class DetailView(MasterDetailCrud.DetailView):
layout_key = 'DocumentoAcessorioAdministrativo'
template_name = "materia/documentoacessorio_detail.html"
def get_context_data(self, **kwargs):
context = super().get_context_data(**kwargs)
context['AppConfig'] = sapl.base.models.AppConfig.objects.all().last()
context['user'] = self.request.user
doc = DocumentoAcessorio.objects.get(
pk=self.kwargs['pk']
)
context['object'] = doc
if doc.restrito and not 'materia.change_documentoacessorio' in self.request.user.get_all_permissions():
context['title'] = 'Documento Restrito'
return context
class AutoriaCrud(MasterDetailCrud): class AutoriaCrud(MasterDetailCrud):
model = Autoria model = Autoria
@ -2867,13 +2907,15 @@ class TipoMateriaCrud(CrudAux):
return fv return fv
def create_zip_docacessorios(materia): def create_zip_docacessorios(materia, excluir_restritos):
""" """
Creates in memory zip files Creates in memory zip files
""" """
logger = logging.getLogger(__name__) logger = logging.getLogger(__name__)
docs = materia.documentoacessorio_set. \ docs = materia.documentoacessorio_set. \
all().values_list('arquivo', flat=True) all().values_list('arquivo', flat=True)
if excluir_restritos:
docs = docs.filter(restrito=False)
if not docs: if not docs:
return None, None return None, None
@ -2903,10 +2945,14 @@ def create_zip_docacessorios(materia):
def get_zip_docacessorios(request, pk): def get_zip_docacessorios(request, pk):
logger = logging.getLogger(__name__) logger = logging.getLogger(__name__)
username = 'Usuário anônimo' if request.user.is_anonymous else request.user.username username = 'Usuário anônimo' if request.user.is_anonymous else request.user.username
if request.user.is_anonymous or not 'protocoloadm.change_documentoacessorioadministrativo' in request.user.get_all_permissions():
excluir_restritos = True
else:
excluir_restritos = False
materia = get_object_or_404(MateriaLegislativa, pk=pk) materia = get_object_or_404(MateriaLegislativa, pk=pk)
data = None data = None
try: try:
external_name, data = create_zip_docacessorios(materia) external_name, data = create_zip_docacessorios(materia, excluir_restritos)
logger.info( logger.info(
"user= {}. Gerou o zip compilado de documento acessorios".format(username)) "user= {}. Gerou o zip compilado de documento acessorios".format(username))
except FileNotFoundError: except FileNotFoundError:
@ -2935,13 +2981,15 @@ def get_zip_docacessorios(request, pk):
return response return response
def create_pdf_docacessorios(materia): def create_pdf_docacessorios(materia,excluir_restritos):
""" """
Creates a unified in memory PDF file Creates a unified in memory PDF file
""" """
logger = logging.getLogger(__name__) logger = logging.getLogger(__name__)
docs = materia.documentoacessorio_set. \ docs = materia.documentoacessorio_set. \
all().values_list('arquivo', flat=True) all().values_list('arquivo', flat=True)
if excluir_restritos:
docs = docs.filter(restrito=False)
if not docs: if not docs:
return None, None return None, None
@ -2976,8 +3024,12 @@ def get_pdf_docacessorios(request, pk):
materia = get_object_or_404(MateriaLegislativa, pk=pk) materia = get_object_or_404(MateriaLegislativa, pk=pk)
logger = logging.getLogger(__name__) logger = logging.getLogger(__name__)
username = 'Usuário anônimo' if request.user.is_anonymous else request.user.username username = 'Usuário anônimo' if request.user.is_anonymous else request.user.username
if request.user.is_anonymous or not 'protocoloadm.change_documentoacessorioadministrativo' in request.user.get_all_permissions():
excluir_restritos = True
else:
excluir_restritos = False
try: try:
external_name, data = create_pdf_docacessorios(materia) external_name, data = create_pdf_docacessorios(materia, excluir_restritos)
logger.info( logger.info(
"user= {}. Gerou o pdf compilado de documento acessorios".format(username)) "user= {}. Gerou o pdf compilado de documento acessorios".format(username))
except FileNotFoundError: except FileNotFoundError:

10
sapl/protocoloadm/forms.py

@ -623,14 +623,22 @@ class ProtocoloMateriaForm(ModelForm):
class DocumentoAcessorioAdministrativoForm(FileFieldCheckMixin, ModelForm): class DocumentoAcessorioAdministrativoForm(FileFieldCheckMixin, ModelForm):
restrito = forms.ChoiceField(
label=_('Documento Restrito?'),
widget=forms.RadioSelect(),
choices=YES_NO_CHOICES,
initial=False)
class Meta: class Meta:
model = DocumentoAcessorioAdministrativo model = DocumentoAcessorioAdministrativo
fields = ['tipo', fields = ['tipo',
'nome', 'nome',
'restrito',
'data', 'data',
'autor', 'autor',
'arquivo', 'arquivo',
'assunto'] 'assunto',
'justificativa_restricao']
widgets = { widgets = {
'data': forms.DateInput(format='%d/%m/%Y') 'data': forms.DateInput(format='%d/%m/%Y')

5
sapl/protocoloadm/models.py

@ -362,6 +362,11 @@ class DocumentoAcessorioAdministrativo(models.Model):
assunto = models.TextField( assunto = models.TextField(
blank=True, verbose_name=_('Assunto')) blank=True, verbose_name=_('Assunto'))
indexacao = models.TextField(blank=True) indexacao = models.TextField(blank=True)
restrito = models.BooleanField(default=False,
verbose_name=_('Restrito'),
blank=True)
justificativa_restricao = models.TextField(
blank=True, verbose_name=_('Justificativa de Restrição'))
class Meta: class Meta:
verbose_name = _('Documento Acessório') verbose_name = _('Documento Acessório')

47
sapl/protocoloadm/views.py

@ -329,10 +329,15 @@ class AcompanhamentoDocumentoView(CreateView):
class DocumentoAdministrativoMixin: class DocumentoAdministrativoMixin:
def has_permission(self): def has_permission(self):
if self.model == DocumentoAcessorioAdministrativo and 'docadm/' + self.kwargs['pk'] in str(self.request):
doc_adm = DocumentoAdministrativo.objects.get(id=self.kwargs['pk'])
if doc_adm.restrito and not 'protocoloadm.change_documentoacessorioadministrativo' in self.request.user.get_all_permissions():
return False
app_config = AppConfig.objects.last() app_config = AppConfig.objects.last()
if app_config and app_config.documentos_administrativos == 'O': if app_config and app_config.documentos_administrativos == 'O':
return True return True
return super().has_permission() return super().has_permission()
@ -1390,15 +1395,51 @@ class DocumentoAcessorioAdministrativoCrud(MasterDetailCrud):
form_class = DocumentoAcessorioAdministrativoForm form_class = DocumentoAcessorioAdministrativoForm
class ListView(DocumentoAdministrativoMixin, MasterDetailCrud.ListView): class ListView(DocumentoAdministrativoMixin, MasterDetailCrud.ListView):
template_name = "protocoloadm/documentoacessorioadministrativo_list.html"
def get_queryset(self): def get_queryset(self):
qs = super(MasterDetailCrud.ListView, self).get_queryset() qs = super(MasterDetailCrud.ListView, self).get_queryset()
kwargs = {self.crud.parent_field: self.kwargs['pk']} kwargs = {self.crud.parent_field: self.kwargs['pk']}
return qs.filter(**kwargs).order_by('-data', '-id') exibir_restritos = 'protocoloadm.change_documentoacessorioadministrativo' in self.request.user.get_all_permissions()
if 'o' in self.request.GET:
o = self.request.GET['o']
indice_field = abs(int(o)) - 1
if '-' in o:
order_by = '-' + self.list_field_names[indice_field]
else:
order_by = self.list_field_names[indice_field]
if exibir_restritos:
return qs.filter(**kwargs).order_by(order_by, '-data', '-id')
else:
return qs.filter(**kwargs).order_by('restrito', order_by, '-data', '-id')
if exibir_restritos:
return qs.filter(**kwargs).order_by('-data', '-id')
else:
return qs.filter(**kwargs).order_by('restrito', '-data', '-id')
class DetailView(DocumentoAdministrativoMixin, class DetailView(DocumentoAdministrativoMixin,
MasterDetailCrud.DetailView): MasterDetailCrud.DetailView):
pass layout_key = 'DocumentoAcessorioAdministrativo'
template_name = "protocoloadm/documentoacessorioadministrativo_detail.html"
def get_context_data(self, **kwargs):
context = super().get_context_data(**kwargs)
context['AppConfig'] = sapl.base.models.AppConfig.objects.all().last()
context['user'] = self.request.user
doc_acessorio = DocumentoAcessorioAdministrativo.objects.get(
pk=self.kwargs['pk'])
doc_adm = DocumentoAdministrativo.objects.get(
id=doc_acessorio.documento_id)
if not doc_adm.restrito:
context['doc_adm_restrito'] = False
context['object'] = doc_acessorio
else:
context['doc_adm_restrito'] = True
if (doc_adm.restrito or doc_acessorio.restrito) and not 'protocoloadm.change_documentoacessorioadministrativo' in self.request.user.get_all_permissions():
context['title'] = 'Documento Restrito'
return context
def atualizar_numero_documento(request): def atualizar_numero_documento(request):

71
sapl/templates/materia/documentoacessorio_list.html

@ -1,5 +1,74 @@
{% extends "crud/list.html" %} {% extends "crud/list.html" %}
{% load i18n %} {% load i18n common_tags %}
{% block container_table_list %}
{% if not rows %}
<p>{{ NO_ENTRIES_MSG }}</p>
{% else %}
<div class="container-table">
<div class="result-count">{% blocktrans with verbose_name_plural=view.verbose_name_plural %}Total de {{ verbose_name_plural }}: <strong>{{count}}</strong>{% endblocktrans %}</div>
<table class="table table-striped table-hover table-link-ordering">
<thead>
<tr>
{% for name in headers %}
{% comment %}
{% if name != 'Restrito' and name != 'Justificativa de Restrição' %}
{% endcomment %}
<th>
{% if view.ordered_list %}
<a title="{% trans 'Clique para alterar a ordem a listagem'%}" href="?o={% if 'o' not in request.GET and forloop.counter == 1 or 'o' in request.GET and forloop.counter|safe == request.GET.o %}-{%endif%}{{forloop.counter}}{{ordering_url}}">
{{ name }}
{% if 'o' in request.GET or not view.ordering %}
{% if 'o' not in request.GET and forloop.counter == 1 or 'o' in request.GET and forloop.counter|safe == request.GET.o %}
<span class="caret top" title="{% trans 'Listado na Ordem Ascendente'%}"></span>
{% elif 'o' in request.GET and forloop.counter == request.GET.o|str2intabs %}
<span class="caret" title="{% trans 'Listado na Ordem Descendente'%}"></span>
{% endif %}
{% endif %}
</a>
{% else %}
{{ name }}
{% endif %}
</th>
{% comment %}
{% endif %}
{% endcomment %}
{% endfor %}
</tr>
</thead>
<tbody>
{% for doc in object_list %}
<tr>
{% if not doc.restrito or 'materia.change_documentoacessorio' in request.user.get_all_permissions %}
<td>
<a href="{% url 'sapl.materia:documentoacessorio_detail' doc.id %}">{{ doc.nome }}</a>
</td>
<td>
{{ doc.tipo }}
</td>
<td>
{% if doc.data %}{{ doc.data }}{% endif %}
</td>
<td>
{% if doc.autor %}{{ doc.autor }}{% endif %}
</td>
<td>
{% if doc.arquivo %}<a href="{{ doc.arquivo.url }}">Ver arquivo{% endif %}
</td>
{% else %}
<td colspan=5>
<strong style="color: red">Documento Restrito</strong></br>{{ doc.justificativa_restricao|safe }}
</td>
{% endif %}
</tr>
{% endfor %}
</tbody>
</table>
</div>
{% endif %}
{% endblock container_table_list %}
{% block base_content %} {% block base_content %}
{{ block.super }} {{ block.super }}
<div style="display:flex;padding-left: 600px;padding-top: 10px;"> <div style="display:flex;padding-left: 600px;padding-top: 10px;">

2
sapl/templates/materia/layouts.yaml

@ -75,6 +75,8 @@ DocumentoAcessorio:
- ementa - ementa
- indexacao - indexacao
- arquivo - arquivo
- restrito
- justificativa_restricao
Numeracao: Numeracao:
{% trans 'Numeração' %}: {% trans 'Numeração' %}:

2
sapl/templates/protocoloadm/layouts.yaml

@ -24,6 +24,8 @@ DocumentoAcessorioAdministrativo:
- nome data - nome data
- arquivo - arquivo
- assunto - assunto
- restrito
- justificativa_restricao
StatusTramitacaoAdministrativo: StatusTramitacaoAdministrativo:
{% trans 'Status Tramitação Administrativo' %}: {% trans 'Status Tramitação Administrativo' %}:

Loading…
Cancel
Save