diff --git a/aleksis/core/filters.py b/aleksis/core/filters.py index 12265c33c6b9e0d94a9dfb5e87d9111c721fc4ad..aa287690b3d72d7f4e72f1a41708e38401f2b0db 100644 --- a/aleksis/core/filters.py +++ b/aleksis/core/filters.py @@ -1,11 +1,74 @@ -from django_filters import CharFilter, FilterSet +from typing import Sequence + +from django.db.models import Q +from django.utils.translation import gettext as _ + +from django_filters import CharFilter, FilterSet, ModelChoiceFilter, ModelMultipleChoiceFilter from material import Layout, Row +from aleksis.core.models import Group, GroupType, Person, SchoolTerm + + +class MultipleCharFilter(CharFilter): + """Filter for filtering multiple fields with one input. + + >>> multiple_filter = MultipleCharFilter(["name__icontains", "short_name__icontains"]) + """ + + def filter(self, qs, value): # noqa + q = None + for field in self.fields: + if not q: + q = Q(**{field: value}) + else: + q = q | Q(**{field: value}) + return qs.filter(q) + + def __init__(self, fields: Sequence[str], *args, **kwargs): + self.fields = fields + super().__init__(self, *args, **kwargs) + class GroupFilter(FilterSet): - name = CharFilter(lookup_expr="icontains") - short_name = CharFilter(lookup_expr="icontains") + school_term = ModelChoiceFilter(queryset=SchoolTerm.objects.all()) + group_type = ModelChoiceFilter(queryset=GroupType.objects.all()) + parent_groups = ModelMultipleChoiceFilter(queryset=Group.objects.all()) + + search = MultipleCharFilter(["name__icontains", "short_name__icontains"], label=_("Search")) def __init__(self, *args, **kwargs): super().__init__(*args, **kwargs) - self.form.layout = Layout(Row("name", "short_name")) + self.form.layout = Layout(Row("search"), Row("school_term", "group_type", "parent_groups")) + self.form.initial = {"school_term": SchoolTerm.current} + + +class PersonFilter(FilterSet): + name = MultipleCharFilter( + [ + "first_name__icontains", + "additional_name__icontains", + "last_name__icontains", + "short_name__icontains", + ], + label=_("Search by name"), + ) + contact = MultipleCharFilter( + [ + "street__icontains", + "housenumber__icontains", + "postal_code__icontains", + "place__icontains", + "phone_number__icontains", + "mobile_number__icontains", + "email__icontains", + ], + label=_("Search by contact details"), + ) + + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + self.form.layout = Layout(Row("name", "contact"), Row("is_active", "sex", "primary_group")) + + class Meta: + model = Person + fields = ["sex", "is_active", "primary_group"] diff --git a/aleksis/core/forms.py b/aleksis/core/forms.py index 208cbff93f76b7d44451550b73d3b5659ba4aa59..e1c0be7ef4f28e609c93f3b58208aa0ea4d58b3a 100644 --- a/aleksis/core/forms.py +++ b/aleksis/core/forms.py @@ -80,10 +80,7 @@ class EditPersonForm(ExtensibleForm): Fieldset(_("Address"), Row("street", "housenumber"), Row("postal_code", "place")), Fieldset(_("Contact data"), "email", Row("phone_number", "mobile_number")), Fieldset( - _("Advanced personal data"), - Row("sex", "date_of_birth"), - Row("photo", "photo_cropping"), - "guardians", + _("Advanced personal data"), Row("sex", "date_of_birth"), Row("photo"), "guardians", ), ) @@ -106,11 +103,12 @@ class EditPersonForm(ExtensibleForm): "date_of_birth", "sex", "photo", - "photo_cropping", "guardians", "primary_group", ] - widgets = {"user": Select2Widget} + widgets = { + "user": Select2Widget, + } new_user = forms.CharField( required=False, label=_("New user"), help_text=_("Create a new account") diff --git a/aleksis/core/migrations/0003_drop_image_cropping.py b/aleksis/core/migrations/0003_drop_image_cropping.py new file mode 100644 index 0000000000000000000000000000000000000000..1fecda4aeed0c9c0c0075bc8972f648a8ec981a0 --- /dev/null +++ b/aleksis/core/migrations/0003_drop_image_cropping.py @@ -0,0 +1,22 @@ +# Generated by Django 3.0.7 on 2020-06-28 11:37 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('core', '0002_school_term'), + ] + + operations = [ + migrations.RemoveField( + model_name='person', + name='photo_cropping', + ), + migrations.AlterField( + model_name='person', + name='photo', + field=models.ImageField(blank=True, null=True, upload_to='', verbose_name='Photo'), + ), + ] diff --git a/aleksis/core/mixins.py b/aleksis/core/mixins.py index 88a01c15cf28489fde10c788895a35758bf857db..d7108b88b4452c0e25cf7dd0df613d004e82536e 100644 --- a/aleksis/core/mixins.py +++ b/aleksis/core/mixins.py @@ -16,12 +16,12 @@ from django.http import HttpResponse from django.utils.functional import lazy from django.utils.translation import gettext as _ from django.views.generic import CreateView, UpdateView -from django.views.generic.edit import ModelFormMixin +from django.views.generic.edit import DeleteView, ModelFormMixin import reversion from easyaudit.models import CRUDEvent from guardian.admin import GuardedModelAdmin -from jsonstore.fields import JSONField, JSONFieldMixin +from jsonstore.fields import IntegerField, JSONField, JSONFieldMixin from material.base import Layout, LayoutNode from rules.contrib.admin import ObjectPermissionsModelAdmin @@ -103,7 +103,7 @@ class ExtensibleModel(models.Model, metaclass=_ExtensibleModelBase): objects_all_sites = models.Manager() extra_permissions = [] - + def get_absolute_url(self) -> str: """Get the URL o a view representing this model instance.""" pass @@ -209,6 +209,66 @@ class ExtensibleModel(models.Model, metaclass=_ExtensibleModelBase): cls._safe_add(field, name) + @classmethod + def foreign_key( + cls, + field_name: str, + to: models.Model, + to_field: str = "pk", + to_field_type: JSONFieldMixin = IntegerField, + related_name: Optional[str] = None, + ) -> None: + """Add a virtual ForeignKey. + + This works by storing the primary key (or any field passed in the to_field argument) + and adding a property that queries the desired model. + + If the foreign model also is an ExtensibleModel, a reverse mapping is also added under + the related_name passed as argument, or this model's default related name. + """ + + id_field_name = f"{field_name}_id" + if related_name is None: + related_name = cls.Meta.default_related_name + + # Add field to hold key to foreign model + id_field = to_field_type(blank=True, null=True) + cls.field(**{id_field_name: id_field}) + + @property + def _virtual_fk(self) -> Optional[models.Model]: + id_field_val = getattr(self, id_field_name) + if id_field_val: + try: + return to.objects.get(**{to_field: id_field_val}) + except to.DoesNotExist: + # We found a stale foreign key + setattr(self, id_field_name, None) + self.save() + return None + else: + return None + + @_virtual_fk.setter + def _virtual_fk(self, value: Optional[models.Model] = None) -> None: + if value is None: + id_field_val = None + else: + id_field_val = getattr(value, to_field) + setattr(self, id_field_name, id_field_val) + + # Add property to wrap get/set on foreign model instance + cls._safe_add(_virtual_fk, field_name) + + # Add related property on foreign model instance if it provides such an interface + if hasattr(to, "_safe_add"): + + def _virtual_related(self) -> models.QuerySet: + id_field_val = getattr(self, to_field) + return cls.objects.filter(**{id_field_name: id_field_val}) + + to.property_(_virtual_related, related_name) + @classmethod def syncable_fields(cls) -> List[models.Field]: """Collect all fields that can be synced on a model.""" @@ -315,6 +375,24 @@ class AdvancedEditView(UpdateView, SuccessMessageMixin): pass +class AdvancedDeleteView(DeleteView): + """Common confirm view for deleting. + + .. warning :: + + Using this view, objects are deleted permanently after confirming. + We recommend to include the mixin :class:`reversion.views.RevisionMixin` + from `django-reversion` to enable soft-delete. + """ + success_message: Optional[str] = None + + def delete(self, request, *args, **kwargs): + r = super().delete(request, *args, **kwargs) + if self.success_message: + messages.success(self.request, self.success_message) + return r + + class SchoolTermRelatedExtensibleModel(ExtensibleModel): """Add relation to school term.""" diff --git a/aleksis/core/models.py b/aleksis/core/models.py index fa24519076a2cb1951aabe01e973d9f1380c1588..8910d5abeae6f02c6ebd781676002b804faef91b 100644 --- a/aleksis/core/models.py +++ b/aleksis/core/models.py @@ -20,7 +20,6 @@ from django.utils.translation import gettext_lazy as _ import jsonstore from dynamic_preferences.models import PerInstancePreferenceModel -from image_cropping import ImageCropField, ImageRatioField from phonenumber_field.modelfields import PhoneNumberField from polymorphic.models import PolymorphicModel @@ -79,7 +78,7 @@ class SchoolTerm(ExtensibleModel): qs = SchoolTerm.objects.within_dates(self.date_start, self.date_end) if self.pk: - qs.exclude(pk=self.pk) + qs = qs.exclude(pk=self.pk) if qs.exists(): raise ValidationError( _("There is already a school term for this time or a part of this time.") @@ -149,8 +148,7 @@ class Person(ExtensibleModel): date_of_birth = models.DateField(verbose_name=_("Date of birth"), blank=True, null=True) sex = models.CharField(verbose_name=_("Sex"), max_length=1, choices=SEX_CHOICES, blank=True) - photo = ImageCropField(verbose_name=_("Photo"), blank=True, null=True) - photo_cropping = ImageRatioField("photo", "600x800", size_warning=True) + photo = models.ImageField(verbose_name=_("Photo"), blank=True, null=True) guardians = models.ManyToManyField( "self", diff --git a/aleksis/core/settings.py b/aleksis/core/settings.py index ad5d7077eba8bccb0981f70b2ea7fffce7c9474c..e6de512b3be12e211fbc7f91453848f326e17da3 100644 --- a/aleksis/core/settings.py +++ b/aleksis/core/settings.py @@ -70,7 +70,6 @@ INSTALLED_APPS = [ "django_yarnpkg", "django_tables2", "easy_thumbnails", - "image_cropping", "maintenance_mode", "menu_generator", "reversion", @@ -157,12 +156,7 @@ TEMPLATES = [ }, ] -THUMBNAIL_PROCESSORS = ( - "image_cropping.thumbnail_processors.crop_corners", -) + thumbnail_settings.THUMBNAIL_PROCESSORS - -# Already included by base template / Bootstrap -IMAGE_CROPPING_JQUERY_URL = None +THUMBNAIL_PROCESSORS = () + thumbnail_settings.THUMBNAIL_PROCESSORS WSGI_APPLICATION = "aleksis.core.wsgi.application" diff --git a/aleksis/core/templates/core/group/full.html b/aleksis/core/templates/core/group/full.html index ba5bba0aa1208565331d748d106b4d2e982f7918..def54269f884515d3f07f96ee353616df2db3c7e 100644 --- a/aleksis/core/templates/core/group/full.html +++ b/aleksis/core/templates/core/group/full.html @@ -3,7 +3,7 @@ {% extends "core/base.html" %} {% load rules %} -{% load i18n static %} +{% load i18n static material_form %} {% load render_table from django_tables2 %} {% block browser_title %}{{ group.name }}{% endblock %} diff --git a/aleksis/core/templates/core/group/list.html b/aleksis/core/templates/core/group/list.html index fab23516a458f55347ebcc152ee6ffac20e004d5..436b193e74f4c7e7e718fe5ba69a0b9fa0812a2b 100644 --- a/aleksis/core/templates/core/group/list.html +++ b/aleksis/core/templates/core/group/list.html @@ -2,7 +2,7 @@ {% extends "core/base.html" %} -{% load i18n %} +{% load i18n material_form %} {% load render_table from django_tables2 %} {% block browser_title %}{% blocktrans %}Groups{% endblocktrans %}{% endblock %} @@ -14,5 +14,17 @@ {% trans "Create group" %} </a> + <h5>{% trans "Filter groups" %}</h5> + <form method="get"> + {% form form=groups_filter.form %}{% endform %} + {% trans "Search" as caption %} + {% include "core/partials/save_button.html" with caption=caption icon="search" %} + <button type="reset" class="btn red waves-effect waves-light"> + <i class="material-icons left">clear</i> + {% trans "Clear" %} + </button> + </form> + + <h5>{% trans "Selected groups" %}</h5> {% render_table groups_table %} {% endblock %} diff --git a/aleksis/core/templates/core/pages/delete.html b/aleksis/core/templates/core/pages/delete.html new file mode 100644 index 0000000000000000000000000000000000000000..db64ee8fc3b977ed9cb74333e1a25834ce3e58b7 --- /dev/null +++ b/aleksis/core/templates/core/pages/delete.html @@ -0,0 +1,25 @@ +{% extends "core/base.html" %} +{% load data_helpers i18n %} + +{% block browser_title %} + {% verbose_name_object object as object_name %} + {% blocktrans with object_name=object_name %}Delete {{ object_name }}{% endblocktrans %} +{% endblock %} + +{% block content %} + {% verbose_name_object object as object_name %} + + <p class="flow-text"> + {% blocktrans with object_name=object_name object=object %} + Do you really want to delete the {{ object_name }} "{{ object }}"? + {% endblocktrans %} + </p> + + <form method="post" action=""> + {% csrf_token %} + <button type="submit" class="btn red waves-effect waves-light"> + <i class="material-icons left">delete</i> + {% trans "Delete" %} + </button> + </form> +{% endblock %} diff --git a/aleksis/core/templates/core/person/edit.html b/aleksis/core/templates/core/person/edit.html index 8f854610e3424b9da47f142cef41b71c9e0fb097..261249f6867a4370745f66d3bd0abc05891442cf 100644 --- a/aleksis/core/templates/core/person/edit.html +++ b/aleksis/core/templates/core/person/edit.html @@ -4,6 +4,9 @@ {% load material_form i18n %} +{% block extra_head %} + {{ edit_person_form.media }} +{% endblock %} {% block browser_title %}{% blocktrans %}Edit person{% endblocktrans %}{% endblock %} {% block page_title %}{% blocktrans %}Edit person{% endblocktrans %}{% endblock %} diff --git a/aleksis/core/templates/core/person/full.html b/aleksis/core/templates/core/person/full.html index 73d2a4cf663999b2648e497e25559e83df69bb24..4d8bea526492c236d015e7263da05e0bc5dca99e 100644 --- a/aleksis/core/templates/core/person/full.html +++ b/aleksis/core/templates/core/person/full.html @@ -2,7 +2,7 @@ {% extends "core/base.html" %} -{% load i18n static cropping rules %} +{% load i18n static rules material_form %} {% load render_table from django_tables2 %} {% block browser_title %}{{ person.first_name }} {{ person.last_name }}{% endblock %} @@ -44,7 +44,7 @@ <div class="col s12 m4"> {% has_perm 'core.view_photo' user person as can_view_photo %} {% if person.photo and can_view_photo %} - <img class="person-img" src="{% cropped_thumbnail person 'photo_cropping' max_size='300x400' %}" + <img class="person-img" src="{{ person.photo.url }}" alt="{{ person.first_name }} {{ person.last_name }}"/> {% else %} <img class="person-img" src="{% static 'img/fallback.png' %}" diff --git a/aleksis/core/templates/core/person/list.html b/aleksis/core/templates/core/person/list.html index 1103199e002bda1394b3028df4483c69804268d8..b48baf3e5ff8b0f229ab13dfb7c6b5a5885d105f 100644 --- a/aleksis/core/templates/core/person/list.html +++ b/aleksis/core/templates/core/person/list.html @@ -2,7 +2,7 @@ {% extends "core/base.html" %} -{% load i18n rules %} +{% load i18n rules material_form %} {% load render_table from django_tables2 %} {% block browser_title %}{% blocktrans %}Persons{% endblocktrans %}{% endblock %} @@ -18,5 +18,17 @@ </a> {% endif %} + <h5>{% trans "Filter persons" %}</h5> + <form method="get"> + {% form form=persons_filter.form %}{% endform %} + {% trans "Search" as caption %} + {% include "core/partials/save_button.html" with caption=caption icon="search" %} + <button type="reset" class="btn red waves-effect waves-light"> + <i class="material-icons left">clear</i> + {% trans "Clear" %} + </button> + </form> + + <h5>{% trans "Selected persons" %}</h5> {% render_table persons_table %} {% endblock %} diff --git a/aleksis/core/templatetags/data_helpers.py b/aleksis/core/templatetags/data_helpers.py index f7393c73fd86eba6f861f87e321bf8dc5cbce6fd..ab2309f260dd6b039cc722bae86fa71637967a1f 100644 --- a/aleksis/core/templatetags/data_helpers.py +++ b/aleksis/core/templatetags/data_helpers.py @@ -3,6 +3,7 @@ from typing import Any, Optional, Union from django import template from django.contrib.contenttypes.models import ContentType +from django.db.models import Model register = template.Library() @@ -33,6 +34,17 @@ def verbose_name(app_label: str, model: str, field: Optional[str] = None) -> str return ct._meta.verbose_name.title() +@register.simple_tag +def verbose_name_object(model: Model, field: Optional[str] = None) -> str: + """Get a verbose name of a model or a field by a model or an instance of a model.""" + if field: + # Field + return model._meta.get_field(field).verbose_name.title() + else: + # Whole model + return model._meta.verbose_name.title() + + @register.simple_tag def parse_json(value: Optional[str] = None) -> Union[dict, None]: """Template tag for parsing JSON from a string.""" diff --git a/aleksis/core/util/apps.py b/aleksis/core/util/apps.py index 72e0a2fcd0c46c7ca8bae47dd79e79b70ec9d432..79548a79dfbbb94a36a40bfad54fef5a24d3653c 100644 --- a/aleksis/core/util/apps.py +++ b/aleksis/core/util/apps.py @@ -204,7 +204,5 @@ class AppConfig(django.apps.AppConfig): ct = ContentType.objects.get_for_model(model) for perm, verbose_name in model.extra_permissions: Permission.objects.get_or_create( - codename=perm, - content_type=ct, - defaults={"name": verbose_name}, + codename=perm, content_type=ct, defaults={"name": verbose_name}, ) diff --git a/aleksis/core/views.py b/aleksis/core/views.py index 80ddd946c02d7e051f33cbdb5e4f4157c13384c5..968077c9bbe1224972cdd399b72ed118591fe0f0 100644 --- a/aleksis/core/views.py +++ b/aleksis/core/views.py @@ -19,7 +19,7 @@ from haystack.views import SearchView from health_check.views import MainView from rules.contrib.views import PermissionRequiredMixin, permission_required -from .filters import GroupFilter +from .filters import GroupFilter, PersonFilter from .forms import ( AnnouncementForm, ChildGroupsForm, @@ -143,8 +143,12 @@ def persons(request: HttpRequest) -> HttpResponse: request.user, "core.view_person", Person.objects.filter(is_active=True) ) + # Get filter + persons_filter = PersonFilter(request.GET, queryset=persons) + context["persons_filter"] = persons_filter + # Build table - persons_table = PersonsTable(persons) + persons_table = PersonsTable(persons_filter.qs) RequestConfig(request).configure(persons_table) context["persons_table"] = persons_table @@ -210,8 +214,12 @@ def groups(request: HttpRequest) -> HttpResponse: # Get all groups groups = get_objects_for_user(request.user, "core.view_group", Group) + # Get filter + groups_filter = GroupFilter(request.GET, queryset=groups) + context["groups_filter"] = groups_filter + # Build table - groups_table = GroupsTable(groups) + groups_table = GroupsTable(groups_filter.qs) RequestConfig(request).configure(groups_table) context["groups_table"] = groups_table @@ -285,23 +293,21 @@ def edit_person(request: HttpRequest, id_: Optional[int] = None) -> HttpResponse if id_: # Edit form for existing group - edit_person_form = EditGroupForm(request.POST or None, instance=person) + edit_person_form = EditPersonForm( + request.POST or None, request.FILES or None, instance=person + ) else: # Empty form to create a new group if request.user.has_perm("core.create_person"): - edit_person_form = EditPersonForm(request.POST or None) + edit_person_form = EditPersonForm(request.POST or None, request.FILES or None) else: raise PermissionDenied() - if request.method == "POST": if edit_person_form.is_valid(): with reversion.create_revision(): edit_person_form.save(commit=True) messages.success(request, _("The person has been saved.")) - # Redirect to self to ensure post-processed data is displayed - return redirect("edit_person_by_id", id_=person.id) - context["edit_person_form"] = edit_person_form return render(request, "core/person/edit.html", context) diff --git a/poetry.lock b/poetry.lock index 6b6e22ef2ddaa1765d78b32b42c7daf0a89f4e79..86a814e190cd2ab13930c35044e129b0828756c7 100644 --- a/poetry.lock +++ b/poetry.lock @@ -321,7 +321,7 @@ description = "A high-level Python Web framework that encourages rapid developme name = "django" optional = false python-versions = ">=3.6" -version = "3.0.7" +version = "3.0.8" [package.dependencies] asgiref = ">=3.2,<4.0" @@ -488,7 +488,7 @@ description = "Dynamic global and instance settings for your django project" name = "django-dynamic-preferences" optional = false python-versions = "*" -version = "1.9" +version = "1.10" [package.dependencies] django = ">=1.11" @@ -501,7 +501,7 @@ description = "Yet another Django audit log app, hopefully the simplest one." name = "django-easy-audit" optional = false python-versions = "*" -version = "1.2.3a5" +version = "1.2.3" [package.dependencies] beautifulsoup4 = "*" @@ -613,7 +613,7 @@ description = "A Django utility application that returns client's real IP addres name = "django-ipware" optional = false python-versions = "*" -version = "2.1.0" +version = "3.0.0" [[package]] category = "main" @@ -690,7 +690,7 @@ description = "A pluggable framework for adding two-factor authentication to Dja name = "django-otp" optional = false python-versions = "*" -version = "0.9.1" +version = "0.9.3" [package.dependencies] django = ">=1.11" @@ -744,7 +744,7 @@ description = "A Django app to include a manifest.json and Service Worker instan name = "django-pwa" optional = false python-versions = "*" -version = "1.0.9" +version = "1.0.10" [package.dependencies] django = ">=1.8" @@ -1177,7 +1177,7 @@ description = "Internationalized Domain Names in Applications (IDNA)" name = "idna" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" -version = "2.9" +version = "2.10" [[package]] category = "dev" @@ -1208,14 +1208,12 @@ category = "dev" description = "A Python utility / library to sort Python imports." name = "isort" optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" -version = "4.3.21" +python-versions = ">=3.6,<4.0" +version = "5.0.9" [package.extras] -pipfile = ["pipreqs", "requirementslib"] -pyproject = ["toml"] -requirements = ["pipreqs", "pip-api"] -xdg_home = ["appdirs (>=1.4.0)"] +pipfile_deprecated_finder = ["pipreqs", "requirementslib", "tomlkit (>=0.5.3)"] +requirements_deprecated_finder = ["pipreqs", "pip-api"] [[package]] category = "dev" @@ -1393,7 +1391,7 @@ description = "Python Imaging Library (Fork)" name = "pillow" optional = false python-versions = ">=3.5" -version = "7.1.2" +version = "7.2.0" [[package]] category = "dev" @@ -1645,7 +1643,7 @@ description = "Add .env support to your django/flask apps in development and dep name = "python-dotenv" optional = false python-versions = "*" -version = "0.13.0" +version = "0.14.0" [package.extras] cli = ["click (>=5.0)"] @@ -1656,7 +1654,7 @@ description = "Python modules for implementing LDAP clients" name = "python-ldap" optional = true python-versions = ">=2.7,!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*" -version = "3.3.0" +version = "3.3.1" [package.dependencies] pyasn1 = ">=0.3.7" @@ -1843,7 +1841,7 @@ description = "Python documentation generator" name = "sphinx" optional = false python-versions = ">=3.5" -version = "3.1.1" +version = "3.1.2" [package.dependencies] Jinja2 = ">=2.3" @@ -2051,7 +2049,7 @@ description = "Fast, Extensible Progress Meter" name = "tqdm" optional = false python-versions = ">=2.6, !=3.0.*, !=3.1.*" -version = "4.46.1" +version = "4.47.0" [package.extras] dev = ["py-make (>=0.1.0)", "twine", "argopt", "pydoc-markdown"] @@ -2156,7 +2154,7 @@ celery = ["Celery", "django-celery-results", "django-celery-beat", "django-celer ldap = ["django-auth-ldap"] [metadata] -content-hash = "e775ef8cb21bb2621d2088dfbd9cac6e1c165e6ffa50e17c40e8f769e322b1c8" +content-hash = "a3a9d462489de57ed0d52f922959a885c5502329f81283fd5e29eecfef0c8545" python-versions = "^3.7" [metadata.files] @@ -2290,8 +2288,8 @@ dj-database-url = [ {file = "dj_database_url-0.5.0-py2.py3-none-any.whl", hash = "sha256:851785365761ebe4994a921b433062309eb882fedd318e1b0fcecc607ed02da9"}, ] django = [ - {file = "Django-3.0.7-py3-none-any.whl", hash = "sha256:e1630333248c9b3d4e38f02093a26f1e07b271ca896d73097457996e0fae12e8"}, - {file = "Django-3.0.7.tar.gz", hash = "sha256:5052b34b34b3425233c682e0e11d658fd6efd587d11335a0203d827224ada8f2"}, + {file = "Django-3.0.8-py3-none-any.whl", hash = "sha256:5457fc953ec560c5521b41fad9e6734a4668b7ba205832191bbdff40ec61073c"}, + {file = "Django-3.0.8.tar.gz", hash = "sha256:31a5fbbea5fc71c99e288ec0b2f00302a0a92c44b13ede80b73a6a4d6d205582"}, ] django-any-js = [ {file = "django-any-js-1.0.3.post0.tar.gz", hash = "sha256:1da88b44b861b0f54f6b8ea0eb4c7c4fa1a5772e9a4320532cd4e0871a4e23f7"}, @@ -2344,12 +2342,12 @@ django-debug-toolbar = [ {file = "django_debug_toolbar-2.2-py3-none-any.whl", hash = "sha256:ff94725e7aae74b133d0599b9bf89bd4eb8f5d2c964106e61d11750228c8774c"}, ] django-dynamic-preferences = [ - {file = "django-dynamic-preferences-1.9.tar.gz", hash = "sha256:407db27bf55d391c4c8a4944e0521f35eff82c2f2fd5a2fc843fb1b4cc1a31f4"}, - {file = "django_dynamic_preferences-1.9-py2.py3-none-any.whl", hash = "sha256:a3c84696f0459d8d6d9c43374ff3db7daa59b46670b461bb954057d08af607e1"}, + {file = "django-dynamic-preferences-1.10.tar.gz", hash = "sha256:2310291c7f40606be045938d65e117383549aa8a979c6c4b700464c6a6204a34"}, + {file = "django_dynamic_preferences-1.10-py2.py3-none-any.whl", hash = "sha256:d5852c720c1989a67d87669035e11f6c033e7a507de6ec9bd28941cba24a2dc4"}, ] django-easy-audit = [ - {file = "django-easy-audit-1.2.3a5.tar.gz", hash = "sha256:48fc3042760485eb9baae232c4ce21fb96476246e341ee7656a09db3ab3df047"}, - {file = "django_easy_audit-1.2.3a5-py3-none-any.whl", hash = "sha256:32a60f1278f8b34aeda7c6f16a8f04bfab777584a38829e28e468ce0d5a45313"}, + {file = "django-easy-audit-1.2.3.tar.gz", hash = "sha256:9e0baae1cc06a9b7766bc6743695ff5e199129577649ce8f6e7c7c8904943a30"}, + {file = "django_easy_audit-1.2.3-py3-none-any.whl", hash = "sha256:425d4e9c03a48916e309675d520639ff9ce9c5c4d561eabd595b2b42f1a97a89"}, ] django-favicon-plus-reloaded = [ {file = "django-favicon-plus-reloaded-1.0.4.tar.gz", hash = "sha256:90c761c636a338e6e9fb1d086649d82095085f92cff816c9cf074607f28c85a5"}, @@ -2387,7 +2385,7 @@ django-impersonate = [ {file = "django-impersonate-1.5.1.tar.gz", hash = "sha256:7c786ffaa7a5dd430f9277b53a64676c470b684eee5aa52c3b483298860d09b4"}, ] django-ipware = [ - {file = "django-ipware-2.1.0.tar.gz", hash = "sha256:a7c7a8fd019dbdc9c357e6e582f65034e897572fc79a7e467674efa8aef9d00b"}, + {file = "django-ipware-3.0.0.tar.gz", hash = "sha256:161605eb011439550dd3ee496d0e999720b13f01952be25ea9e88982fbe48e83"}, ] django-js-asset = [ {file = "django-js-asset-1.2.2.tar.gz", hash = "sha256:c163ae80d2e0b22d8fb598047cd0dcef31f81830e127cfecae278ad574167260"}, @@ -2415,8 +2413,8 @@ django-middleware-global-request = [ {file = "django-middleware-global-request-0.1.2.tar.gz", hash = "sha256:f6490759bc9f7dbde4001709554e29ca715daf847f2222914b4e47117dca9313"}, ] django-otp = [ - {file = "django-otp-0.9.1.tar.gz", hash = "sha256:f456639addace8b6d1eb77f9edaada1a53dbb4d6f3c19f17c476c4e3e4beb73f"}, - {file = "django_otp-0.9.1-py3-none-any.whl", hash = "sha256:0c67cf6f4bd6fca84027879ace9049309213b6ac81f88e954376a6b5535d96c4"}, + {file = "django-otp-0.9.3.tar.gz", hash = "sha256:d2390e61794bc10dea2fd949cbcfb7946e9ae4fb248df5494ccc4ef9ac50427e"}, + {file = "django_otp-0.9.3-py3-none-any.whl", hash = "sha256:97849f7bf1b50c4c36a5845ab4d2e11dd472fa8e6bcc34fe18b6d3af6e4aa449"}, ] django-otp-yubikey = [ {file = "django-otp-yubikey-0.5.2.tar.gz", hash = "sha256:f0b1881562fb42ee9f12c28d284cbdb90d1f0383f2d53a595373b080a19bc261"}, @@ -2431,8 +2429,8 @@ django-polymorphic = [ {file = "django_polymorphic-2.1.2-py2.py3-none-any.whl", hash = "sha256:0a25058e95e5e99fe0beeabb8f4734effe242d7b5b77dca416fba9fd3062da6a"}, ] django-pwa = [ - {file = "django-pwa-1.0.9.tar.gz", hash = "sha256:c11bcb40bbbb65f9037e4ae4d7233e6fa724c4410419b257cce4b6624a9542e9"}, - {file = "django_pwa-1.0.9-py3-none-any.whl", hash = "sha256:8706cbd84489fb63d3523d5037d2cdfd8ff177417292bd7845b0f177d3c4ed3f"}, + {file = "django-pwa-1.0.10.tar.gz", hash = "sha256:07ed9dd57108838e3fe44b551a82032ca4ed76e31cb3c3e8d51604e0fe7e81e9"}, + {file = "django_pwa-1.0.10-py3-none-any.whl", hash = "sha256:b1a2057b1e72c40c3a14beb90b958482da185f1d40a141fcae3d76580984b930"}, ] django-render-block = [ {file = "django_render_block-0.6-py2.py3-none-any.whl", hash = "sha256:95c7dc9610378a10e0c4a10d8364ec7307210889afccd6a67a6aaa0fd599bd4d"}, @@ -2553,8 +2551,8 @@ html2text = [ {file = "html2text-2020.1.16.tar.gz", hash = "sha256:e296318e16b059ddb97f7a8a1d6a5c1d7af4544049a01e261731d2d5cc277bbb"}, ] idna = [ - {file = "idna-2.9-py2.py3-none-any.whl", hash = "sha256:a068a21ceac8a4d63dbfd964670474107f541babbd2250d61922f029858365fa"}, - {file = "idna-2.9.tar.gz", hash = "sha256:7588d1c14ae4c77d74036e8c22ff447b26d0fde8f007354fd48a7814db15b7cb"}, + {file = "idna-2.10-py2.py3-none-any.whl", hash = "sha256:b97d804b1e9b523befed77c48dacec60e6dcb0b5391d57af6a65a312a90648c0"}, + {file = "idna-2.10.tar.gz", hash = "sha256:b307872f855b18632ce0c21c5e45be78c0ea7ae4c15c828c20788b26921eb3f6"}, ] imagesize = [ {file = "imagesize-1.2.0-py2.py3-none-any.whl", hash = "sha256:6965f19a6a2039c7d48bca7dba2473069ff854c36ae6f19d2cde309d998228a1"}, @@ -2565,8 +2563,8 @@ importlib-metadata = [ {file = "importlib_metadata-1.7.0.tar.gz", hash = "sha256:90bb658cdbbf6d1735b6341ce708fc7024a3e14e99ffdc5783edea9f9b077f83"}, ] isort = [ - {file = "isort-4.3.21-py2.py3-none-any.whl", hash = "sha256:6e811fcb295968434526407adb8796944f1988c5b65e8139058f2014cbe100fd"}, - {file = "isort-4.3.21.tar.gz", hash = "sha256:54da7e92468955c4fceacd0c86bd0ec997b0e1ee80d97f67c35a78b719dccab1"}, + {file = "isort-5.0.9-py3-none-any.whl", hash = "sha256:3d8e0d7678b66af1e3a6936f2a380a5d4f3faf4b1b38c6aaa4bed6695c6bdcde"}, + {file = "isort-5.0.9.tar.gz", hash = "sha256:639b8084644ceb13a806f42d690273b9d844793ac2f515fbc575ba65dc044de0"}, ] jinja2 = [ {file = "Jinja2-2.11.2-py2.py3-none-any.whl", hash = "sha256:f0a4641d3cf955324a89c04f3d94663aa4d638abe8f733ecd3582848e1c37035"}, @@ -2682,29 +2680,32 @@ phonenumbers = [ {file = "phonenumbers-8.12.6.tar.gz", hash = "sha256:d332078fe71c6153b5a263ac87283618b2afe514a248a14f06a0d39ce1f5ce0b"}, ] pillow = [ - {file = "Pillow-7.1.2-cp35-cp35m-macosx_10_10_intel.whl", hash = "sha256:ae2b270f9a0b8822b98655cb3a59cdb1bd54a34807c6c56b76dd2e786c3b7db3"}, - {file = "Pillow-7.1.2-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:d23e2aa9b969cf9c26edfb4b56307792b8b374202810bd949effd1c6e11ebd6d"}, - {file = "Pillow-7.1.2-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:b532bcc2f008e96fd9241177ec580829dee817b090532f43e54074ecffdcd97f"}, - {file = "Pillow-7.1.2-cp35-cp35m-win32.whl", hash = "sha256:12e4bad6bddd8546a2f9771485c7e3d2b546b458ae8ff79621214119ac244523"}, - {file = "Pillow-7.1.2-cp35-cp35m-win_amd64.whl", hash = "sha256:9744350687459234867cbebfe9df8f35ef9e1538f3e729adbd8fde0761adb705"}, - {file = "Pillow-7.1.2-cp36-cp36m-macosx_10_10_x86_64.whl", hash = "sha256:f54be399340aa602066adb63a86a6a5d4f395adfdd9da2b9a0162ea808c7b276"}, - {file = "Pillow-7.1.2-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:1f694e28c169655c50bb89a3fa07f3b854d71eb47f50783621de813979ba87f3"}, - {file = "Pillow-7.1.2-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:f784aad988f12c80aacfa5b381ec21fd3f38f851720f652b9f33facc5101cf4d"}, - {file = "Pillow-7.1.2-cp36-cp36m-win32.whl", hash = "sha256:b37bb3bd35edf53125b0ff257822afa6962649995cbdfde2791ddb62b239f891"}, - {file = "Pillow-7.1.2-cp36-cp36m-win_amd64.whl", hash = "sha256:b67a6c47ed963c709ed24566daa3f95a18f07d3831334da570c71da53d97d088"}, - {file = "Pillow-7.1.2-cp37-cp37m-macosx_10_10_x86_64.whl", hash = "sha256:eaa83729eab9c60884f362ada982d3a06beaa6cc8b084cf9f76cae7739481dfa"}, - {file = "Pillow-7.1.2-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:f46e0e024346e1474083c729d50de909974237c72daca05393ee32389dabe457"}, - {file = "Pillow-7.1.2-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:0e2a3bceb0fd4e0cb17192ae506d5f082b309ffe5fc370a5667959c9b2f85fa3"}, - {file = "Pillow-7.1.2-cp37-cp37m-win32.whl", hash = "sha256:ccc9ad2460eb5bee5642eaf75a0438d7f8887d484490d5117b98edd7f33118b7"}, - {file = "Pillow-7.1.2-cp37-cp37m-win_amd64.whl", hash = "sha256:b943e71c2065ade6fef223358e56c167fc6ce31c50bc7a02dd5c17ee4338e8ac"}, - {file = "Pillow-7.1.2-cp38-cp38-macosx_10_10_x86_64.whl", hash = "sha256:04766c4930c174b46fd72d450674612ab44cca977ebbcc2dde722c6933290107"}, - {file = "Pillow-7.1.2-cp38-cp38-manylinux1_i686.whl", hash = "sha256:f455efb7a98557412dc6f8e463c1faf1f1911ec2432059fa3e582b6000fc90e2"}, - {file = "Pillow-7.1.2-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:ee94fce8d003ac9fd206496f2707efe9eadcb278d94c271f129ab36aa7181344"}, - {file = "Pillow-7.1.2-cp38-cp38-win32.whl", hash = "sha256:4b02b9c27fad2054932e89f39703646d0c543f21d3cc5b8e05434215121c28cd"}, - {file = "Pillow-7.1.2-cp38-cp38-win_amd64.whl", hash = "sha256:3d25dd8d688f7318dca6d8cd4f962a360ee40346c15893ae3b95c061cdbc4079"}, - {file = "Pillow-7.1.2-pp373-pypy36_pp73-win32.whl", hash = "sha256:0f01e63c34f0e1e2580cc0b24e86a5ccbbfa8830909a52ee17624c4193224cd9"}, - {file = "Pillow-7.1.2-py3.8-macosx-10.9-x86_64.egg", hash = "sha256:70e3e0d99a0dcda66283a185f80697a9b08806963c6149c8e6c5f452b2aa59c0"}, - {file = "Pillow-7.1.2.tar.gz", hash = "sha256:a0b49960110bc6ff5fead46013bcb8825d101026d466f3a4de3476defe0fb0dd"}, + {file = "Pillow-7.2.0-cp35-cp35m-macosx_10_10_intel.whl", hash = "sha256:1ca594126d3c4def54babee699c055a913efb01e106c309fa6b04405d474d5ae"}, + {file = "Pillow-7.2.0-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:c92302a33138409e8f1ad16731568c55c9053eee71bb05b6b744067e1b62380f"}, + {file = "Pillow-7.2.0-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:8dad18b69f710bf3a001d2bf3afab7c432785d94fcf819c16b5207b1cfd17d38"}, + {file = "Pillow-7.2.0-cp35-cp35m-manylinux2014_aarch64.whl", hash = "sha256:431b15cffbf949e89df2f7b48528be18b78bfa5177cb3036284a5508159492b5"}, + {file = "Pillow-7.2.0-cp35-cp35m-win32.whl", hash = "sha256:09d7f9e64289cb40c2c8d7ad674b2ed6105f55dc3b09aa8e4918e20a0311e7ad"}, + {file = "Pillow-7.2.0-cp35-cp35m-win_amd64.whl", hash = "sha256:0295442429645fa16d05bd567ef5cff178482439c9aad0411d3f0ce9b88b3a6f"}, + {file = "Pillow-7.2.0-cp36-cp36m-macosx_10_10_x86_64.whl", hash = "sha256:ec29604081f10f16a7aea809ad42e27764188fc258b02259a03a8ff7ded3808d"}, + {file = "Pillow-7.2.0-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:612cfda94e9c8346f239bf1a4b082fdd5c8143cf82d685ba2dba76e7adeeb233"}, + {file = "Pillow-7.2.0-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:0a80dd307a5d8440b0a08bd7b81617e04d870e40a3e46a32d9c246e54705e86f"}, + {file = "Pillow-7.2.0-cp36-cp36m-manylinux2014_aarch64.whl", hash = "sha256:06aba4169e78c439d528fdeb34762c3b61a70813527a2c57f0540541e9f433a8"}, + {file = "Pillow-7.2.0-cp36-cp36m-win32.whl", hash = "sha256:f7e30c27477dffc3e85c2463b3e649f751789e0f6c8456099eea7ddd53be4a8a"}, + {file = "Pillow-7.2.0-cp36-cp36m-win_amd64.whl", hash = "sha256:ffe538682dc19cc542ae7c3e504fdf54ca7f86fb8a135e59dd6bc8627eae6cce"}, + {file = "Pillow-7.2.0-cp37-cp37m-macosx_10_10_x86_64.whl", hash = "sha256:94cf49723928eb6070a892cb39d6c156f7b5a2db4e8971cb958f7b6b104fb4c4"}, + {file = "Pillow-7.2.0-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:6edb5446f44d901e8683ffb25ebdfc26988ee813da3bf91e12252b57ac163727"}, + {file = "Pillow-7.2.0-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:52125833b070791fcb5710fabc640fc1df07d087fc0c0f02d3661f76c23c5b8b"}, + {file = "Pillow-7.2.0-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:9ad7f865eebde135d526bb3163d0b23ffff365cf87e767c649550964ad72785d"}, + {file = "Pillow-7.2.0-cp37-cp37m-win32.whl", hash = "sha256:c79f9c5fb846285f943aafeafda3358992d64f0ef58566e23484132ecd8d7d63"}, + {file = "Pillow-7.2.0-cp37-cp37m-win_amd64.whl", hash = "sha256:d350f0f2c2421e65fbc62690f26b59b0bcda1b614beb318c81e38647e0f673a1"}, + {file = "Pillow-7.2.0-cp38-cp38-macosx_10_10_x86_64.whl", hash = "sha256:6d7741e65835716ceea0fd13a7d0192961212fd59e741a46bbed7a473c634ed6"}, + {file = "Pillow-7.2.0-cp38-cp38-manylinux1_i686.whl", hash = "sha256:edf31f1150778abd4322444c393ab9c7bd2af271dd4dafb4208fb613b1f3cdc9"}, + {file = "Pillow-7.2.0-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:d08b23fdb388c0715990cbc06866db554e1822c4bdcf6d4166cf30ac82df8c41"}, + {file = "Pillow-7.2.0-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:5e51ee2b8114def244384eda1c82b10e307ad9778dac5c83fb0943775a653cd8"}, + {file = "Pillow-7.2.0-cp38-cp38-win32.whl", hash = "sha256:725aa6cfc66ce2857d585f06e9519a1cc0ef6d13f186ff3447ab6dff0a09bc7f"}, + {file = "Pillow-7.2.0-cp38-cp38-win_amd64.whl", hash = "sha256:a060cf8aa332052df2158e5a119303965be92c3da6f2d93b6878f0ebca80b2f6"}, + {file = "Pillow-7.2.0-pp36-pypy36_pp73-win32.whl", hash = "sha256:25930fadde8019f374400f7986e8404c8b781ce519da27792cbe46eabec00c4d"}, + {file = "Pillow-7.2.0.tar.gz", hash = "sha256:97f9e7953a77d5a70f49b9a48da7776dc51e9b738151b22dacf101641594a626"}, ] pluggy = [ {file = "pluggy-0.13.1-py2.py3-none-any.whl", hash = "sha256:966c145cd83c96502c3c3868f50408687b38434af77734af1e9ca461a4081d2d"}, @@ -2859,11 +2860,11 @@ python-dateutil = [ {file = "python_dateutil-2.8.1-py2.py3-none-any.whl", hash = "sha256:75bb3f31ea686f1197762692a9ee6a7550b59fc6ca3a1f4b5d7e32fb98e2da2a"}, ] python-dotenv = [ - {file = "python-dotenv-0.13.0.tar.gz", hash = "sha256:3b9909bc96b0edc6b01586e1eed05e71174ef4e04c71da5786370cebea53ad74"}, - {file = "python_dotenv-0.13.0-py2.py3-none-any.whl", hash = "sha256:25c0ff1a3e12f4bde8d592cc254ab075cfe734fc5dd989036716fd17ee7e5ec7"}, + {file = "python-dotenv-0.14.0.tar.gz", hash = "sha256:8c10c99a1b25d9a68058a1ad6f90381a62ba68230ca93966882a4dbc3bc9c33d"}, + {file = "python_dotenv-0.14.0-py2.py3-none-any.whl", hash = "sha256:c10863aee750ad720f4f43436565e4c1698798d763b63234fb5021b6c616e423"}, ] python-ldap = [ - {file = "python-ldap-3.3.0.tar.gz", hash = "sha256:de04939485b53ee5d9a6855562d415b73060c52e681644386de4d5bd18e3f540"}, + {file = "python-ldap-3.3.1.tar.gz", hash = "sha256:4711cacf013e298754abd70058ccc995758177fb425f1c2d30e71adfc1d00aa5"}, ] python-memcached = [ {file = "python-memcached-1.59.tar.gz", hash = "sha256:a2e28637be13ee0bf1a8b6843e7490f9456fd3f2a4cb60471733c7b5d5557e4f"}, @@ -2960,8 +2961,8 @@ spdx-license-list = [ {file = "spdx_license_list-0.5.0.tar.gz", hash = "sha256:40cd53ff16401bab7059e6d1ef61839196b12079929a2763a50145d3b6949bc1"}, ] sphinx = [ - {file = "Sphinx-3.1.1-py3-none-any.whl", hash = "sha256:97c9e3bcce2f61d9f5edf131299ee9d1219630598d9f9a8791459a4d9e815be5"}, - {file = "Sphinx-3.1.1.tar.gz", hash = "sha256:74fbead182a611ce1444f50218a1c5fc70b6cc547f64948f5182fb30a2a20258"}, + {file = "Sphinx-3.1.2-py3-none-any.whl", hash = "sha256:97dbf2e31fc5684bb805104b8ad34434ed70e6c588f6896991b2fdfd2bef8c00"}, + {file = "Sphinx-3.1.2.tar.gz", hash = "sha256:b9daeb9b39aa1ffefc2809b43604109825300300b987a24f45976c001ba1a8fd"}, ] sphinx-autodoc-typehints = [ {file = "sphinx-autodoc-typehints-1.11.0.tar.gz", hash = "sha256:bbf0b203f1019b0f9843ee8eef0cff856dc04b341f6dbe1113e37f2ebf243e11"}, @@ -3027,8 +3028,8 @@ toml = [ {file = "toml-0.10.1.tar.gz", hash = "sha256:926b612be1e5ce0634a2ca03470f95169cf16f939018233a670519cb4ac58b0f"}, ] tqdm = [ - {file = "tqdm-4.46.1-py2.py3-none-any.whl", hash = "sha256:07c06493f1403c1380b630ae3dcbe5ae62abcf369a93bbc052502279f189ab8c"}, - {file = "tqdm-4.46.1.tar.gz", hash = "sha256:cd140979c2bebd2311dfb14781d8f19bd5a9debb92dcab9f6ef899c987fcf71f"}, + {file = "tqdm-4.47.0-py2.py3-none-any.whl", hash = "sha256:7810e627bcf9d983a99d9ff8a0c09674400fd2927eddabeadf153c14a2ec8656"}, + {file = "tqdm-4.47.0.tar.gz", hash = "sha256:63ef7a6d3eb39f80d6b36e4867566b3d8e5f1fe3d6cb50c5e9ede2b3198ba7b7"}, ] twilio = [ {file = "twilio-6.43.0.tar.gz", hash = "sha256:1ff3b66992ebb59411794f669eab7f11bcfaacc5549eec1afb47af1c755872ac"}, diff --git a/pyproject.toml b/pyproject.toml index 07cacfc62d4b4cfd12475b83930725d11a21e6ee..f040f012a853acc82ad7f9ad53e058b7ef31edc2 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -45,7 +45,7 @@ dynaconf = {version = "^2.0", extras = ["yaml", "toml", "ini"]} django-settings-context-processor = "^0.2" django-auth-ldap = { version = "^2.2", optional = true } django-maintenance-mode = "^0.14.0" -django-ipware = "^2.1" +django-ipware = "^3.0" easy-thumbnails = "^2.6" django-image-cropping = "^1.2" django-impersonate = "^1.4" @@ -72,7 +72,7 @@ django-celery-beat = {version="^2.0.0", optional=true} django-celery-email = {version="^3.0.0", optional=true} django-jsonstore = "^0.4.1" django-polymorphic = "^2.1.2" -django-otp = "0.9.1" +django-otp = "0.9.3" django-colorfield = "^0.3.0" django-bleach = "^0.6.1" django-guardian = "^2.2.0" @@ -113,7 +113,7 @@ flake8-docstrings = "^1.5.0" flake8-rst-docstrings = "^0.0.13" black = "^19.10b0" flake8-black = "^0.2.0" -isort = "^4.3.21" +isort = "^5.0.0" flake8-isort = "^3.0.0" pytest-cov = "^2.8.1" pytest-sugar = "^0.9.2"