From 28d11bc31baa0c84d700504a7196eedeae6a1d16 Mon Sep 17 00:00:00 2001 From: Serghei Mihai Date: Tue, 9 May 2023 14:42:45 +0200 Subject: [PATCH] code refactoring (#12412) --- .../__init__.py | 29 ------- passerelle_montpellier_encombrants/models.py | 76 +++++++++++++++---- .../base.html | 8 -- .../collectday_list.html | 7 +- .../commune_list.html | 7 +- .../detail.html | 17 ----- .../sector_list.html | 8 +- passerelle_montpellier_encombrants/urls.py | 35 --------- passerelle_montpellier_encombrants/utils.py | 3 +- passerelle_montpellier_encombrants/views.py | 49 ------------ tests/test_main.py | 62 +++++++++++---- 11 files changed, 132 insertions(+), 169 deletions(-) delete mode 100644 passerelle_montpellier_encombrants/templates/passerelle_montpellier_encombrants/base.html diff --git a/passerelle_montpellier_encombrants/__init__.py b/passerelle_montpellier_encombrants/__init__.py index f1a0cf8..e69de29 100644 --- a/passerelle_montpellier_encombrants/__init__.py +++ b/passerelle_montpellier_encombrants/__init__.py @@ -1,29 +0,0 @@ -# passerelle-montpellier-encombrants -# Copyright (C) 2015 Entr'ouvert -# -# This program is free software: you can redistribute it and/or modify it -# under the terms of the GNU Affero General Public License as published -# by the Free Software Foundation, either version 3 of the License, or -# (at your option) any later version. -# -# This program is distributed in the hope that it will be useful, -# but WITHOUT ANY WARRANTY; without even the implied warranty of -# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -# GNU Affero General Public License for more details. -# -# You should have received a copy of the GNU Affero General Public License -# along with this program. If not, see . - -import django.apps - - -class AppConfig(django.apps.AppConfig): - name = 'passerelle_montpellier_encombrants' - - def get_after_urls(self): - from . import urls - - return urls.urlpatterns - - -default_app_config = 'passerelle_montpellier_encombrants.AppConfig' diff --git a/passerelle_montpellier_encombrants/models.py b/passerelle_montpellier_encombrants/models.py index 16445ab..8c1eaf8 100644 --- a/passerelle_montpellier_encombrants/models.py +++ b/passerelle_montpellier_encombrants/models.py @@ -14,10 +14,14 @@ # You should have received a copy of the GNU Affero General Public License # along with this program. If not, see . +import datetime + from django.db import models -from django.urls import reverse from django.utils.translation import ugettext_lazy as _ from passerelle.base.models import BaseResource +from passerelle.utils.api import endpoint + +from .utils import get_sector class EncombrantsManagement(BaseResource): @@ -26,20 +30,34 @@ class EncombrantsManagement(BaseResource): class Meta: verbose_name = 'Gestion des encombrants Montpellier' - @classmethod - def get_icon_class(cls): - return 'ressources' + manager_view_template_name = 'passerelle_montpellier_encombrants/detail.html' - @classmethod - def get_verbose_name(cls): - return cls._meta.verbose_name - - def get_absolute_url(self): - return reverse('montpellier-encombrants-view', kwargs={'slug': self.slug}) - - @classmethod - def get_add_url(cls): - return reverse('montpellier-encombrants-add') + @endpoint( + perm='can_access', + pattern=r'^(?P\w+)$', + parameters={ + 'adresse': {'description': _('Address')}, + 'limit': {'description': _('Results number limit')}, + }, + ) + def available(self, request, insee, adresse=None, limit=None): + sector = get_sector(insee, adresse) + if sector: + if datetime.datetime.today().weekday() == 4: + tomorrow = datetime.datetime.today() + datetime.timedelta(days=3) + elif datetime.datetime.today().weekday() == 5: + tomorrow = datetime.datetime.today() + datetime.timedelta(days=2) + else: + tomorrow = datetime.datetime.today() + datetime.timedelta(days=1) + collect_days = sector.collectday_set.filter(date__gt=tomorrow).order_by('date') + else: + collect_days = [] + result = [ + {'id': x.date.strftime('%Y-%m-%d'), 'text': x.date.strftime('%d/%m/%Y')} for x in collect_days + ] + if limit is not None and limit.isnumeric(): + result = result[: int(limit)] + return {'data': result} class Sector(models.Model): @@ -50,6 +68,14 @@ class Sector(models.Model): class Meta: verbose_name = _('Sector') + @classmethod + def get_verbose_name(cls): + return cls._meta.verbose_name + + @property + def title(self): + return self + def __str__(self): communes = Commune.objects.filter(sector=self) if not communes: @@ -62,6 +88,17 @@ class Commune(models.Model): insee = models.CharField(max_length=10, blank=False, verbose_name=_('INSEE Code')) sector = models.ForeignKey(Sector, verbose_name=_('Sector'), on_delete=models.CASCADE) + class Meta: + verbose_name = _('Commune') + + @classmethod + def get_verbose_name(cls): + return cls._meta.verbose_name + + @property + def title(self): + return self + def __str__(self): return '%s (%s)' % (self.name, self.insee) @@ -78,6 +115,17 @@ class CollectDay(models.Model): sector = models.ForeignKey(Sector, verbose_name=_('Sector'), on_delete=models.CASCADE) date = models.DateField(verbose_name=_('Date')) + class Meta: + verbose_name = _('Collect day') + + @classmethod + def get_verbose_name(cls): + return cls._meta.verbose_name + + @property + def title(self): + return self + class Meta: ordering = ['date'] unique_together = ('sector', 'date') diff --git a/passerelle_montpellier_encombrants/templates/passerelle_montpellier_encombrants/base.html b/passerelle_montpellier_encombrants/templates/passerelle_montpellier_encombrants/base.html deleted file mode 100644 index 2476551..0000000 --- a/passerelle_montpellier_encombrants/templates/passerelle_montpellier_encombrants/base.html +++ /dev/null @@ -1,8 +0,0 @@ -{% extends "passerelle/manage.html" %} - -{% block breadcrumb %} - {{ block.super }} - {% if object.id %} - {{ object.title }} - {% endif %} -{% endblock %} diff --git a/passerelle_montpellier_encombrants/templates/passerelle_montpellier_encombrants/collectday_list.html b/passerelle_montpellier_encombrants/templates/passerelle_montpellier_encombrants/collectday_list.html index 240336f..7ecaa1e 100644 --- a/passerelle_montpellier_encombrants/templates/passerelle_montpellier_encombrants/collectday_list.html +++ b/passerelle_montpellier_encombrants/templates/passerelle_montpellier_encombrants/collectday_list.html @@ -1,6 +1,11 @@ -{% extends "passerelle_montpellier_encombrants/base.html" %} +{% extends "passerelle_montpellier_encombrants/detail.html" %} {% load i18n %} +{% block breadcrumb %} + {{ block.super }} + {{ object.title }} +{% endblock %} + {% block appbar %}

{% trans 'Collect Days' %}

{% trans 'New' %} diff --git a/passerelle_montpellier_encombrants/templates/passerelle_montpellier_encombrants/commune_list.html b/passerelle_montpellier_encombrants/templates/passerelle_montpellier_encombrants/commune_list.html index 6b24cea..c9e3610 100644 --- a/passerelle_montpellier_encombrants/templates/passerelle_montpellier_encombrants/commune_list.html +++ b/passerelle_montpellier_encombrants/templates/passerelle_montpellier_encombrants/commune_list.html @@ -1,6 +1,11 @@ -{% extends "passerelle_montpellier_encombrants/base.html" %} +{% extends "passerelle_montpellier_encombrants/detail.html" %} {% load i18n %} +{% block breadcrumb %} + {{ block.super }} + {{ object.title }} +{% endblock %} + {% block appbar %}

{% trans 'Communes' %}

{% trans 'New' %} diff --git a/passerelle_montpellier_encombrants/templates/passerelle_montpellier_encombrants/detail.html b/passerelle_montpellier_encombrants/templates/passerelle_montpellier_encombrants/detail.html index 77446a3..3682216 100644 --- a/passerelle_montpellier_encombrants/templates/passerelle_montpellier_encombrants/detail.html +++ b/passerelle_montpellier_encombrants/templates/passerelle_montpellier_encombrants/detail.html @@ -1,14 +1,6 @@ {% extends "passerelle/manage/service_view.html" %} {% load i18n passerelle %} -{% block endpoints %} - - -{% endblock %} - {% block description %} {% if perms.passerelle_montpellier_encombrants.change_passerelle_montpellier_encombrants %}
@@ -19,15 +11,6 @@
  • {% trans 'Communes' %}
  • {% trans 'Collect Days' %}
  • -
    {% endif %} {% endblock %} - -{% block security %} -

    - {% trans 'Accessing the listings is open, but posting requests is limited to the following API users:' %} -

    - - {% access_rights_table resource=object permission='can_post_request' %} -{% endblock %} diff --git a/passerelle_montpellier_encombrants/templates/passerelle_montpellier_encombrants/sector_list.html b/passerelle_montpellier_encombrants/templates/passerelle_montpellier_encombrants/sector_list.html index 9a76245..e60d4d7 100644 --- a/passerelle_montpellier_encombrants/templates/passerelle_montpellier_encombrants/sector_list.html +++ b/passerelle_montpellier_encombrants/templates/passerelle_montpellier_encombrants/sector_list.html @@ -1,6 +1,12 @@ -{% extends "passerelle_montpellier_encombrants/base.html" %} +{% extends "passerelle_montpellier_encombrants/detail.html" %} {% load i18n %} +{% block breadcrumb %} + {{ block.super }} + {{ object.title }} +{% endblock %} + + {% block appbar %}

    {% trans 'Sectors' %}

    {% trans 'New' %} diff --git a/passerelle_montpellier_encombrants/urls.py b/passerelle_montpellier_encombrants/urls.py index 07f4db8..a477a34 100644 --- a/passerelle_montpellier_encombrants/urls.py +++ b/passerelle_montpellier_encombrants/urls.py @@ -20,31 +20,7 @@ from passerelle.urls_utils import app_enabled, decorated_includes, required from .views import * -public_urlpatterns = [ - re_path( - r'^(?P[\w,-]+)/$', - EncombrantsManagementDetailView.as_view(), - name='montpellier-encombrants-view', - ), - re_path( - r'^(?P[\w,-]+)/available/(?P\d+)$', - AvailableDaysView.as_view(), - name='montpellier-encombrants-available-days', - ), -] - management_urlpatterns = [ - path('add', EncombrantsManagementCreateView.as_view(), name='montpellier-encombrants-add'), - re_path( - r'^(?P[\w,-]+)/edit$', - EncombrantsManagementUpdateView.as_view(), - name='montpellier-encombrants-edit', - ), - re_path( - r'^(?P[\w,-]+)/delete$', - EncombrantsManagementDeleteView.as_view(), - name='montpellier-encombrants-delete', - ), path('sectors/', SectorListView.as_view(), name='montpellier-encombrants-sector-listing'), path('sectors/add', SectorCreateView.as_view(), name='montpellier-encombrants-sector-add'), re_path( @@ -85,14 +61,3 @@ management_urlpatterns = [ name='montpellier-encombrants-collectday-remove', ), ] - -urlpatterns = required( - app_enabled('passerelle_montpellier_encombrants'), - [ - re_path(r'^montpellier-encombrants/', include(public_urlpatterns)), - re_path( - r'^manage/montpellier-encombrants/', - decorated_includes(login_required, include(management_urlpatterns)), - ), - ], -) diff --git a/passerelle_montpellier_encombrants/utils.py b/passerelle_montpellier_encombrants/utils.py index f70e287..1f15d24 100644 --- a/passerelle_montpellier_encombrants/utils.py +++ b/passerelle_montpellier_encombrants/utils.py @@ -23,7 +23,6 @@ from django.core.mail import EmailMessage from django.template.loader import get_template from django.utils.translation import ugettext_lazy as _ -from .models import Commune from .ods import Workbook prefix_map = { @@ -73,6 +72,8 @@ def prefix_cleanup(name): def get_sector(insee, address=None): + from .models import Commune + communes = Commune.objects.filter(insee=insee) for commune in communes: streets = commune.street_set.all() diff --git a/passerelle_montpellier_encombrants/views.py b/passerelle_montpellier_encombrants/views.py index ceb0ab7..65867d9 100644 --- a/passerelle_montpellier_encombrants/views.py +++ b/passerelle_montpellier_encombrants/views.py @@ -36,31 +36,6 @@ from .models import CollectDay, Commune, EncombrantsManagement, Sector, Street from .utils import get_sector, prefix_cleanup -class EncombrantsManagementDetailView(DetailView): - model = EncombrantsManagement - template_name = 'passerelle_montpellier_encombrants/detail.html' - - -class EncombrantsManagementCreateView(CreateView): - model = EncombrantsManagement - form_class = EncombrantsManagementForm - template_name = 'passerelle/manage/service_form.html' - - -class EncombrantsManagementUpdateView(UpdateView): - model = EncombrantsManagement - form_class = EncombrantsManagementUpdateForm - template_name = 'passerelle/manage/service_form.html' - - -class EncombrantsManagementDeleteView(DeleteView): - model = EncombrantsManagement - template_name = 'passerelle/manage/service_confirm_delete.html' - - def get_success_url(self): - return reverse('manage-home') - - class SectorListView(ListView): model = Sector @@ -152,30 +127,6 @@ class CollectDayDeleteView(DeleteView): success_url = reverse_lazy('montpellier-encombrants-collectday-listing') -class AvailableDaysView(View, SingleObjectMixin): - model = EncombrantsManagement - - def get(self, request, *args, **kwargs): - sector = get_sector(kwargs.get('insee'), request.GET.get('adresse')) - if sector: - if datetime.datetime.today().weekday() == 4: - tomorrow = datetime.datetime.today() + datetime.timedelta(days=3) - elif datetime.datetime.today().weekday() == 5: - tomorrow = datetime.datetime.today() + datetime.timedelta(days=2) - else: - tomorrow = datetime.datetime.today() + datetime.timedelta(days=1) - collect_days = sector.collectday_set.filter(date__gt=tomorrow).order_by('date') - else: - collect_days = [] - limit = request.GET.get('limit') - result = [ - {'id': x.date.strftime('%Y-%m-%d'), 'text': x.date.strftime('%d/%m/%Y')} for x in collect_days - ] - if limit: - result = result[: int(limit)] - return passerelle_utils.response_for_json(request, {'data': result}) - - class StreetEditView(FormView): form_class = StreetsForm template_name = 'passerelle/manage/service_form.html' diff --git a/tests/test_main.py b/tests/test_main.py index 857765d..4deeaa1 100644 --- a/tests/test_main.py +++ b/tests/test_main.py @@ -2,10 +2,12 @@ import datetime import json from random import randint +from django.contrib.contenttypes.models import ContentType from django.test import TestCase from django.test.client import Client from django.urls import reverse from django.utils.encoding import force_text +from passerelle.base.models import AccessRight, ApiUser from passerelle_montpellier_encombrants.models import ( CollectDay, @@ -41,6 +43,11 @@ class EncombrantsTestCase(TestCase): self.instance = EncombrantsManagement.objects.create( title='montpellier', slug='montpellier', description='encombrants' ) + api, _ = ApiUser.objects.get_or_create(username='all', keytype='', key='') + obj_type = ContentType.objects.get_for_model(self.instance) + AccessRight.objects.update_or_create( + codename='can_access', apiuser=api, resource_type=obj_type, resource_pk=self.instance.pk + ) self.now = datetime.datetime.now() for mail, insee, name in ( ('cournonsec@montpellier3m.fr', '34087', 'Cournonsec'), @@ -59,14 +66,18 @@ class EncombrantsTestCase(TestCase): for commune in Commune.objects.all(): r = self.client.get( reverse( - 'montpellier-encombrants-available-days', - kwargs={'slug': self.instance.slug, 'insee': commune.insee}, + 'generic-endpoint', + kwargs={ + 'connector': 'passerelle-montpellier-encombrants', + 'endpoint': 'available', + 'slug': self.instance.slug, + 'rest': commune.insee, + }, ) ) data = json.loads(force_text(r.content)) def test_collectdays_in_communes_with_street(self): - maurin_sector = Sector.objects.create(contact_email='maurin@montpellier3m.fr') maurin_commune = Commune.objects.create(sector=maurin_sector, name='Maurin', insee='34970') @@ -81,8 +92,13 @@ class EncombrantsTestCase(TestCase): # Maurin r = self.client.get( reverse( - 'montpellier-encombrants-available-days', - kwargs={'slug': self.instance.slug, 'insee': '34970'}, + 'generic-endpoint', + kwargs={ + 'connector': 'passerelle-montpellier-encombrants', + 'endpoint': 'available', + 'slug': self.instance.slug, + 'rest': '34970', + }, ), {'adresse': 'PLAN DU MAS DE SARDAN'}, ) @@ -95,8 +111,13 @@ class EncombrantsTestCase(TestCase): r = self.client.get( reverse( - 'montpellier-encombrants-available-days', - kwargs={'slug': self.instance.slug, 'insee': '34970'}, + 'generic-endpoint', + kwargs={ + 'connector': 'passerelle-montpellier-encombrants', + 'endpoint': 'available', + 'slug': self.instance.slug, + 'rest': '34970', + }, ), {'adresse': 'ROUTE DEPARTEMENTALE 172'}, ) @@ -107,8 +128,13 @@ class EncombrantsTestCase(TestCase): r = self.client.get( reverse( - 'montpellier-encombrants-available-days', - kwargs={'slug': self.instance.slug, 'insee': '34970'}, + 'generic-endpoint', + kwargs={ + 'connector': 'passerelle-montpellier-encombrants', + 'endpoint': 'available', + 'slug': self.instance.slug, + 'rest': '34970', + }, ), {'adresse': 'ROUTE DEPARTEMENTALE 172'}, ) @@ -118,8 +144,13 @@ class EncombrantsTestCase(TestCase): def test_collectdays_nonexisting_street(self): r = self.client.get( reverse( - 'montpellier-encombrants-available-days', - kwargs={'slug': self.instance.slug, 'insee': '34970'}, + 'generic-endpoint', + kwargs={ + 'connector': 'passerelle-montpellier-encombrants', + 'endpoint': 'available', + 'slug': self.instance.slug, + 'rest': '34970', + }, ), {'adresse': 'Nonexisting street'}, ) @@ -137,8 +168,13 @@ class EncombrantsTestCase(TestCase): r = self.client.get( reverse( - 'montpellier-encombrants-available-days', - kwargs={'slug': self.instance.slug, 'insee': '34970'}, + 'generic-endpoint', + kwargs={ + 'connector': 'passerelle-montpellier-encombrants', + 'endpoint': 'available', + 'slug': self.instance.slug, + 'rest': '34970', + }, ), {'adresse': 'route departementale 172'}, )