diff --git a/.gitignore b/.gitignore
index a30a3fa095b66e018388937f3ffb79eed4692301..767581f76153ad5ac9f16422ee895a14b7a67f32 100644
--- a/.gitignore
+++ b/.gitignore
@@ -65,7 +65,6 @@ docs/_build/
 aleksis/node_modules/
 aleksis/static/
 aleksis/whoosh_index/
-aleksis/xapian_index/
 
 .coverage
 .mypy_cache/
diff --git a/CHANGELOG.rst b/CHANGELOG.rst
index 5e3d62bcac1140ac60e3ce5a841c487a7912b8f4..be6b8c58c87807f1b143ccfc683aeaefb2a03dfa 100644
--- a/CHANGELOG.rst
+++ b/CHANGELOG.rst
@@ -1,126 +1,447 @@
 Changelog
 =========
 
-`2.0a2`_
---------
-
-New features
-~~~~~~~~~~~~
-
-* Frontend-ased announcement management
-* Auto-create Person on User creation
-* Select primary group by pattern if unset
-* Shortcut to personal information page
-* Support for defining group types
-* Add description to Person
-* age_at method and age property to Person
-* Synchronise AlekSIS groups with Django groups
-* Add celery worker, celery-beat worker and celery broker to docker-compose setup
-* Global search
-* License information page
-* Roles and permissions
-* User preferences
-* Additional fields for people per group
-* Support global permission flags by LDAP group
-* Persistent announcements
-* Custom menu entries (e.g. in footer)
-* New logo for AlekSIS
-* Two factor authentication with Yubikey, OTP or SMS
-* Devs: Add ExtensibleModel to allow apps to add fields, properties
-* Devs: Support multiple recipient object for one announcement
-
-Minor changes
-~~~~~~~~~~~~~
-
-* Make short_name for group optional
-* Generalised live loading of widgets for dashboard
-* Devs: Add some CSS helper classes for colours
-* Devs: Mandate use of AlekSIS base model
-* Devs: Drop import_ref field(s); apps shold now define their own reference fields
-
-Bug fixes
-~~~~~~~~~
-
-* DateTimeField Announcement.valid_from received a naive datetime
-* Enable SASS processor in production
-* Fix too short fields
-* Load select2 locally
-
-`2.0a1`_
---------
-
-New features
-~~~~~~~~~~~~
-
-* Migrate to materialize.css
-* Dashboard
-* Notifications via SMS (Twilio), Email or on the dashboard
-* Admin interface
-* Turn into installable, progressive web app
-* Devs: Background Tasks with Celery
-
-Minor changes
-~~~~~~~~~~~~~
-
-* Customisable save_button template
-* Redesign error pages
-
-Bug fixes
-~~~~~~~~~
-
-* setup_data no longer forces database connection
-
-`1.0a4`_
---------
-
-New features
-~~~~~~~~~~~~
+All notable changes to this project will be documented in this file.
+
+The format is based on `Keep a Changelog`_,
+and this project adheres to `Semantic Versioning`_.
+
+Unreleased
+----------
+
+`2.1`_ - 2021-11-05
+-------------------
+
+Added
+~~~~~
+
+* Provide an ``ExtensiblePolymorphicModel`` to support the features of extensible models for polymorphic models and vice-versa.
+* Implement optional Sentry integration for error and performance tracing.
+* Option to limit allowed scopes per application, including mixin to enforce that limit on OAuth resource views
+* Support trusted OAuth applications that leave out the authorisation screen.
+* Add birthplace to Person model.
+
+Changed
+~~~~~~~
+
+* Replace dev.sh helper script with tox environments.
+* OAuth Grant Flows are now configured system-wide instead of per app.
+* Refactor OAuth2 application management views.
+
+Fixed
+~~~~~
+
+* Fix default admin contacts
+
+Credits
+~~~~~~~
+
+* We welcome new contributor 🐧 Jonathan Krüger!
+* We welcome new contributor 🐭 Lukas Weichelt!
+
+`2.0`_ - 2021-10-29
+-------------------
+
+Changed
+~~~~~~~
+
+* Refactor views/forms for creating/editing persons.
+
+Fixed
+~~~~~
+
+* Fix order of submit buttons in login form and restructure login template
+  to make 2FA work correctly.
+* Fix page title bug on the impersonate page.
+* Users were able to edit the linked user if self-editing was activated.
+* Users weren't able to edit the allowed fields although they were configured correctly.
+* Provide `style.css` and icon files without any authentication to avoid caching issues.
+
+
+Removed
+~~~~~~~
+
+* Remove mass linking of persons to accounts, bevcause the view had performance issues,
+  but was practically unused.
+
+`2.0rc7`_ - 2021-10-18
+----------------------
+
+Fixed
+~~~~~
+
+* Configuration mechanisms for OpenID Connect were broken.
+* Set a fixed version for django-sass-processor to avoid a bug with searching ``style.css`` in the wrong storage.
+* Correct the z-index of the navbar to display the main title again on mobile devices.
+
+Removed
+~~~~~~~
+
+* Leftovers from a functionality already dropped in the development process
+  (custom authentication backends and alternative login views).
+
+`2.0rc6`_ - 2021-10-11
+----------------------
+
+Added
+~~~~~
+
+* OpenID Connect scope and accompanying claim ``groups``
+* Support config files in JSON format
+* Allow apps to dynamically generate OAuth scopes
+
+Changed
+~~~~~~~
+
+* Do not log or e-mail ALLOWED_HOSTS violations
+* Update translations.
+* Use initial superuser settings as default contact and from addresses
+
+Fixed
+~~~~~
+
+* Show link to imprint in footer
+* Fix API for adding OAuth scopes in AppConfigs
+* Deleting persons is possible again.
+* Removed wrong changelog section
+
+Removed
+~~~~~~~
+
+* Dropped data anonymization (django-hattori) support for now
+* ``OAUTH2_SCOPES`` setting in apps is not supported anymore. Use ``get_all_scopes`` method
+  on ``AppConfig`` class instead.
+
+`2.0rc5`_ - 2021-08-25
+----------------------
+
+Fixed
+~~~~~
+
+* The view for getting the progress of celery tasks didn't respect that there can be anonymous users.
+* Updated django to latest 3.2.x
+
+
+`2.0rc4`_ - 2021-08-01
+----------------------
+
+Added
+~~~~~
+
+* Allow to configure port for prometheus metrics endpoint.
+
+Fixed
+~~~~~
+
+* Correctly deliver server errors to user
+* Use text HTTP response for serviceworker.js insteas of binary stream
+* Use Django permission instead of rule to prevent performance issues.
+
+`2.0rc3`_ - 2021-07-26
+----------------------
+
+Added
+~~~~~
+
+* Support PDF generation without available request object (started completely from background).
+* Display a loading animation while fetching search results in the sidebar.
+
+Fixed
+~~~~~
+
+* Make search suggestions selectable using the arrow keys.
+
+Fixed
+~~~~~
+
+* Use correct HTML 5 elements for the search frontend and fix CSS accordingly.
+
+`2.0rc2`_ - 2021-06-24
+---------------------
+
+Added
+~~~~~
+
+* Allow to install system and build dependencies in docker build
+
+
+`2.0rc1`_ - 2021-06-23
+----------------------
+
+Added
+~~~~~
+
+* Add option to disable dashboard auto updating as a user and sitewide.
+
+Changed
+~~~~~~~
+
+* Use semantically correct html elements for headings and alerts.
+
+Fixed
+~~~~~
+
+* Add missing dependency python-gnupg
+* Add missing AWS options to ignore invalid ssl certificates
+
+`2.0b2`_ - 2021-06-15
+--------------------
+
+Added
+~~~~~~~
+
+* Add option to disable dashboard auto updating as a user and sitewide.
+
+Changed
+~~~~~~~
+
+* Add verbose names for all preference sections.
+* Add verbose names for all openid connect scopes and show them in grant
+  view.
+* Include public dashboard in navigation
+* Update German translations.
+
+Fixed
+~~~~~
+
+* Fix broken backup health check
+* Make error recovery in about page work
+
+Removed
+~~~~~~~
+
+* Drop all leftovers of DataTables.
+
+`2.0b1`_ - 2021-06-01
+---------------------
+
+Changed
+~~~~~~~
+
+* Rename every occurance of "social account" by "third-party account".
+* Use own templates and views for PWA meta and manifest.
+* Use term "application" for all authorized OAuth2 applications/tokens.
+* Use importlib instead of pkg_resources (no functional changes)
+
+Fixed
+~~~~~
+
+* Fix installation documentation (nginx, uWSGI).
+* Use a set for data checks registry to prevent double entries.
+* Progress page tried to redirect even if the URL is empty.
+
+Removed
+~~~~~~~
+
+* Drop django-pwa completely.
+
+`2.0b0`_ - 2021-05-21
+---------------------
+
+Added
+~~~~~
+
+* Allow defining several search configs for LDAP users and groups
+* Use setuptools entrypoints to find apps
+* Add django-cachalot as query cache
+* Add ``syncable_fields`` property to ``ExtensibleModel`` to discover fields
+  sync backends can write to
+* Add ``aleksis-admin`` script to wrap django-admin with pre-configured settings
+* Auto-create persons for users if matching attributes are found
+* Add ``django-allauth`` to allow authentication using OAuth, user registration,
+  password changes and password reset
+* Add OAuth2 and OpenID Connect provider support
+* Add ``django-uwsgi`` to use uWSGI and Celery in development
+* Add loading page for displaying Celery task progress
+* Implement generic PDF generation using Chromium
+* Support Amazon S3 storage for /media files
+* Enable Django REST framework for apps to use at own discretion
+* Add method to inject permissions to ExtensibleModels dynamically
+* Add helper function which filters queryset by permission and user
+* Add generic support for Select 2 with materialize theme
+* Add simple message that is shown whenever a page is served from the PWA cache
+* Add possibility to upload files using ckeditor
+* Show guardians and children on person full page
+* Manage object-level permissions in frontend
+* Add a generic deletion confirmation view
+* Serve Prometheus metrics from app
+* Provide system health check endpoint and checks for some components
+* Add impersonate button to person view
+* Implement a data check system for sanity checks and guided resolution of inconsistencies
+* Make the dashboard configurable for users and as default dashboard by admins
+* Support dynamic badges in menu items
+* Auto-delete old /media files when related model instance is deleted
+* Add SortableJS
+* Add a widget for links/buttons to other websites
+
+Changed
+~~~~~~~
+
+* Make Redis non-optional (see documentation)
+* Use Redis as caching and session store to allow horizontal scaling
+* Enable PostgreSQL connection pooling
+* Use uWSGI to serve /static under development
+* Use a token-secured storage as default /media storage
+* Rewrite Docker image to serve as generic base image for AlekSIS distributions
+* Make Docker image run completely read-only
+* Ensure Docker image is compatible with K8s
+* Remove legacy file upload functoin; all code is required to use the storage API
+* Default search index backend is now Whoosh with Redis storage
+* Re-style search result page
+* Move notifications to separate page with indicator in menu
+* Move to ``BigAutoField`` for all AlekSIS apps
+* Require Django 3.2 and Python 3.9
+* Person and group lists can now be filtered
+* Allow displaying the default widget to anonymous users
+
+Fixed
+~~~~~
+
+* Correct behavious of celery-beat in development
+* Fix precaching of offline fallback page
+* Use correct styling for language selector
+* Rewrite notification e-mail template for AlekSIS
+* Global search now obeys permissions correctly
+* Improve performance of favicon generation
+* Dashboard widgets now handle exceptions gracefully
+* Roboto font was not available for serving locally
+
+Removed
+~~~~~~~
+
+* Dropped support for other search backends than Whoosh
+* Drop django-middleware-global-request completely
+
+`2.0a2`_ - 2020-05-04
+---------------------
+
+Added
+~~~~~
+
+* Frontend-ased announcement management.
+* Auto-create Person on User creation.
+* Select primary group by pattern if unset.
+* Shortcut to personal information page.
+* Support for defining group types.
+* Add description to Person.
+* age_at method and age property to Person.
+* Synchronise AlekSIS groups with Django groups.
+* Add celery worker, celery-beat worker and celery broker to docker-compose setup.
+* Global search.
+* License information page.
+* Roles and permissions.
+* User preferences.
+* Additional fields for people per group.
+* Support global permission flags by LDAP group.
+* Persistent announcements.
+* Custom menu entries (e.g. in footer).
+* New logo for AlekSIS.
+* Two factor authentication with Yubikey, OTP or SMS.
+* Devs: Add ExtensibleModel to allow apps to add fields, properties.
+* Devs: Support multiple recipient object for one announcement.
+
+Changes
+~~~~~~~
+
+* Make short_name for group optional.
+* Generalised live loading of widgets for dashboard.
+* Devs: Add some CSS helper classes for colours.
+* Devs: Mandate use of AlekSIS base model.
+* Devs: Drop import_ref field(s); apps shold now define their own reference fields.
+
+Fixed
+~~~~~
+
+* DateTimeField Announcement.valid_from received a naive datetime.
+* Enable SASS processor in production.
+* Fix too short fields.
+* Load select2 locally.
+
+`2.0a1`_ - 2020-02-01
+---------------------
+
+Added
+~~~~~
+
+* Migrate to MaterializeCSS.
+* Dashboard.
+* Notifications via SMS (Twilio), Email or on the dashboard.
+* Admin interface.
+* Turn into installable, progressive web app.
+* Devs: Background Tasks with Celery.
+
+Changed
+~~~~~~~
+
+* Customisable save_button template.
+* Redesign error pages.
+
+Fixed
+~~~~~
+
+* setup_data no longer forces database connection.
+
+`1.0a4`_ - 2019-11-25
+---------------------
+
+Added
+~~~~~
 
 * Two-factor authentication with TOTP (Google Authenticator), Yubikey, SMS
   and phone call.
 * Devs: CRUDMixin provides a crud_event relation that returns all CRUD
-  events for an object
+  events for an object.
 
-`1.0a2`_
---------
+`1.0a2`_ - 2019-11-11
+---------------------
 
-New features
-~~~~~~~~~~~~
+Added
+~~~~~
 
 * Devs: Add ExtensibleModel to allow injection of methods and properties into models.
 
 
-`1.0a1`_
---------
-
-New features
-~~~~~~~~~~~~
-
-* Devs: Add API to get an audit trail for any school-related object
-* Devs: Provide template snippet to display an audit trail
-* Devs: Provide base template for views that allow browsing back/forth
-* Add management command and Cron job for full backups
-* Add system status overview page
-* Allow enabling and disabling maintenance mode from frontend
-* Allow editing the dates of the current school term
-* Add logo to school information
-* Allow editing school information
-* Ensure all actions are reverted if something fails (atomic requests)
+`1.0a1`_ - 2019-09-17
+---------------------
 
-Bugfixes
-~~~~~~~~
+Added
+~~~~~
 
-* Only show active persons in group and persons views
-* Silence KeyError in get_dict template tag
+* Devs: Add API to get an audit trail for any school-related object.
+* Devs: Provide template snippet to display an audit trail.
+* Devs: Provide base template for views that allow browsing back/forth.
+* Add management command and Cron job for full backups.
+* Add system status overview page.
+* Allow enabling and disabling maintenance mode from frontend.
+* Allow editing the dates of the current school term.
+* Add logo to school information.
+* Allow editing school information.
+* Ensure all actions are reverted if something fails (atomic requests).
 
-Minor changes
-~~~~~~~~~~~~~
+Fixed
+~~~~~
 
-* Use bootstrap buttons everywhere
+* Only show active persons in group and persons views.
+* Silence KeyError in get_dict template tag.
+* Use bootstrap buttons everywhere.
 
+.. _Keep a Changelog: https://keepachangelog.com/en/1.0.0/
+.. _Semantic Versioning: https://semver.org/spec/v2.0.0.html
 
-_`1.0a1`: https://edugit.org/AlekSIS/Official/AlekSIS/-/tags/1.0a1
-_`1.0a2`: https://edugit.org/AlekSIS/Official/AlekSIS/-/tags/1.0a2
-_`1.0a4`: https://edugit.org/AlekSIS/Official/AlekSIS/-/tags/1.0a4
-_`2.0a1`: https://edugit.org/AlekSIS/Official/AlekSIS/-/tags/2.0a1
-_`2.0a2`: https://edugit.org/AlekSIS/Official/AlekSIS/-/tags/2.0a2
+.. _1.0a1: https://edugit.org/AlekSIS/Official/AlekSIS/-/tags/1.0a1
+.. _1.0a2: https://edugit.org/AlekSIS/Official/AlekSIS/-/tags/1.0a2
+.. _1.0a4: https://edugit.org/AlekSIS/Official/AlekSIS/-/tags/1.0a4
+.. _2.0a1: https://edugit.org/AlekSIS/Official/AlekSIS/-/tags/2.0a1
+.. _2.0a2: https://edugit.org/AlekSIS/Official/AlekSIS/-/tags/2.0a2
+.. _2.0b0: https://edugit.org/AlekSIS/Official/AlekSIS/-/tags/2.0b0
+.. _2.0b1: https://edugit.org/AlekSIS/Official/AlekSIS/-/tags/2.0b1
+.. _2.0b2: https://edugit.org/AlekSIS/Official/AlekSIS/-/tags/2.0b2
+.. _2.0rc1: https://edugit.org/AlekSIS/Official/AlekSIS/-/tags/2.0rc1
+.. _2.0rc2: https://edugit.org/AlekSIS/Official/AlekSIS/-/tags/2.0rc2
+.. _2.0rc3: https://edugit.org/AlekSIS/Official/AlekSIS/-/tags/2.0rc3
+.. _2.0rc4: https://edugit.org/AlekSIS/Official/AlekSIS/-/tags/2.0rc4
+.. _2.0rc5: https://edugit.org/AlekSIS/Official/AlekSIS/-/tags/2.0rc5
+.. _2.0rc6: https://edugit.org/AlekSIS/Official/AlekSIS/-/tags/2.0rc6
+.. _2.0rc7: https://edugit.org/AlekSIS/Official/AlekSIS/-/tags/2.0rc7
+.. _2.0: https://edugit.org/AlekSIS/Official/AlekSIS/-/tags/2.0
+.. _2.1: https://edugit.org/AlekSIS/Official/AlekSIS/-/tags/2.1
diff --git a/Dockerfile b/Dockerfile
index c2b7db5ebe8f4b8f2198ceb62fdfd984e541e897..a44a0ce8021f06d34e39c7d6c594602cb623f690 100644
--- a/Dockerfile
+++ b/Dockerfile
@@ -26,7 +26,7 @@ RUN apt-get -y update && \
     eatmydata apt-get -y upgrade && \
     eatmydata apt-get install -y --no-install-recommends \
         build-essential \
-    chromium \
+        chromium \
 	dumb-init \
 	gettext \
 	libpq5 \
@@ -34,6 +34,7 @@ RUN apt-get -y update && \
 	libssl-dev \
 	postgresql-client \
 	python3-dev \
+	python3-magic \
 	python3-pip \
 	uwsgi \
 	uwsgi-plugin-python3 \
@@ -94,7 +95,18 @@ USER 33:33
 
 # Additional steps
 ONBUILD ARG APPS
+ONBUILD ARG BUILD_DEPS
+ONBUILD ARG SYSTEM_DEPS
 ONBUILD USER 0:0
+ONBUILD RUN set -e; \
+            if [ -n "$BUILD_DEPS" ]; then \
+                eatmydata apt-get update; \
+                eatmydata apt-get install -y $BUILD_DEPS; \
+            fi; \
+            if [ -n "$SYSTEM_DEPS" ]; then \
+                eatmydata apt-get update; \
+                eatmydata apt-get install -y $SYSTEM_DEPS; \
+            fi;
 ONBUILD RUN set -e; \
             if [ -n "$APPS" ]; then \
                 eatmydata pip install $APPS; \
@@ -102,7 +114,7 @@ ONBUILD RUN set -e; \
             eatmydata aleksis-admin yarn install; \
             eatmydata aleksis-admin collectstatic --no-input; \
             rm -rf /usr/local/share/.cache; \
-            eatmydata apt-get remove --purge -y yarnpkg; \
+            eatmydata apt-get remove --purge -y yarnpkg $BUILD_DEPS; \
             eatmydata apt-get autoremove --purge -y; \
             apt-get clean -y; \
             rm -f /var/lib/apt/lists/*_*; \
diff --git a/README.rst b/README.rst
index 5946b0d4eb15331b918e95446fddbc440250bfcb..de1fb1fe17c9e8e57f49ad94987e8847f6263a08 100644
--- a/README.rst
+++ b/README.rst
@@ -1,48 +1,72 @@
 AlekSIS (School Information System) — Core (Core functionality and app framework)
 =================================================================================
 
-AlekSIS standard distribution
------------------------------
+This is the core of the AlekSIS framework and the official distribution
+(see below). It bundles functionality for all apps, and utilities for
+developers and administrators.
 
-The AlekSIS standard distribution with information about all official apps
-can be found on `EduGit`_.
+If you are looking for the AlekSIS standard distribution, i.e. the complete
+software product ready for installation and usage, please visit the `AlekSIS`_
+website or the distribution repository on `EduGit`_.
 
 Features
 --------
 
-The AlekSIS-Core currently provides the following features:
+The AlekSIS core currently provides the following features:
 
 * For users:
 
+ * Authentication via OAuth applications
+ * Configurable dashboard
  * Custom menu entries (e.g. in footer)
  * Global preferences
+ * Global search
  * Group types
  * Manage announcements
  * Manage groups
  * Manage persons
  * Notifications via SMS email or dashboard
+ * PWA with offline caching
  * Rules and permissions for users, objects and pages
  * Two factor authentication via Yubikey, OTP or SMS
  * User preferences
+ * User registration, password changes and password reset
 
 * For admins
 
  * Asynchronous tasks with celery
  * Authentication via LDAP
  * Automatic backup of database, static and media files
+ * Generic PDF generation with chromium
+ * OAuth2 and OpenID Connect provider support
+ * Serve prometheus metrics
+ * System health and data checks
+
+* For developers
+
+ * `aleksis-admin` script to wrap django-admin with pre-configured settings
+ * Caching with Redis
+ * Django REST framework for apps to use at own discretion
+ * Injection of fields, methods, permissions and properties via custom `ExtensibleModel`
+ * K8s compatible, read-only Docker image
+ * Object-level permissions and rules with `django-guardian` and `django-rules`
+ * Query caching with `django-cachalot`
+ * Search with `django-haystack` and `Whoosh` backend
+ * uWSGI and Celery via `django-uwsgi` in development
 
 Licence
 -------
 
 ::
 
-  Copyright © 2017, 2018, 2019, 2020, 2021 Jonathan Weth <wethjo@katharineum.de>
+  Copyright © 2017, 2018, 2019, 2020, 2021 Jonathan Weth <dev@jonathanweth.de>
   Copyright © 2017, 2018, 2019, 2020 Frank Poetzsch-Heffter <p-h@katharineum.de>
   Copyright © 2018, 2019, 2020, 2021 Julian Leucker <leuckeju@katharineum.de>
   Copyright © 2018, 2019, 2020, 2021 Hangzhi Yu <yuha@katharineum.de>
   Copyright © 2019, 2020, 2021 Dominik George <dominik.george@teckids.org>
   Copyright © 2019, 2020, 2021 Tom Teichler <tom.teichler@teckids.org>
   Copyright © 2019 mirabilos <thorsten.glaser@teckids.org>
+  Copyright © 2021 Lloyd Meins <meinsll@katharineum.de>
   Copyright © 2021 magicfelix <felix@felix-zauberer.de>
 
   Licenced under the EUPL, version 1.2 or later, by Teckids e.V. (Bonn, Germany).
@@ -52,6 +76,6 @@ full licence text or on the `European Union Public Licence`_ website
 https://joinup.ec.europa.eu/collection/eupl/guidelines-users-and-developers
 (including all other official language versions).
 
-.. _AlekSIS: https://edugit.org/AlekSIS/Official/AlekSIS
+.. _AlekSIS: https://aleksis.org
 .. _European Union Public Licence: https://eupl.eu/
 .. _EduGit: https://edugit.org/AlekSIS/official/AlekSIS
diff --git a/aleksis/core/__init__.py b/aleksis/core/__init__.py
index 0cce3caca0b209dd4cd9bc26de1ae50ad9482a2d..79928246f83965c5c39479326114e3a7a89e27cb 100644
--- a/aleksis/core/__init__.py
+++ b/aleksis/core/__init__.py
@@ -1,4 +1,4 @@
-import pkg_resources
+from importlib import metadata
 
 try:
     from .celery import app as celery_app
@@ -7,7 +7,7 @@ except ModuleNotFoundError:
     celery_app = None
 
 try:
-    __version__ = pkg_resources.get_distribution("AlekSIS-Core").version
+    __version__ = metadata.distribution("AlekSIS-Core").version
 except Exception:
     __version__ = "unknown"
 
diff --git a/aleksis/core/anonymizers.py b/aleksis/core/anonymizers.py
deleted file mode 100644
index 3a9094c4140347ac5b738402ef5731582d064a2e..0000000000000000000000000000000000000000
--- a/aleksis/core/anonymizers.py
+++ /dev/null
@@ -1,26 +0,0 @@
-from hattori.base import BaseAnonymizer, faker
-
-from .models import Person
-
-
-class PersonAnonymizer(BaseAnonymizer):
-    model = Person
-
-    attributes = [
-        ("first_name", faker.first_name),
-        ("last_name", faker.last_name),
-        ("additional_name", ""),
-        ("short_name", lambda **kwargs: faker.pystr(min_chars=3, max_chars=5, **kwargs)),
-        ("street", faker.street_name),
-        ("housenumber", faker.building_number),
-        ("postal_code", faker.postcode),
-        ("place", faker.city),
-        ("phone_number", ""),
-        ("mobile_number", ""),
-        ("email", faker.email),
-        (
-            "date_of_birth",
-            lambda **kwargs: faker.date_of_birth(minimum_age=8, maximum_age=66, **kwargs),
-        ),
-        ("photo", ""),
-    ]
diff --git a/aleksis/core/apps.py b/aleksis/core/apps.py
index fa48d04407a4e047ea98e6ce7b1102158ba25fbb..b549995264811d731cb61a8ed934e579dcefc33e 100644
--- a/aleksis/core/apps.py
+++ b/aleksis/core/apps.py
@@ -2,8 +2,10 @@ from typing import Any, Optional
 
 import django.apps
 from django.apps import apps
+from django.conf import settings
 from django.http import HttpRequest
 from django.utils.module_loading import autodiscover_modules
+from django.utils.translation import gettext as _
 
 from dynamic_preferences.registries import preference_models
 from health_check.plugins import plugin_dir
@@ -14,7 +16,7 @@ from .registries import (
     site_preferences_registry,
 )
 from .util.apps import AppConfig
-from .util.core_helpers import has_person
+from .util.core_helpers import get_or_create_favicon, has_person
 from .util.sass_helpers import clean_scss
 
 
@@ -35,12 +37,15 @@ class CoreConfig(AppConfig):
         ([2019, 2020, 2021], "Dominik George", "dominik.george@teckids.org"),
         ([2019, 2020, 2021], "Tom Teichler", "tom.teichler@teckids.org"),
         ([2019], "mirabilos", "thorsten.glaser@teckids.org"),
+        ([2021], "Lloyd Meins", "meinsll@katharineum.de"),
         ([2021], "magicfelix", "felix@felix-zauberer.de"),
     )
 
     def ready(self):
         super().ready()
 
+        from django.conf import settings  # noqa
+
         # Autodiscover various modules defined by AlekSIS
         autodiscover_modules("form_extensions", "model_extensions", "checks")
 
@@ -71,9 +76,9 @@ class CoreConfig(AppConfig):
         """Get all data checks from all loaded models."""
         from aleksis.core.data_checks import DataCheckRegistry
 
-        data_checks = []
+        data_checks = set()
         for model in apps.get_models():
-            data_checks += getattr(model, "data_checks", [])
+            data_checks.update(getattr(model, "data_checks", []))
         DataCheckRegistry.data_checks = data_checks
 
     def preference_updated(
@@ -85,6 +90,8 @@ class CoreConfig(AppConfig):
         new_value: Optional[Any] = None,
         **kwargs,
     ) -> None:
+        from django.conf import settings  # noqa
+
         if section == "theme":
             if name in ("primary", "secondary"):
                 clean_scss()
@@ -95,11 +102,14 @@ class CoreConfig(AppConfig):
 
                 if new_value:
                     Favicon.on_site.update_or_create(
-                        title=name,
-                        defaults={"isFavicon": name == "favicon", "faviconImage": new_value,},
+                        title=name, defaults={"isFavicon": is_favicon, "faviconImage": new_value},
                     )
                 else:
                     Favicon.on_site.filter(title=name, isFavicon=is_favicon).delete()
+                    if name in settings.DEFAULT_FAVICON_PATHS:
+                        get_or_create_favicon(
+                            name, settings.DEFAULT_FAVICON_PATHS[name], is_favicon=is_favicon
+                        )
 
     def post_migrate(
         self,
@@ -109,6 +119,8 @@ class CoreConfig(AppConfig):
         using: str,
         **kwargs,
     ) -> None:
+        from django.conf import settings  # noqa
+
         super().post_migrate(app_config, verbosity, interactive, using, **kwargs)
 
         # Ensure presence of an OTP YubiKey default config
@@ -116,9 +128,29 @@ class CoreConfig(AppConfig):
             name="default", defaults={"use_ssl": True, "param_sl": "", "param_timeout": ""}
         )
 
+        # Ensure that default Favicon object exists
+        for name, default in settings.DEFAULT_FAVICON_PATHS.items():
+            get_or_create_favicon(name, default, is_favicon=name == "favicon")
+
     def user_logged_in(
         self, sender: type, request: Optional[HttpRequest], user: "User", **kwargs
     ) -> None:
         if has_person(user):
             # Save the associated person to pick up defaults
             user.person.save()
+
+    @classmethod
+    def get_all_scopes(cls) -> dict[str, str]:
+        scopes = {
+            "read": "Read anything the resource owner can read",
+            "write": "Write anything the resource owner can write",
+        }
+        if settings.OAUTH2_PROVIDER.get("OIDC_ENABLED", False):
+            scopes |= {
+                "openid": _("OpenID Connect scope"),
+                "profile": _("Given name, family name, link to profile and picture if existing."),
+                "address": _("Full home postal address"),
+                "email": _("Email address"),
+                "phone": _("Home and mobile phone"),
+            }
+        return scopes
diff --git a/aleksis/core/data_checks.py b/aleksis/core/data_checks.py
index a7d43d293a34e69f329cf9506116cd36d8a5103c..234c82c0ca12b46c7e1aebd7151fb2f2a1c0c2ca 100644
--- a/aleksis/core/data_checks.py
+++ b/aleksis/core/data_checks.py
@@ -1,5 +1,6 @@
 import logging
 
+from django.apps import apps
 from django.contrib.contenttypes.models import ContentType
 from django.db.models.aggregates import Count
 from django.utils.functional import classproperty
@@ -156,11 +157,19 @@ class DataCheck:
 
     solve_options = {IgnoreSolveOption.name: IgnoreSolveOption}
 
+    _current_results = []
+
     @classmethod
     def check_data(cls):
         """Find all objects with data issues and register them."""
         pass
 
+    @classmethod
+    def run_check_data(cls):
+        """Wrap ``check_data`` to ensure that post-processing tasks are run."""
+        cls.check_data()
+        cls.delete_old_results()
+
     @classmethod
     def solve(cls, check_result: "DataCheckResult", solve_option: str):
         """Execute a solve option for an object detected by this check.
@@ -188,16 +197,35 @@ class DataCheck:
         from aleksis.core.models import DataCheckResult
 
         ct = ContentType.objects.get_for_model(instance)
-        result = DataCheckResult.objects.get_or_create(
+        result, __ = DataCheckResult.objects.get_or_create(
             check=cls.name, content_type=ct, object_id=instance.id
         )
+
+        # Track all existing problems (for deleting old results)
+        cls._current_results.append(result)
+
         return result
 
+    @classmethod
+    def delete_old_results(cls):
+        """Delete old data check results for problems which exist no longer."""
+        DataCheckResult = apps.get_model("core", "DataCheckResult")
+
+        pks = [r.pk for r in cls._current_results]
+        old_results = DataCheckResult.objects.filter(check=cls.name).exclude(pk__in=pks)
+
+        if old_results:
+            logging.info(f"Delete {old_results.count()} old data check results.")
+            old_results.delete()
+
+        # Reset list with existing problems
+        cls._current_results = []
+
 
 class DataCheckRegistry:
     """Create central registry for all data checks in AlekSIS."""
 
-    data_checks = []
+    data_checks: set = set()
 
     @classproperty
     def data_checks_by_name(cls):
@@ -213,7 +241,7 @@ def check_data(recorder: ProgressRecorder):
     """Execute all registered data checks and send email if activated."""
     for check in recorder.iterate(DataCheckRegistry.data_checks):
         logging.info(f"Run check: {check.verbose_name}")
-        check.check_data()
+        check.run_check_data()
 
     if get_site_preferences()["general__data_checks_send_emails"]:
         send_emails_for_data_checks()
@@ -256,3 +284,35 @@ def send_emails_for_data_checks():
         logging.info("Sent notification email because of unsent data checks")
 
         results.update(sent=True)
+
+
+class DeactivateDashboardWidgetSolveOption(SolveOption):
+    name = "deactivate_dashboard_widget"
+    verbose_name = _("Deactivate DashboardWidget")
+
+    @classmethod
+    def solve(cls, check_result: "DataCheckResult"):
+        widget = check_result.related_object
+        widget.active = False
+        widget.save()
+        check_result.delete()
+
+
+class BrokenDashboardWidgetDataCheck(DataCheck):
+    name = "broken_dashboard_widgets"
+    verbose_name = _("Ensure that there are no broken DashboardWidgets.")
+    problem_name = _("The DashboardWidget was reported broken automatically.")
+    solve_options = {
+        IgnoreSolveOption.name: IgnoreSolveOption,
+        DeactivateDashboardWidgetSolveOption.name: DeactivateDashboardWidgetSolveOption,
+    }
+
+    @classmethod
+    def check_data(cls):
+        from .models import DashboardWidget
+
+        broken_widgets = DashboardWidget.objects.filter(broken=True, active=True)
+
+        for widget in broken_widgets:
+            logging.info("Check DashboardWidget %s", widget)
+            cls.register_result(widget)
diff --git a/aleksis/core/forms.py b/aleksis/core/forms.py
index 76e78077cfb6491e570e6d1ee29e98a148066e16..6057d63184d29116756ff5db3ac36cdc8bea7466 100644
--- a/aleksis/core/forms.py
+++ b/aleksis/core/forms.py
@@ -1,7 +1,8 @@
 from datetime import datetime, time
-from typing import Any, Callable, Dict, List, Sequence, Tuple
+from typing import Any, Callable, Dict, Sequence
 
 from django import forms
+from django.conf import settings
 from django.contrib.auth import get_user_model
 from django.contrib.auth.models import Permission
 from django.core.exceptions import ValidationError
@@ -9,9 +10,11 @@ from django.db.models import QuerySet
 from django.http import HttpRequest
 from django.utils.translation import gettext_lazy as _
 
+from allauth.account.adapter import get_adapter
+from allauth.account.forms import SignupForm
+from allauth.account.utils import setup_user_email
 from django_select2.forms import ModelSelect2MultipleWidget, ModelSelect2Widget, Select2Widget
 from dynamic_preferences.forms import PreferenceForm
-from guardian.core import ObjectPermissionChecker
 from guardian.shortcuts import assign_perm
 from material import Fieldset, Layout, Row
 
@@ -22,6 +25,7 @@ from .models import (
     DashboardWidget,
     Group,
     GroupType,
+    OAuthApplication,
     Person,
     SchoolTerm,
 )
@@ -30,58 +34,12 @@ from .registries import (
     person_preferences_registry,
     site_preferences_registry,
 )
+from .util.auth_helpers import AppScopes
+from .util.core_helpers import get_site_preferences
 
 
-class PersonAccountForm(forms.ModelForm):
-    """Form to assign user accounts to persons in the frontend."""
-
-    class Meta:
-        model = Person
-        fields = ["last_name", "first_name", "user"]
-        widgets = {"user": Select2Widget(attrs={"class": "browser-default"})}
-
-    new_user = forms.CharField(required=False)
-
-    def __init__(self, *args, **kwargs):
-        super().__init__(*args, **kwargs)
-
-        # Fields displayed only for informational purposes
-        self.fields["first_name"].disabled = True
-        self.fields["last_name"].disabled = True
-
-    def clean(self) -> None:
-        user = get_user_model()
-
-        if self.cleaned_data.get("new_user", None):
-            if self.cleaned_data.get("user", None):
-                # The user selected both an existing user and provided a name to create a new one
-                self.add_error(
-                    "new_user",
-                    _("You cannot set a new username when also selecting an existing user."),
-                )
-            elif user.objects.filter(username=self.cleaned_data["new_user"]).exists():
-                # The user tried to create a new user with the name of an existing user
-                self.add_error("new_user", _("This username is already in use."))
-            else:
-                # Create new User object and assign to form field for existing user
-                new_user_obj = user.objects.create_user(
-                    self.cleaned_data["new_user"],
-                    self.instance.email,
-                    first_name=self.instance.first_name,
-                    last_name=self.instance.last_name,
-                )
-
-                self.cleaned_data["user"] = new_user_obj
-
-
-# Formset for batch-processing of assignments of users to persons
-PersonsAccountsFormSet = forms.modelformset_factory(
-    Person, form=PersonAccountForm, max_num=0, extra=0
-)
-
-
-class EditPersonForm(ExtensibleForm):
-    """Form to edit an existing person object in the frontend."""
+class PersonForm(ExtensibleForm):
+    """Form to edit or add a person object in the frontend."""
 
     layout = Layout(
         Fieldset(
@@ -94,7 +52,11 @@ 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"), "guardians",
+            _("Advanced personal data"),
+            Row("date_of_birth", "place_of_birth"),
+            Row("sex"),
+            Row("photo"),
+            "guardians",
         ),
     )
 
@@ -115,6 +77,7 @@ class EditPersonForm(ExtensibleForm):
             "mobile_number",
             "email",
             "date_of_birth",
+            "place_of_birth",
             "sex",
             "photo",
             "guardians",
@@ -140,25 +103,49 @@ class EditPersonForm(ExtensibleForm):
         required=False, label=_("New user"), help_text=_("Create a new account")
     )
 
-    def __init__(self, request: HttpRequest, *args, **kwargs):
+    def __init__(self, *args, **kwargs):
+        request = kwargs.pop("request", None)
         super().__init__(*args, **kwargs)
 
         # Disable non-editable fields
-        person_fields = set([field.name for field in Person.syncable_fields()]).intersection(
-            set(self.fields)
-        )
+        allowed_person_fields = get_site_preferences()["account__editable_fields_person"]
 
-        if self.instance:
-            checker = ObjectPermissionChecker(request.user)
-            checker.prefetch_perms([self.instance])
+        if (
+            request
+            and self.instance
+            and not request.user.has_perm("core.change_person", self.instance)
+        ):
+            # First, disable all fields
+            for field in self.fields:
+                self.fields[field].disabled = True
 
-            for field in person_fields:
-                if not checker.has_perm(f"core.change_person_field_{field}", self.instance):
-                    self.fields[field].disabled = True
+            # Then, activate allowed fields
+            for field in allowed_person_fields:
+                self.fields[field].disabled = False
 
     def clean(self) -> None:
-        # Use code implemented in dedicated form to verify user selection
-        return PersonAccountForm.clean(self)
+        user = get_user_model()
+
+        if self.cleaned_data.get("new_user", None):
+            if self.cleaned_data.get("user", None):
+                # The user selected both an existing user and provided a name to create a new one
+                self.add_error(
+                    "new_user",
+                    _("You cannot set a new username when also selecting an existing user."),
+                )
+            elif user.objects.filter(username=self.cleaned_data["new_user"]).exists():
+                # The user tried to create a new user with the name of an existing user
+                self.add_error("new_user", _("This username is already in use."))
+            else:
+                # Create new User object and assign to form field for existing user
+                new_user_obj = user.objects.create_user(
+                    self.cleaned_data["new_user"],
+                    self.instance.email,
+                    first_name=self.instance.first_name,
+                    last_name=self.instance.last_name,
+                )
+
+                self.cleaned_data["user"] = new_user_obj
 
 
 class EditGroupForm(SchoolTermRelatedExtensibleForm):
@@ -379,8 +366,7 @@ class SchoolTermForm(ExtensibleForm):
 
 class DashboardWidgetOrderForm(ExtensibleForm):
     pk = forms.ModelChoiceField(
-        queryset=DashboardWidget.objects.all(),
-        widget=forms.HiddenInput(attrs={"class": "pk-input"}),
+        queryset=None, widget=forms.HiddenInput(attrs={"class": "pk-input"}),
     )
     order = forms.IntegerField(initial=0, widget=forms.HiddenInput(attrs={"class": "order-input"}))
 
@@ -388,6 +374,12 @@ class DashboardWidgetOrderForm(ExtensibleForm):
         model = DashboardWidget
         fields = []
 
+    def __init__(self, *args, **kwargs):
+        super().__init__(*args, **kwargs)
+
+        # Set queryset here to prevent problems with not migrated database due to special queryset
+        self.fields["pk"].queryset = DashboardWidget.objects.all()
+
 
 DashboardWidgetOrderFormSet = forms.formset_factory(
     form=DashboardWidgetOrderForm, max_num=0, extra=0
@@ -504,6 +496,81 @@ class AssignPermissionForm(forms.Form):
                 assign_perm(permission_name, django_group, instance)
 
 
+class AccountRegisterForm(SignupForm, ExtensibleForm):
+    """Form to register new user accounts."""
+
+    class Meta:
+        model = Group
+        fields = []
+
+    layout = Layout(
+        Fieldset(_("Base data"), Row("first_name", "last_name"),),
+        Fieldset(
+            _("Account data"), "username", Row("email", "email2"), Row("password1", "password2"),
+        ),
+        Fieldset(_("Consents"), Row("privacy_policy"),),
+    )
+
+    def __init__(self, *args, **kwargs):
+        super(AccountRegisterForm, self).__init__(*args, **kwargs)
+        self.fields["password1"] = forms.CharField(label=_("Password"), widget=forms.PasswordInput)
+
+        privacy_policy = get_site_preferences()["footer__privacy_url"]
+
+        if settings.SIGNUP_PASSWORD_ENTER_TWICE:
+            self.fields["password2"] = forms.CharField(
+                label=_("Password (again)"), widget=forms.PasswordInput
+            )
+
+        self.fields["first_name"] = forms.CharField(required=True,)
+
+        self.fields["last_name"] = forms.CharField(required=True,)
+
+        self.fields["privacy_policy"] = forms.BooleanField(
+            help_text=_(
+                f"I have read the <a href='{privacy_policy}'>Privacy policy</a>"
+                " and agree with them."
+            ),
+            required=True,
+        )
+
+    def clean(self):
+        super(AccountRegisterForm, self).clean()
+
+        dummy_user = get_user_model()
+        password = self.cleaned_data.get("password1")
+        if password:
+            try:
+                get_adapter().clean_password(password, user=dummy_user)
+            except forms.ValidationError as e:
+                self.add_error("password1", e)
+
+        if (
+            settings.SIGNUP_PASSWORD_ENTER_TWICE
+            and "password1" in self.cleaned_data
+            and "password2" in self.cleaned_data
+        ):
+            if self.cleaned_data["password1"] != self.cleaned_data["password2"]:
+                self.add_error(
+                    "password2", _("You must type the same password each time."),
+                )
+        return self.cleaned_data
+
+    def save(self, request):
+        adapter = get_adapter(request)
+        user = adapter.new_user(request)
+        adapter.save_user(request, user, self)
+        Person.objects.create(
+            first_name=self.cleaned_data["first_name"],
+            last_name=self.cleaned_data["last_name"],
+            email=self.cleaned_data["email"],
+            user=user,
+        )
+        self.custom_signup(request, user)
+        setup_user_email(request, user, [])
+        return user
+
+
 class ActionForm(forms.Form):
     """Generic form for executing actions on multiple items of a queryset.
 
@@ -555,11 +622,11 @@ class ActionForm(forms.Form):
         """Get all defined actions."""
         return self.actions
 
-    def _get_actions_dict(self) -> Dict[str, Callable]:
+    def _get_actions_dict(self) -> dict[str, Callable]:
         """Get all defined actions as dictionary."""
         return {value.__name__: value for value in self.get_actions()}
 
-    def _get_action_choices(self) -> List[Tuple[str, str]]:
+    def _get_action_choices(self) -> list[tuple[str, str]]:
         """Get all defined actions as Django choices."""
         return [
             (value.__name__, getattr(value, "short_description", value.__name__))
@@ -616,15 +683,15 @@ class ListActionForm(ActionForm):
         # Return None in order not to raise an unwanted exception
         return None
 
-    def _get_dict(self) -> Dict[str, dict]:
+    def _get_dict(self) -> dict[str, dict]:
         """Get the items sorted by pk attribute."""
         return {item["pk"]: item for item in self.items}
 
-    def _get_choices(self) -> List[Tuple[str, str]]:
+    def _get_choices(self) -> list[tuple[str, str]]:
         """Get the items as Django choices."""
         return [(item["pk"], item["pk"]) for item in self.items]
 
-    def _get_real_items(self, items: Sequence[dict]) -> List[dict]:
+    def _get_real_items(self, items: Sequence[dict]) -> list[dict]:
         """Get the real dictionaries from a list of pks."""
         items_dict = self._get_dict()
         real_items = []
@@ -634,7 +701,7 @@ class ListActionForm(ActionForm):
             real_items.append(items_dict[item])
         return real_items
 
-    def clean_selected_objects(self) -> List[dict]:
+    def clean_selected_objects(self) -> list[dict]:
         data = self.cleaned_data["selected_objects"]
         items = self._get_real_items(data)
         return items
@@ -643,3 +710,23 @@ class ListActionForm(ActionForm):
         self.items = items
         super().__init__(request, *args, **kwargs)
         self.fields["selected_objects"].choices = self._get_choices()
+
+
+class OAuthApplicationForm(forms.ModelForm):
+    def __init__(self, *args, **kwargs):
+        super().__init__(*args, **kwargs)
+        self.fields["allowed_scopes"].widget = forms.SelectMultiple(
+            choices=list(AppScopes().get_all_scopes().items())
+        )
+
+    class Meta:
+        model = OAuthApplication
+        fields = (
+            "name",
+            "client_id",
+            "client_secret",
+            "client_type",
+            "allowed_scopes",
+            "redirect_uris",
+            "skip_authorization",
+        )
diff --git a/aleksis/core/health_checks.py b/aleksis/core/health_checks.py
index 49677765f6a2fcc52eafe227027ee381ff6da4c9..200a6eaf0a35a75c29ca4cf7e8c564925b1e29e8 100644
--- a/aleksis/core/health_checks.py
+++ b/aleksis/core/health_checks.py
@@ -34,10 +34,13 @@ class BaseBackupHealthCheck(BaseHealthCheckBackend):
     def check_status(self):
         storage = get_storage()
         backups = storage.list_backups(content_type=self.content_type)
+        if not storage.storage.exists(""):
+            self.add_error(_("The backup folder doesn't exist."))
+            return
         if backups:
-            last_backup = backups[-1]
-            last_backup_time = dbbackup_utils.filename_to_date(last_backup)
-            time_gone_since_backup = last_backup_time - datetime.now()
+            last_backup = backups[:1]
+            last_backup_time = dbbackup_utils.filename_to_date(last_backup[0])
+            time_gone_since_backup = datetime.now() - last_backup_time
 
             # Check if backup is older than configured time
             if time_gone_since_backup.seconds > self.configured_seconds:
diff --git a/aleksis/core/locale/ar/LC_MESSAGES/django.po b/aleksis/core/locale/ar/LC_MESSAGES/django.po
index 26513bbf79d22e23b2c3f8f90b93b5474034a972..29ee921af8694936e102e32d65b310ac44979fbc 100644
--- a/aleksis/core/locale/ar/LC_MESSAGES/django.po
+++ b/aleksis/core/locale/ar/LC_MESSAGES/django.po
@@ -8,7 +8,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: AlekSIS (School Information System) 0.1\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2021-01-11 21:30+0100\n"
+"POT-Creation-Date: 2021-10-28 16:18+0200\n"
 "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
@@ -18,822 +18,1113 @@ msgstr ""
 "Content-Transfer-Encoding: 8bit\n"
 "Plural-Forms: nplurals=6; plural=n==0 ? 0 : n==1 ? 1 : n==2 ? 2 : n%100>=3 && n%100<=10 ? 3 : n%100>=11 && n%100<=99 ? 4 : 5;\n"
 
-#: data_checks.py:53
+#: aleksis/core/apps.py:150
+msgid "OpenID Connect scope"
+msgstr ""
+
+#: aleksis/core/apps.py:151
+msgid "Given name, family name, link to profile and picture if existing."
+msgstr ""
+
+#: aleksis/core/apps.py:152
+msgid "Full home postal address"
+msgstr ""
+
+#: aleksis/core/apps.py:153
+msgid "Email address"
+msgstr ""
+
+#: aleksis/core/apps.py:154
+msgid "Home and mobile phone"
+msgstr ""
+
+#: aleksis/core/data_checks.py:55
 msgid "Ignore problem"
 msgstr ""
 
-#: data_checks.py:174
+#: aleksis/core/data_checks.py:184
 #, python-brace-format
 msgid "Solve option '{solve_option_obj.verbose_name}' "
 msgstr ""
 
-#: filters.py:37 templates/core/base.html:77 templates/core/group/list.html:20
-#: templates/core/person/list.html:24 templates/search/search.html:7
-#: templates/search/search.html:22
-msgid "Search"
+#: aleksis/core/data_checks.py:291
+msgid "Deactivate DashboardWidget"
 msgstr ""
 
-#: filters.py:53
-msgid "Search by name"
+#: aleksis/core/data_checks.py:303
+msgid "Ensure that there are no broken DashboardWidgets."
 msgstr ""
 
-#: filters.py:65
-msgid "Search by contact details"
+#: aleksis/core/data_checks.py:304
+msgid "The DashboardWidget was reported broken automatically."
 msgstr ""
 
-#: forms.py:54
-msgid "You cannot set a new username when also selecting an existing user."
+#: aleksis/core/filters.py:37 aleksis/core/templates/core/base.html:83
+#: aleksis/core/templates/core/base.html:84
+#: aleksis/core/templates/core/group/list.html:20
+#: aleksis/core/templates/core/person/list.html:24
+#: aleksis/core/templates/search/search.html:7
+#: aleksis/core/templates/search/search.html:22
+msgid "Search"
 msgstr ""
 
-#: forms.py:58
-msgid "This username is already in use."
+#: aleksis/core/filters.py:53
+msgid "Search by name"
 msgstr ""
 
-#: forms.py:82
+#: aleksis/core/filters.py:65
+msgid "Search by contact details"
+msgstr ""
+
+#: aleksis/core/forms.py:41 aleksis/core/forms.py:387
 msgid "Base data"
 msgstr ""
 
-#: forms.py:88
+#: aleksis/core/forms.py:47
 msgid "Address"
 msgstr ""
 
-#: forms.py:89
+#: aleksis/core/forms.py:48
 msgid "Contact data"
 msgstr ""
 
-#: forms.py:91
+#: aleksis/core/forms.py:50
 msgid "Advanced personal data"
 msgstr ""
 
-#: forms.py:134
+#: aleksis/core/forms.py:93
 msgid "New user"
 msgstr ""
 
-#: forms.py:134
+#: aleksis/core/forms.py:93
 msgid "Create a new account"
 msgstr ""
 
-#: forms.py:146 models.py:102
+#: aleksis/core/forms.py:124
+msgid "You cannot set a new username when also selecting an existing user."
+msgstr ""
+
+#: aleksis/core/forms.py:128
+msgid "This username is already in use."
+msgstr ""
+
+#: aleksis/core/forms.py:145 aleksis/core/models.py:117
 msgid "School term"
 msgstr ""
 
-#: forms.py:147
+#: aleksis/core/forms.py:146
 msgid "Common data"
 msgstr ""
 
-#: forms.py:148 forms.py:197 menus.py:169 models.py:116
-#: templates/core/person/list.html:8 templates/core/person/list.html:9
+#: aleksis/core/forms.py:147 aleksis/core/forms.py:196
+#: aleksis/core/menus.py:238 aleksis/core/models.py:140
+#: aleksis/core/templates/core/person/list.html:8
+#: aleksis/core/templates/core/person/list.html:9
 msgid "Persons"
 msgstr ""
 
-#: forms.py:149
+#: aleksis/core/forms.py:148
 msgid "Additional data"
 msgstr ""
 
-#: forms.py:189 forms.py:192 models.py:45
+#: aleksis/core/forms.py:188 aleksis/core/forms.py:191
+#: aleksis/core/models.py:60
 msgid "Date"
 msgstr ""
 
-#: forms.py:190 forms.py:193 models.py:53
+#: aleksis/core/forms.py:189 aleksis/core/forms.py:192
+#: aleksis/core/models.py:68
 msgid "Time"
 msgstr ""
 
-#: forms.py:210 menus.py:177 models.py:338 templates/core/group/list.html:8
-#: templates/core/group/list.html:9 templates/core/person/full.html:144
+#: aleksis/core/forms.py:209 aleksis/core/menus.py:249
+#: aleksis/core/models.py:398 aleksis/core/templates/core/group/list.html:8
+#: aleksis/core/templates/core/group/list.html:9
+#: aleksis/core/templates/core/person/full.html:144
 msgid "Groups"
 msgstr ""
 
-#: forms.py:220
+#: aleksis/core/forms.py:219
 msgid "From when until when should the announcement be displayed?"
 msgstr ""
 
-#: forms.py:223
+#: aleksis/core/forms.py:222
 msgid "Who should see the announcement?"
 msgstr ""
 
-#: forms.py:224
+#: aleksis/core/forms.py:223
 msgid "Write your announcement:"
 msgstr ""
 
-#: forms.py:263
+#: aleksis/core/forms.py:262
 msgid "You are not allowed to create announcements which are only valid in the past."
 msgstr ""
 
-#: forms.py:267
+#: aleksis/core/forms.py:266
 msgid "The from date and time must be earlier then the until date and time."
 msgstr ""
 
-#: forms.py:276
+#: aleksis/core/forms.py:275
 msgid "You need at least one recipient."
 msgstr ""
 
-#: health_checks.py:15
+#: aleksis/core/forms.py:389
+msgid "Account data"
+msgstr ""
+
+#: aleksis/core/forms.py:391
+msgid "Consents"
+msgstr ""
+
+#: aleksis/core/forms.py:396
+msgid "Password"
+msgstr ""
+
+#: aleksis/core/forms.py:402
+msgid "Password (again)"
+msgstr ""
+
+#: aleksis/core/forms.py:411
+#, python-brace-format
+msgid "I have read the <a href='{privacy_policy}'>Privacy policy</a> and agree with them."
+msgstr ""
+
+#: aleksis/core/forms.py:435
+msgid "You must type the same password each time."
+msgstr ""
+
+#: aleksis/core/forms.py:580
+msgid "No valid selection."
+msgstr ""
+
+#: aleksis/core/health_checks.py:21
 msgid "There are unresolved data problems."
 msgstr ""
 
-#: menus.py:7 templates/two_factor/core/login.html:6
-#: templates/two_factor/core/login.html:10
-#: templates/two_factor/core/login.html:86
+#: aleksis/core/health_checks.py:38
+msgid "The backup folder doesn't exist."
+msgstr ""
+
+#: aleksis/core/health_checks.py:47
+#, python-brace-format
+msgid "Last backup {time_gone_since_backup}!"
+msgstr ""
+
+#: aleksis/core/health_checks.py:49
+msgid "No backup found!"
+msgstr ""
+
+#: aleksis/core/health_checks.py:76
+msgid "No backup result found!"
+msgstr ""
+
+#: aleksis/core/health_checks.py:78
+#, python-brace-format
+msgid "{task.status} - {task.result}"
+msgstr ""
+
+#: aleksis/core/menus.py:9 aleksis/core/templates/two_factor/core/login.html:6
+#: aleksis/core/templates/two_factor/core/login.html:22
+#: aleksis/core/templates/two_factor/core/login.html:76
 msgid "Login"
 msgstr ""
 
-#: menus.py:13
+#: aleksis/core/menus.py:15 aleksis/core/templates/account/signup.html:20
+#: aleksis/core/templates/socialaccount/signup.html:23
+msgid "Sign up"
+msgstr ""
+
+#: aleksis/core/menus.py:24
 msgid "Dashboard"
 msgstr ""
 
-#: menus.py:19
+#: aleksis/core/menus.py:32 aleksis/core/models.py:605
+#: aleksis/core/preferences.py:26
+#: aleksis/core/templates/core/notifications.html:4
+#: aleksis/core/templates/core/notifications.html:5
+msgid "Notifications"
+msgstr ""
+
+#: aleksis/core/menus.py:41
 msgid "Account"
 msgstr ""
 
-#: menus.py:26
+#: aleksis/core/menus.py:48
 msgid "Stop impersonation"
 msgstr ""
 
-#: menus.py:35 templates/core/base.html:56
+#: aleksis/core/menus.py:57 aleksis/core/templates/core/base.html:62
 msgid "Logout"
 msgstr ""
 
-#: menus.py:41
+#: aleksis/core/menus.py:63
 msgid "2FA"
 msgstr ""
 
-#: menus.py:47
+#: aleksis/core/menus.py:69
+#: aleksis/core/templates/account/password_change.html:5
+#: aleksis/core/templates/account/password_change.html:6
+#: aleksis/core/templates/account/password_change.html:19
+#: aleksis/core/templates/account/password_reset_from_key.html:5
+#: aleksis/core/templates/account/password_reset_from_key.html:42
+#: aleksis/core/templates/account/password_reset_from_key.html:46
+#: aleksis/core/templates/account/password_reset_from_key_done.html:5
+#: aleksis/core/templates/account/password_reset_from_key_done.html:6
+msgid "Change password"
+msgstr ""
+
+#: aleksis/core/menus.py:81
 msgid "Me"
 msgstr ""
 
-#: menus.py:56 templates/dynamic_preferences/form.html:5
+#: aleksis/core/menus.py:90
+#: aleksis/core/templates/dynamic_preferences/form.html:5
 msgid "Preferences"
 msgstr ""
 
-#: menus.py:67
+#: aleksis/core/menus.py:99
+msgid "Third-party accounts"
+msgstr ""
+
+#: aleksis/core/menus.py:108
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:5
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:6
+msgid "Authorized applications"
+msgstr ""
+
+#: aleksis/core/menus.py:119
 msgid "Admin"
 msgstr ""
 
-#: menus.py:75 models.py:602 templates/core/announcement/list.html:7
-#: templates/core/announcement/list.html:8
+#: aleksis/core/menus.py:127 aleksis/core/models.py:704
+#: aleksis/core/templates/core/announcement/list.html:7
+#: aleksis/core/templates/core/announcement/list.html:8
 msgid "Announcements"
 msgstr ""
 
-#: menus.py:86 models.py:103 templates/core/school_term/list.html:8
-#: templates/core/school_term/list.html:9
+#: aleksis/core/menus.py:138 aleksis/core/models.py:118
+#: aleksis/core/templates/core/school_term/list.html:8
+#: aleksis/core/templates/core/school_term/list.html:9
 msgid "School terms"
 msgstr ""
 
-#: menus.py:97 templates/core/dashboard_widget/list.html:8
-#: templates/core/dashboard_widget/list.html:9
+#: aleksis/core/menus.py:149
+#: aleksis/core/templates/core/dashboard_widget/list.html:8
+#: aleksis/core/templates/core/dashboard_widget/list.html:9
 msgid "Dashboard widgets"
 msgstr ""
 
-#: menus.py:108 templates/core/management/data_management.html:6
-#: templates/core/management/data_management.html:7
+#: aleksis/core/menus.py:160
+#: aleksis/core/templates/core/management/data_management.html:6
+#: aleksis/core/templates/core/management/data_management.html:7
 msgid "Data management"
 msgstr ""
 
-#: menus.py:116 templates/core/pages/system_status.html:5
-#: templates/core/pages/system_status.html:7
+#: aleksis/core/menus.py:171
+#: aleksis/core/templates/core/pages/system_status.html:5
+#: aleksis/core/templates/core/pages/system_status.html:7
 msgid "System status"
 msgstr ""
 
-#: menus.py:127
+#: aleksis/core/menus.py:182
 msgid "Impersonation"
 msgstr ""
 
-#: menus.py:135
+#: aleksis/core/menus.py:193
 msgid "Configuration"
 msgstr ""
 
-#: menus.py:146 templates/core/data_check/list.html:9
-#: templates/core/data_check/list.html:10
+#: aleksis/core/menus.py:204 aleksis/core/templates/core/data_check/list.html:9
+#: aleksis/core/templates/core/data_check/list.html:10
 msgid "Data checks"
 msgstr ""
 
-#: menus.py:152
+#: aleksis/core/menus.py:210
 msgid "Backend Admin"
 msgstr ""
 
-#: menus.py:160
-msgid "People"
+#: aleksis/core/menus.py:216
+#: aleksis/core/templates/oauth2_provider/application_detail.html:5
+#: aleksis/core/templates/oauth2_provider/application_list.html:5
+msgid "OAuth2 Applications"
 msgstr ""
 
-#: menus.py:185 models.py:812 templates/core/group_type/list.html:8
-#: templates/core/group_type/list.html:9
-msgid "Group types"
+#: aleksis/core/menus.py:229
+msgid "People"
 msgstr ""
 
-#: menus.py:196
-msgid "Persons and accounts"
+#: aleksis/core/menus.py:260 aleksis/core/models.py:958
+#: aleksis/core/templates/core/group_type/list.html:8
+#: aleksis/core/templates/core/group_type/list.html:9
+msgid "Group types"
 msgstr ""
 
-#: menus.py:207
+#: aleksis/core/menus.py:271
 msgid "Groups and child groups"
 msgstr ""
 
-#: menus.py:218 models.py:385 templates/core/additional_field/list.html:8
-#: templates/core/additional_field/list.html:9
+#: aleksis/core/menus.py:282 aleksis/core/models.py:446
+#: aleksis/core/templates/core/additional_field/list.html:8
+#: aleksis/core/templates/core/additional_field/list.html:9
 msgid "Additional fields"
 msgstr ""
 
-#: menus.py:233 templates/core/group/child_groups.html:7
-#: templates/core/group/child_groups.html:9
+#: aleksis/core/menus.py:297
+#: aleksis/core/templates/core/group/child_groups.html:7
+#: aleksis/core/templates/core/group/child_groups.html:9
 msgid "Assign child groups to groups"
 msgstr ""
 
-#: mixins.py:384
+#: aleksis/core/mixins.py:498
 msgid "Linked school term"
 msgstr ""
 
-#: models.py:43
+#: aleksis/core/models.py:58
 msgid "Boolean (Yes/No)"
 msgstr ""
 
-#: models.py:44
+#: aleksis/core/models.py:59
 msgid "Text (one line)"
 msgstr ""
 
-#: models.py:46
+#: aleksis/core/models.py:61
 msgid "Date and time"
 msgstr ""
 
-#: models.py:47
+#: aleksis/core/models.py:62
 msgid "Decimal number"
 msgstr ""
 
-#: models.py:48 models.py:157
+#: aleksis/core/models.py:63 aleksis/core/models.py:186
 msgid "E-mail address"
 msgstr ""
 
-#: models.py:49
+#: aleksis/core/models.py:64
 msgid "Integer"
 msgstr ""
 
-#: models.py:50
+#: aleksis/core/models.py:65
 msgid "IP address"
 msgstr ""
 
-#: models.py:51
+#: aleksis/core/models.py:66
 msgid "Boolean or empty (Yes/No/Neither)"
 msgstr ""
 
-#: models.py:52
+#: aleksis/core/models.py:67
 msgid "Text (multi-line)"
 msgstr ""
 
-#: models.py:54
+#: aleksis/core/models.py:69
 msgid "URL / Link"
 msgstr ""
 
-#: models.py:66 models.py:785
+#: aleksis/core/models.py:81 aleksis/core/models.py:927
 msgid "Name"
 msgstr ""
 
-#: models.py:68
+#: aleksis/core/models.py:83
 msgid "Start date"
 msgstr ""
 
-#: models.py:69
+#: aleksis/core/models.py:84
 msgid "End date"
 msgstr ""
 
-#: models.py:88
+#: aleksis/core/models.py:103
 msgid "The start date must be earlier than the end date."
 msgstr ""
 
-#: models.py:95
+#: aleksis/core/models.py:110
 msgid "There is already a school term for this time or a part of this time."
 msgstr ""
 
-#: models.py:115 models.py:744 templates/core/person/accounts.html:41
+#: aleksis/core/models.py:139 aleksis/core/models.py:876
 msgid "Person"
 msgstr ""
 
-#: models.py:118
+#: aleksis/core/models.py:142
 msgid "Can view address"
 msgstr ""
 
-#: models.py:119
+#: aleksis/core/models.py:143
 msgid "Can view contact details"
 msgstr ""
 
-#: models.py:120
+#: aleksis/core/models.py:144
 msgid "Can view photo"
 msgstr ""
 
-#: models.py:121
+#: aleksis/core/models.py:145
 msgid "Can view persons groups"
 msgstr ""
 
-#: models.py:122
+#: aleksis/core/models.py:146
 msgid "Can view personal details"
 msgstr ""
 
-#: models.py:127
+#: aleksis/core/models.py:156
 msgid "female"
 msgstr ""
 
-#: models.py:127
+#: aleksis/core/models.py:156
 msgid "male"
 msgstr ""
 
-#: models.py:135
+#: aleksis/core/models.py:164
 msgid "Linked user"
 msgstr ""
 
-#: models.py:137
+#: aleksis/core/models.py:166
 msgid "Is person active?"
 msgstr ""
 
-#: models.py:139
+#: aleksis/core/models.py:168
 msgid "First name"
 msgstr ""
 
-#: models.py:140
+#: aleksis/core/models.py:169
 msgid "Last name"
 msgstr ""
 
-#: models.py:142
+#: aleksis/core/models.py:171
 msgid "Additional name(s)"
 msgstr ""
 
-#: models.py:146 models.py:354
+#: aleksis/core/models.py:175 aleksis/core/models.py:415
 msgid "Short name"
 msgstr ""
 
-#: models.py:149
+#: aleksis/core/models.py:178
 msgid "Street"
 msgstr ""
 
-#: models.py:150
+#: aleksis/core/models.py:179
 msgid "Street number"
 msgstr ""
 
-#: models.py:151
+#: aleksis/core/models.py:180
 msgid "Postal code"
 msgstr ""
 
-#: models.py:152
+#: aleksis/core/models.py:181
 msgid "Place"
 msgstr ""
 
-#: models.py:154
+#: aleksis/core/models.py:183
 msgid "Home phone"
 msgstr ""
 
-#: models.py:155
+#: aleksis/core/models.py:184
 msgid "Mobile phone"
 msgstr ""
 
-#: models.py:159
+#: aleksis/core/models.py:188
 msgid "Date of birth"
 msgstr ""
 
-#: models.py:160
+#: aleksis/core/models.py:189
 msgid "Sex"
 msgstr ""
 
-#: models.py:162
+#: aleksis/core/models.py:191
 msgid "Photo"
 msgstr ""
 
-#: models.py:166 templates/core/person/full.html:137
+#: aleksis/core/models.py:195 aleksis/core/templates/core/person/full.html:137
 msgid "Guardians / Parents"
 msgstr ""
 
-#: models.py:173
+#: aleksis/core/models.py:202
 msgid "Primary group"
 msgstr ""
 
-#: models.py:176 models.py:461 models.py:485 models.py:570 models.py:805
-#: templates/core/person/full.html:120
+#: aleksis/core/models.py:205 aleksis/core/models.py:563
+#: aleksis/core/models.py:587 aleksis/core/models.py:672
+#: aleksis/core/models.py:951 aleksis/core/templates/core/person/full.html:120
 msgid "Description"
 msgstr ""
 
-#: models.py:313
+#: aleksis/core/models.py:370
 msgid "Title of field"
 msgstr ""
 
-#: models.py:315
+#: aleksis/core/models.py:372
 msgid "Type of field"
 msgstr ""
 
-#: models.py:322
+#: aleksis/core/models.py:379
 msgid "Addtitional field for groups"
 msgstr ""
 
-#: models.py:323
+#: aleksis/core/models.py:380
 msgid "Addtitional fields for groups"
 msgstr ""
 
-#: models.py:337
+#: aleksis/core/models.py:397
 msgid "Group"
 msgstr ""
 
-#: models.py:340
+#: aleksis/core/models.py:400
 msgid "Can assign child groups to groups"
 msgstr ""
 
-#: models.py:341
+#: aleksis/core/models.py:401
 msgid "Can view statistics about group."
 msgstr ""
 
-#: models.py:352
+#: aleksis/core/models.py:413
 msgid "Long name"
 msgstr ""
 
-#: models.py:362 templates/core/group/full.html:85
+#: aleksis/core/models.py:423 aleksis/core/templates/core/group/full.html:85
 msgid "Members"
 msgstr ""
 
-#: models.py:365 templates/core/group/full.html:82
+#: aleksis/core/models.py:426 aleksis/core/templates/core/group/full.html:82
 msgid "Owners"
 msgstr ""
 
-#: models.py:372 templates/core/group/full.html:55
+#: aleksis/core/models.py:433 aleksis/core/templates/core/group/full.html:55
 msgid "Parent groups"
 msgstr ""
 
-#: models.py:380
+#: aleksis/core/models.py:441
 msgid "Type of group"
 msgstr ""
 
-#: models.py:457
+#: aleksis/core/models.py:559
 msgid "User"
 msgstr ""
 
-#: models.py:460 models.py:484 models.py:569
-#: templates/core/announcement/list.html:18
+#: aleksis/core/models.py:562 aleksis/core/models.py:586
+#: aleksis/core/models.py:671
+#: aleksis/core/templates/core/announcement/list.html:18
 msgid "Title"
 msgstr ""
 
-#: models.py:463
+#: aleksis/core/models.py:565
 msgid "Application"
 msgstr ""
 
-#: models.py:469
+#: aleksis/core/models.py:571
 msgid "Activity"
 msgstr ""
 
-#: models.py:470
+#: aleksis/core/models.py:572
 msgid "Activities"
 msgstr ""
 
-#: models.py:476
+#: aleksis/core/models.py:578
 msgid "Sender"
 msgstr ""
 
-#: models.py:481
+#: aleksis/core/models.py:583
 msgid "Recipient"
 msgstr ""
 
-#: models.py:486 models.py:786
+#: aleksis/core/models.py:588 aleksis/core/models.py:928
 msgid "Link"
 msgstr ""
 
-#: models.py:488
+#: aleksis/core/models.py:590
 msgid "Read"
 msgstr ""
 
-#: models.py:489
+#: aleksis/core/models.py:591
 msgid "Sent"
 msgstr ""
 
-#: models.py:502
+#: aleksis/core/models.py:604
 msgid "Notification"
 msgstr ""
 
-#: models.py:503
-msgid "Notifications"
-msgstr ""
-
-#: models.py:571
+#: aleksis/core/models.py:673
 msgid "Link to detailed view"
 msgstr ""
 
-#: models.py:574
+#: aleksis/core/models.py:676
 msgid "Date and time from when to show"
 msgstr ""
 
-#: models.py:577
+#: aleksis/core/models.py:679
 msgid "Date and time until when to show"
 msgstr ""
 
-#: models.py:601
+#: aleksis/core/models.py:703
 msgid "Announcement"
 msgstr ""
 
-#: models.py:639
+#: aleksis/core/models.py:741
 msgid "Announcement recipient"
 msgstr ""
 
-#: models.py:640
+#: aleksis/core/models.py:742
 msgid "Announcement recipients"
 msgstr ""
 
-#: models.py:690
+#: aleksis/core/models.py:797
 msgid "Widget Title"
 msgstr ""
 
-#: models.py:691
+#: aleksis/core/models.py:798
 msgid "Activate Widget"
 msgstr ""
 
-#: models.py:694
+#: aleksis/core/models.py:799
+msgid "Widget is broken"
+msgstr ""
+
+#: aleksis/core/models.py:802
 msgid "Size on mobile devices"
 msgstr ""
 
-#: models.py:695
+#: aleksis/core/models.py:803
 msgid "<= 600 px, 12 columns"
 msgstr ""
 
-#: models.py:700
+#: aleksis/core/models.py:808
 msgid "Size on tablet devices"
 msgstr ""
 
-#: models.py:701
+#: aleksis/core/models.py:809
 msgid "> 600 px, 12 columns"
 msgstr ""
 
-#: models.py:706
+#: aleksis/core/models.py:814
 msgid "Size on desktop devices"
 msgstr ""
 
-#: models.py:707
+#: aleksis/core/models.py:815
 msgid "> 992 px, 12 columns"
 msgstr ""
 
-#: models.py:712
+#: aleksis/core/models.py:820
 msgid "Size on large desktop devices"
 msgstr ""
 
-#: models.py:713
+#: aleksis/core/models.py:821
 msgid "> 1200 px>, 12 columns"
 msgstr ""
 
-#: models.py:734
+#: aleksis/core/models.py:852
 msgid "Can edit default dashboard"
 msgstr ""
 
-#: models.py:735
+#: aleksis/core/models.py:853
 msgid "Dashboard Widget"
 msgstr ""
 
-#: models.py:736
+#: aleksis/core/models.py:854
 msgid "Dashboard Widgets"
 msgstr ""
 
-#: models.py:741
+#: aleksis/core/models.py:860
+msgid "URL"
+msgstr ""
+
+#: aleksis/core/models.py:861
+msgid "Icon URL"
+msgstr ""
+
+#: aleksis/core/models.py:867
+msgid "External link widget"
+msgstr ""
+
+#: aleksis/core/models.py:868
+msgid "External link widgets"
+msgstr ""
+
+#: aleksis/core/models.py:873
 msgid "Dashboard widget"
 msgstr ""
 
-#: models.py:746
+#: aleksis/core/models.py:878
 msgid "Order"
 msgstr ""
 
-#: models.py:747
+#: aleksis/core/models.py:879
 msgid "Part of the default dashboard"
 msgstr ""
 
-#: models.py:755
+#: aleksis/core/models.py:894
 msgid "Dashboard widget order"
 msgstr ""
 
-#: models.py:756
+#: aleksis/core/models.py:895
 msgid "Dashboard widget orders"
 msgstr ""
 
-#: models.py:762
+#: aleksis/core/models.py:901
 msgid "Menu ID"
 msgstr ""
 
-#: models.py:775
+#: aleksis/core/models.py:914
 msgid "Custom menu"
 msgstr ""
 
-#: models.py:776
+#: aleksis/core/models.py:915
 msgid "Custom menus"
 msgstr ""
 
-#: models.py:783
+#: aleksis/core/models.py:925
 msgid "Menu"
 msgstr ""
 
-#: models.py:787
+#: aleksis/core/models.py:929
 msgid "Icon"
 msgstr ""
 
-#: models.py:793
+#: aleksis/core/models.py:935
 msgid "Custom menu item"
 msgstr ""
 
-#: models.py:794
+#: aleksis/core/models.py:936
 msgid "Custom menu items"
 msgstr ""
 
-#: models.py:804
+#: aleksis/core/models.py:950
 msgid "Title of type"
 msgstr ""
 
-#: models.py:811 templates/core/group/full.html:47
+#: aleksis/core/models.py:957 aleksis/core/templates/core/group/full.html:47
 msgid "Group type"
 msgstr ""
 
-#: models.py:821
+#: aleksis/core/models.py:971
 msgid "Can view system status"
 msgstr ""
 
-#: models.py:822
-msgid "Can link persons to accounts"
-msgstr ""
-
-#: models.py:823
+#: aleksis/core/models.py:972
 msgid "Can manage data"
 msgstr ""
 
-#: models.py:824
+#: aleksis/core/models.py:973
 msgid "Can impersonate"
 msgstr ""
 
-#: models.py:825
+#: aleksis/core/models.py:974
 msgid "Can use search"
 msgstr ""
 
-#: models.py:826
+#: aleksis/core/models.py:975
 msgid "Can change site preferences"
 msgstr ""
 
-#: models.py:827
+#: aleksis/core/models.py:976
 msgid "Can change person preferences"
 msgstr ""
 
-#: models.py:828
+#: aleksis/core/models.py:977
 msgid "Can change group preferences"
 msgstr ""
 
-#: models.py:864
+#: aleksis/core/models.py:978
+msgid "Can add oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:979
+msgid "Can list oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:980
+msgid "Can view oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:981
+msgid "Can update oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:982
+msgid "Can delete oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:983
+msgid "Can test PDF generation"
+msgstr ""
+
+#: aleksis/core/models.py:1019
 msgid "Related data check task"
 msgstr ""
 
-#: models.py:872
+#: aleksis/core/models.py:1027
 msgid "Issue solved"
 msgstr ""
 
-#: models.py:873
+#: aleksis/core/models.py:1028
 msgid "Notification sent"
 msgstr ""
 
-#: models.py:886
+#: aleksis/core/models.py:1041
 msgid "Data check result"
 msgstr ""
 
-#: models.py:887
+#: aleksis/core/models.py:1042
 msgid "Data check results"
 msgstr ""
 
-#: models.py:889
+#: aleksis/core/models.py:1044
 msgid "Can run data checks"
 msgstr ""
 
-#: models.py:890
+#: aleksis/core/models.py:1045
 msgid "Can solve data check problems"
 msgstr ""
 
-#: preferences.py:27
+#: aleksis/core/models.py:1060
+msgid "Owner"
+msgstr ""
+
+#: aleksis/core/models.py:1064
+msgid "File expires at"
+msgstr ""
+
+#: aleksis/core/models.py:1066
+msgid "Generated HTML file"
+msgstr ""
+
+#: aleksis/core/models.py:1068
+msgid "Generated PDF file"
+msgstr ""
+
+#: aleksis/core/models.py:1075
+msgid "PDF file"
+msgstr ""
+
+#: aleksis/core/models.py:1076
+msgid "PDF files"
+msgstr ""
+
+#: aleksis/core/models.py:1081
+msgid "Task result"
+msgstr ""
+
+#: aleksis/core/models.py:1084
+msgid "Task user"
+msgstr ""
+
+#: aleksis/core/models.py:1096
+msgid "Task user assignment"
+msgstr ""
+
+#: aleksis/core/models.py:1097
+msgid "Task user assignments"
+msgstr ""
+
+#: aleksis/core/preferences.py:22
+msgid "General"
+msgstr ""
+
+#: aleksis/core/preferences.py:23
+msgid "School"
+msgstr ""
+
+#: aleksis/core/preferences.py:24
+msgid "Theme"
+msgstr ""
+
+#: aleksis/core/preferences.py:25
+msgid "Mail"
+msgstr ""
+
+#: aleksis/core/preferences.py:27
+msgid "Footer"
+msgstr ""
+
+#: aleksis/core/preferences.py:28
+msgid "Accounts"
+msgstr ""
+
+#: aleksis/core/preferences.py:29
 msgid "Authentication"
 msgstr ""
 
-#: preferences.py:28
+#: aleksis/core/preferences.py:30
 msgid "Internationalisation"
 msgstr ""
 
-#: preferences.py:37
+#: aleksis/core/preferences.py:41
 msgid "Site title"
 msgstr ""
 
-#: preferences.py:46
+#: aleksis/core/preferences.py:52
 msgid "Site description"
 msgstr ""
 
-#: preferences.py:55
+#: aleksis/core/preferences.py:63
 msgid "Primary colour"
 msgstr ""
 
-#: preferences.py:64
+#: aleksis/core/preferences.py:75
 msgid "Secondary colour"
 msgstr ""
 
-#: preferences.py:72
+#: aleksis/core/preferences.py:86
 msgid "Logo"
 msgstr ""
 
-#: preferences.py:80
+#: aleksis/core/preferences.py:96
 msgid "Favicon"
 msgstr ""
 
-#: preferences.py:88
+#: aleksis/core/preferences.py:106
 msgid "PWA-Icon"
 msgstr ""
 
-#: preferences.py:97
+#: aleksis/core/preferences.py:117
 msgid "Mail out name"
 msgstr ""
 
-#: preferences.py:106
+#: aleksis/core/preferences.py:128
 msgid "Mail out address"
 msgstr ""
 
-#: preferences.py:116
+#: aleksis/core/preferences.py:140
 msgid "Link to privacy policy"
 msgstr ""
 
-#: preferences.py:126
+#: aleksis/core/preferences.py:152
 msgid "Link to imprint"
 msgstr ""
 
-#: preferences.py:136
+#: aleksis/core/preferences.py:164
 msgid "Name format for addressing"
 msgstr ""
 
-#: preferences.py:150
+#: aleksis/core/preferences.py:180
 msgid "Channels to use for notifications"
 msgstr ""
 
-#: preferences.py:160
+#: aleksis/core/preferences.py:192
 msgid "Regular expression to match primary group, e.g. '^Class .*'"
 msgstr ""
 
-#: preferences.py:169
+#: aleksis/core/preferences.py:203
 msgid "Field on person to match primary group against"
 msgstr ""
 
-#: preferences.py:181
+#: aleksis/core/preferences.py:215
+msgid "Automatically create new persons for new users"
+msgstr ""
+
+#: aleksis/core/preferences.py:224
+msgid "Automatically link existing persons to new users by their e-mail address"
+msgstr ""
+
+#: aleksis/core/preferences.py:235
 msgid "Display name of the school"
 msgstr ""
 
-#: preferences.py:190
+#: aleksis/core/preferences.py:246
 msgid "Official name of the school, e.g. as given by supervisory authority"
 msgstr ""
 
-#: preferences.py:198
-msgid "Enabled custom authentication backends"
+#: aleksis/core/preferences.py:254
+msgid "Allow users to change their passwords"
+msgstr ""
+
+#: aleksis/core/preferences.py:262
+msgid "Enable signup"
 msgstr ""
 
-#: preferences.py:211
+#: aleksis/core/preferences.py:273
 msgid "Available languages"
 msgstr ""
 
-#: preferences.py:223
+#: aleksis/core/preferences.py:285
 msgid "Send emails if data checks detect problems"
 msgstr ""
 
-#: preferences.py:234
+#: aleksis/core/preferences.py:296
 msgid "Email recipients for data checks problem emails"
 msgstr ""
 
-#: preferences.py:245
+#: aleksis/core/preferences.py:307
 msgid "Email recipient groups for data checks problem emails"
 msgstr ""
 
-#: settings.py:322
-msgid "English"
+#: aleksis/core/preferences.py:316
+msgid "Show dashboard to users without login"
 msgstr ""
 
-#: settings.py:323
-msgid "German"
+#: aleksis/core/preferences.py:325
+msgid "Allow users to edit their dashboard"
 msgstr ""
 
-#: settings.py:324
-msgid "French"
+#: aleksis/core/preferences.py:336
+msgid "Fields on person model which are editable by themselves."
 msgstr ""
 
-#: settings.py:325
-msgid "Norwegian (bokmål)"
+#: aleksis/core/preferences.py:350
+msgid "Editable fields on person model which should trigger a notification on change"
 msgstr ""
 
-#: tables.py:19 templates/core/announcement/list.html:36
-#: templates/core/group/full.html:24 templates/core/person/full.html:23
+#: aleksis/core/preferences.py:363
+msgid "Contact for notification if a person changes their data"
+msgstr ""
+
+#: aleksis/core/preferences.py:373
+msgid "PDF file expiration duration"
+msgstr ""
+
+#: aleksis/core/preferences.py:374
+msgid "in minutes"
+msgstr ""
+
+#: aleksis/core/preferences.py:384
+msgid "Automatically update the dashboard and its widgets"
+msgstr ""
+
+#: aleksis/core/preferences.py:394
+msgid "Automatically update the dashboard and its widgets sitewide"
+msgstr ""
+
+#: aleksis/core/settings.py:452
+msgid "English"
+msgstr ""
+
+#: aleksis/core/settings.py:453
+msgid "German"
+msgstr ""
+
+#: aleksis/core/tables.py:19
+#: aleksis/core/templates/core/announcement/list.html:36
+#: aleksis/core/templates/core/group/full.html:24
+#: aleksis/core/templates/core/person/full.html:23
+#: aleksis/core/templates/oauth2_provider/application_detail.html:17
 msgid "Edit"
 msgstr ""
 
-#: tables.py:21 tables.py:89 templates/core/announcement/list.html:22
+#: aleksis/core/tables.py:21 aleksis/core/tables.py:89
+#: aleksis/core/templates/core/announcement/list.html:22
 msgid "Actions"
 msgstr ""
 
-#: tables.py:56 tables.py:57 tables.py:71 tables.py:87
-#: templates/core/announcement/list.html:42 templates/core/group/full.html:31
-#: templates/core/pages/delete.html:22 templates/core/person/full.html:30
+#: aleksis/core/tables.py:56 aleksis/core/tables.py:57
+#: aleksis/core/tables.py:71 aleksis/core/tables.py:87
+#: aleksis/core/templates/core/announcement/list.html:42
+#: aleksis/core/templates/core/group/full.html:31
+#: aleksis/core/templates/core/pages/delete.html:22
+#: aleksis/core/templates/core/person/full.html:30
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:20
+#: aleksis/core/templates/oauth2_provider/application_detail.html:21
 msgid "Delete"
 msgstr ""
 
-#: templates/403.html:14 templates/404.html:10 templates/500.html:10
+#: aleksis/core/templates/403.html:14 aleksis/core/templates/404.html:10
+#: aleksis/core/templates/500.html:10
+#: aleksis/core/templates/oauth2_provider/authorize.html:47
+#: aleksis/core/templates/oauth2_provider/authorized-oob.html:24
 msgid "Error"
 msgstr ""
 
-#: templates/403.html:14
+#: aleksis/core/templates/403.html:14
 msgid ""
 "You are not allowed to access the requested page or\n"
 "          object."
 msgstr ""
 
-#: templates/403.html:19 templates/404.html:17
+#: aleksis/core/templates/403.html:19 aleksis/core/templates/404.html:17
 msgid ""
 "\n"
 "            If you think this is an error in AlekSIS, please contact your site\n"
@@ -841,13 +1132,13 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/404.html:10
+#: aleksis/core/templates/404.html:10
 msgid ""
 "The requested page or object was not\n"
 "          found."
 msgstr ""
 
-#: templates/404.html:13
+#: aleksis/core/templates/404.html:13
 msgid ""
 "\n"
 "            If you were redirected by a link on an external page,\n"
@@ -855,13 +1146,13 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/500.html:10
+#: aleksis/core/templates/500.html:10
 msgid ""
 "An unexpected error has\n"
 "          occured."
 msgstr ""
 
-#: templates/500.html:13
+#: aleksis/core/templates/500.html:13
 msgid ""
 "\n"
 "            Your site administrators will automatically be notified about this\n"
@@ -869,167 +1160,394 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/503.html:10
+#: aleksis/core/templates/503.html:10
 msgid ""
 "The maintenance mode is currently enabled. Please try again\n"
 "          later."
 msgstr ""
 
-#: templates/503.html:13
+#: aleksis/core/templates/503.html:13
 msgid ""
 "\n"
 "            This page is currently unavailable. If this error persists, contact your site administrators:\n"
 "          "
 msgstr ""
 
-#: templates/core/additional_field/edit.html:6
-#: templates/core/additional_field/edit.html:7
+#: aleksis/core/templates/account/account_inactive.html:5
+#: aleksis/core/templates/account/account_inactive.html:6
+msgid "Account inactive"
+msgstr ""
+
+#: aleksis/core/templates/account/account_inactive.html:13
+msgid "Account inactive."
+msgstr ""
+
+#: aleksis/core/templates/account/account_inactive.html:15
+msgid ""
+"\n"
+"            This account is currently inactive. If you think this is an\n"
+"            error, please contact one of your site administrators.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/email/base_message.txt:5
+msgid "Hello!"
+msgstr ""
+
+#: aleksis/core/templates/account/email/base_message.txt:9
+#: aleksis/core/templates/templated_email/notification.email:22
+#: aleksis/core/templates/templated_email/notification.email:46
+msgid "Your AlekSIS team"
+msgstr ""
+
+#: aleksis/core/templates/account/email_confirm.html:5
+#: aleksis/core/templates/account/email_confirm.html:6
+#: aleksis/core/templates/account/email_confirm.html:17
+msgid "Confirm"
+msgstr ""
+
+#: aleksis/core/templates/account/email_confirm.html:12
+#, python-format
+msgid "Please confirm that <a href=\"mailto:%(email)s\">%(email)s</a> is an e-mail address for user %(user_display)s."
+msgstr ""
+
+#: aleksis/core/templates/account/email_confirm.html:25
+#, python-format
+msgid "This e-mail confirmation link expired or is invalid. Please <a href=\"%(email_url)s\">issue a new e-mail confirmation request</a>."
+msgstr ""
+
+#: aleksis/core/templates/account/password_change.html:12
+msgid "Forgot your current password? Click here to reset it:"
+msgstr ""
+
+#: aleksis/core/templates/account/password_change.html:12
+msgid "Forgot Password?"
+msgstr ""
+
+#: aleksis/core/templates/account/password_change_disabled.html:5
+#: aleksis/core/templates/account/password_change_disabled.html:6
+msgid "Changing of password disabled"
+msgstr ""
+
+#: aleksis/core/templates/account/password_change_disabled.html:13
+msgid "Changing of password disabled."
+msgstr ""
+
+#: aleksis/core/templates/account/password_change_disabled.html:15
+msgid ""
+"\n"
+"            Users are not allowed to edit their own passwords. If you think\n"
+"            this is an error please contact one of your site administrators.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset.html:5
+#: aleksis/core/templates/account/password_reset.html:15
+#: aleksis/core/templates/account/password_reset.html:23
+#: aleksis/core/templates/account/password_reset_done.html:5
+#: aleksis/core/templates/account/verification_email_required.html:5
+#: aleksis/core/templates/account/verification_email_required.html:6
+#: aleksis/core/templates/two_factor/core/login.html:81
+msgid "Reset password"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset.html:17
+msgid "Forgotten your password? Enter your e-mail address below, and we'll send you an e-mail allowing you to reset it."
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset.html:30
+msgid ""
+"Please contact one of your site administrators, if you\n"
+"        have any trouble resetting your password:"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_done.html:15
+msgid "Password reset mail sent"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_done.html:18
+#: aleksis/core/templates/account/verification_email_required.html:16
+msgid ""
+"\n"
+"            We have sent you an e-mail. Please contact one of your site\n"
+"            administrators if you do not receive it within a few minutes.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:15
+msgid "Bad token"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:19
+#, python-format
+msgid ""
+"\n"
+"              The password reset link was invalid, possibly because it has already been used. Please request a <a href=\"%(passwd_reset_url)s\"\n"
+"              class=\"blue-text text-lighten-2\">new password reset</a>.\n"
+"            "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:25
+msgid ""
+"\n"
+"              If this issue persists, please contact one of your site\n"
+"              administrators\n"
+"            "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:56
+#: aleksis/core/templates/account/password_reset_from_key_done.html:15
+msgid ""
+"\n"
+"            Your password is now changed!\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:61
+msgid "Back to login"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key_done.html:13
+msgid "Password changed!"
+msgstr ""
+
+#: aleksis/core/templates/account/password_set.html:5
+#: aleksis/core/templates/account/password_set.html:6
+#: aleksis/core/templates/account/password_set.html:12
+msgid "Set password"
+msgstr ""
+
+#: aleksis/core/templates/account/signup.html:5
+#: aleksis/core/templates/account/signup.html:6
+#: aleksis/core/templates/socialaccount/signup.html:5
+#: aleksis/core/templates/socialaccount/signup.html:6
+msgid "Signup"
+msgstr ""
+
+#: aleksis/core/templates/account/signup.html:12
+#, python-format
+msgid "Already have an account? Then please <a href=\"%(login_url)s\">sign in</a>."
+msgstr ""
+
+#: aleksis/core/templates/account/signup_closed.html:5
+#: aleksis/core/templates/account/signup_closed.html:6
+msgid "Signup closed"
+msgstr ""
+
+#: aleksis/core/templates/account/signup_closed.html:13
+msgid "Signup closed."
+msgstr ""
+
+#: aleksis/core/templates/account/signup_closed.html:15
+msgid ""
+"\n"
+"            This sign up is currently closed. If you think this is an\n"
+"            error, please contact one of your site administrators.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/verification_email_required.html:14
+msgid "Password reset mail sent!"
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:5
+#: aleksis/core/templates/account/verification_sent.html:6
+msgid "Verify your email address"
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:14
+msgid "Verify your email!"
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:16
+msgid ""
+"\n"
+"            This part of the site requires us to verify that you are who you claim to be.\n"
+"            For this purpose, we require that you verify ownership of your e-mail address.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:22
+msgid ""
+"\n"
+"            We have sent an e-mail to you for verification.\n"
+"            Please click on the link inside this e-mail. Please\n"
+"            contact us if you do not receive it within a few minutes.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:30
+#, python-format
+msgid "<strong>Note:</strong> you can still <a href=\"%(email_url)s\">change your e-mail address</a>"
+msgstr ""
+
+#: aleksis/core/templates/core/additional_field/edit.html:6
+#: aleksis/core/templates/core/additional_field/edit.html:7
 msgid "Edit additional field"
 msgstr ""
 
-#: templates/core/additional_field/list.html:14
+#: aleksis/core/templates/core/additional_field/list.html:14
 msgid "Create additional field"
 msgstr ""
 
-#: templates/core/announcement/form.html:14
-#: templates/core/announcement/form.html:21
+#: aleksis/core/templates/core/announcement/form.html:14
+#: aleksis/core/templates/core/announcement/form.html:21
 msgid "Edit announcement"
 msgstr ""
 
-#: templates/core/announcement/form.html:16
+#: aleksis/core/templates/core/announcement/form.html:16
 msgid "Publish announcement"
 msgstr ""
 
-#: templates/core/announcement/form.html:23
-#: templates/core/announcement/list.html:13
+#: aleksis/core/templates/core/announcement/form.html:23
+#: aleksis/core/templates/core/announcement/list.html:13
 msgid "Publish new announcement"
 msgstr ""
 
-#: templates/core/announcement/form.html:34
+#: aleksis/core/templates/core/announcement/form.html:34
 msgid "Save und publish announcement"
 msgstr ""
 
-#: templates/core/announcement/list.html:19
+#: aleksis/core/templates/core/announcement/list.html:19
 msgid "Valid from"
 msgstr ""
 
-#: templates/core/announcement/list.html:20
+#: aleksis/core/templates/core/announcement/list.html:20
 msgid "Valid until"
 msgstr ""
 
-#: templates/core/announcement/list.html:21
+#: aleksis/core/templates/core/announcement/list.html:21
 msgid "Recipients"
 msgstr ""
 
-#: templates/core/announcement/list.html:50
+#: aleksis/core/templates/core/announcement/list.html:50
 msgid "There are no announcements."
 msgstr ""
 
-#: templates/core/base.html:54
+#: aleksis/core/templates/core/base.html:60
 msgid "Logged in as"
 msgstr ""
 
-#: templates/core/base.html:147
+#: aleksis/core/templates/core/base.html:154
 msgid "About AlekSIS — The Free School Information System"
 msgstr ""
 
-#: templates/core/base.html:155
+#: aleksis/core/templates/core/base.html:162
 msgid "Impress"
 msgstr ""
 
-#: templates/core/base.html:163
+#: aleksis/core/templates/core/base.html:170
 msgid "Privacy Policy"
 msgstr ""
 
-#: templates/core/base_print.html:64
+#: aleksis/core/templates/core/base_print.html:72
 msgid "Powered by AlekSIS"
 msgstr ""
 
-#: templates/core/dashboard_widget/create.html:8
-#: templates/core/dashboard_widget/create.html:12
+#: aleksis/core/templates/core/dashboard_widget/create.html:8
+#: aleksis/core/templates/core/dashboard_widget/create.html:12
 #, python-format
 msgid "Create %(widget)s"
 msgstr ""
 
-#: templates/core/dashboard_widget/edit.html:8
-#: templates/core/dashboard_widget/edit.html:12
+#: aleksis/core/templates/core/dashboard_widget/dashboardwidget_broken.html:10
+msgid "This widget is currently not available."
+msgstr ""
+
+#: aleksis/core/templates/core/dashboard_widget/dashboardwidget_broken.html:14
+#, python-format
+msgid ""
+"\n"
+"            There is a problem getting the widget \"%(title)s\".\n"
+"            There is no need for you to take any action.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/core/dashboard_widget/edit.html:8
+#: aleksis/core/templates/core/dashboard_widget/edit.html:12
 #, python-format
 msgid "Edit %(widget)s"
 msgstr ""
 
-#: templates/core/dashboard_widget/list.html:17
+#: aleksis/core/templates/core/dashboard_widget/list.html:15
+msgid "Create dashboard widget"
+msgstr ""
+
+#: aleksis/core/templates/core/dashboard_widget/list.html:22
 #, python-format
 msgid "Create %(name)s"
 msgstr ""
 
-#: templates/core/dashboard_widget/list.html:25
-#: templates/core/edit_dashboard.html:8 templates/core/edit_dashboard.html:15
+#: aleksis/core/templates/core/dashboard_widget/list.html:32
+#: aleksis/core/templates/core/edit_dashboard.html:8
+#: aleksis/core/templates/core/edit_dashboard.html:15
 msgid "Edit default dashboard"
 msgstr ""
 
-#: templates/core/data_check/list.html:15
+#: aleksis/core/templates/core/data_check/list.html:15
 msgid "Check data again"
 msgstr ""
 
-#: templates/core/data_check/list.html:22
+#: aleksis/core/templates/core/data_check/list.html:22
 msgid "The system detected some problems with your data."
 msgstr ""
 
-#: templates/core/data_check/list.html:23
+#: aleksis/core/templates/core/data_check/list.html:23
 msgid ""
 "Please go through all data and check whether some extra action is\n"
 "          needed."
 msgstr ""
 
-#: templates/core/data_check/list.html:31
+#: aleksis/core/templates/core/data_check/list.html:31
 msgid "Everything is fine."
 msgstr ""
 
-#: templates/core/data_check/list.html:32
+#: aleksis/core/templates/core/data_check/list.html:32
 msgid "The system hasn't detected any problems with your data."
 msgstr ""
 
-#: templates/core/data_check/list.html:40
+#: aleksis/core/templates/core/data_check/list.html:40
 msgid "Detected problems"
 msgstr ""
 
-#: templates/core/data_check/list.html:45
+#: aleksis/core/templates/core/data_check/list.html:45
 msgid "Affected object"
 msgstr ""
 
-#: templates/core/data_check/list.html:46
+#: aleksis/core/templates/core/data_check/list.html:46
 msgid "Detected problem"
 msgstr ""
 
-#: templates/core/data_check/list.html:47
+#: aleksis/core/templates/core/data_check/list.html:47
 msgid "Show details"
 msgstr ""
 
-#: templates/core/data_check/list.html:48
+#: aleksis/core/templates/core/data_check/list.html:48
 msgid "Options to solve the problem"
 msgstr ""
 
-#: templates/core/data_check/list.html:62
+#: aleksis/core/templates/core/data_check/list.html:62
 msgid "Show object"
 msgstr ""
 
-#: templates/core/data_check/list.html:84
+#: aleksis/core/templates/core/data_check/list.html:84
 msgid "Registered checks"
 msgstr ""
 
-#: templates/core/data_check/list.html:88
+#: aleksis/core/templates/core/data_check/list.html:88
 msgid ""
 "\n"
 "            The system will check for the following problems:\n"
 "          "
 msgstr ""
 
-#: templates/core/edit_dashboard.html:6 templates/core/edit_dashboard.html:13
-#: templates/core/index.html:14
+#: aleksis/core/templates/core/edit_dashboard.html:6
+#: aleksis/core/templates/core/edit_dashboard.html:13
+#: aleksis/core/templates/core/index.html:17
 msgid "Edit dashboard"
 msgstr ""
 
-#: templates/core/edit_dashboard.html:24
+#: aleksis/core/templates/core/edit_dashboard.html:24
 msgid ""
 "\n"
 "          On this page you can arrange your personal dashboard. You can drag any items from \"Available widgets\" to \"Your\n"
@@ -1038,7 +1556,7 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/core/edit_dashboard.html:30
+#: aleksis/core/templates/core/edit_dashboard.html:30
 msgid ""
 "\n"
 "          On this page you can arrange the default dashboard which is shown when a user doesn't arrange his own\n"
@@ -1047,19 +1565,19 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/core/edit_dashboard.html:48
+#: aleksis/core/templates/core/edit_dashboard.html:48
 msgid "Available widgets"
 msgstr ""
 
-#: templates/core/edit_dashboard.html:57
+#: aleksis/core/templates/core/edit_dashboard.html:57
 msgid "Your dashboard"
 msgstr ""
 
-#: templates/core/edit_dashboard.html:59
+#: aleksis/core/templates/core/edit_dashboard.html:59
 msgid "Default dashboard"
 msgstr ""
 
-#: templates/core/group/child_groups.html:18
+#: aleksis/core/templates/core/group/child_groups.html:18
 msgid ""
 "\n"
 "          You can use this to assign child groups to groups. Please use the filters below to select groups you want to\n"
@@ -1067,38 +1585,38 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/core/group/child_groups.html:31
+#: aleksis/core/templates/core/group/child_groups.html:31
 msgid "Update selection"
 msgstr ""
 
-#: templates/core/group/child_groups.html:35
+#: aleksis/core/templates/core/group/child_groups.html:35
 msgid "Clear all filters"
 msgstr ""
 
-#: templates/core/group/child_groups.html:39
+#: aleksis/core/templates/core/group/child_groups.html:39
 msgid "Currently selected groups"
 msgstr ""
 
-#: templates/core/group/child_groups.html:52
+#: aleksis/core/templates/core/group/child_groups.html:52
 msgid "Start assigning child groups for this groups"
 msgstr ""
 
-#: templates/core/group/child_groups.html:61
+#: aleksis/core/templates/core/group/child_groups.html:61
 msgid ""
 "\n"
 "            Please select some groups in order to go on with assigning.\n"
 "          "
 msgstr ""
 
-#: templates/core/group/child_groups.html:72
+#: aleksis/core/templates/core/group/child_groups.html:72
 msgid "Current group:"
 msgstr ""
 
-#: templates/core/group/child_groups.html:78
+#: aleksis/core/templates/core/group/child_groups.html:78
 msgid "Please be careful!"
 msgstr ""
 
-#: templates/core/group/child_groups.html:79
+#: aleksis/core/templates/core/group/child_groups.html:79
 msgid ""
 "\n"
 "            If you click \"Back\" or \"Next\" the current group assignments are not saved.\n"
@@ -1107,127 +1625,135 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/core/group/child_groups.html:93
-#: templates/core/group/child_groups.html:128
-#: templates/two_factor/_wizard_actions.html:15
-#: templates/two_factor/_wizard_actions.html:20
+#: aleksis/core/templates/core/group/child_groups.html:93
+#: aleksis/core/templates/core/group/child_groups.html:128
+#: aleksis/core/templates/oauth2_provider/application_detail.html:9
+#: aleksis/core/templates/two_factor/_wizard_actions.html:15
+#: aleksis/core/templates/two_factor/_wizard_actions.html:20
 msgid "Back"
 msgstr ""
 
-#: templates/core/group/child_groups.html:99
-#: templates/core/group/child_groups.html:134
-#: templates/two_factor/_wizard_actions.html:26
+#: aleksis/core/templates/core/group/child_groups.html:99
+#: aleksis/core/templates/core/group/child_groups.html:134
+#: aleksis/core/templates/two_factor/_wizard_actions.html:26
 msgid "Next"
 msgstr ""
 
-#: templates/core/group/child_groups.html:106
-#: templates/core/group/child_groups.html:141
-#: templates/core/partials/save_button.html:3
+#: aleksis/core/templates/core/group/child_groups.html:106
+#: aleksis/core/templates/core/group/child_groups.html:141
+#: aleksis/core/templates/core/partials/save_button.html:3
 msgid "Save"
 msgstr ""
 
-#: templates/core/group/child_groups.html:112
-#: templates/core/group/child_groups.html:147
+#: aleksis/core/templates/core/group/child_groups.html:112
+#: aleksis/core/templates/core/group/child_groups.html:147
 msgid "Save and next"
 msgstr ""
 
-#: templates/core/group/edit.html:11 templates/core/group/edit.html:12
+#: aleksis/core/templates/core/group/edit.html:11
+#: aleksis/core/templates/core/group/edit.html:12
 msgid "Edit group"
 msgstr ""
 
-#: templates/core/group/full.html:38 templates/core/person/full.html:37
+#: aleksis/core/templates/core/group/full.html:38
+#: aleksis/core/templates/core/person/full.html:37
 msgid "Change preferences"
 msgstr ""
 
-#: templates/core/group/full.html:64
+#: aleksis/core/templates/core/group/full.html:64
 msgid "Statistics"
 msgstr ""
 
-#: templates/core/group/full.html:67
+#: aleksis/core/templates/core/group/full.html:67
 msgid "Count of members"
 msgstr ""
 
-#: templates/core/group/full.html:71
+#: aleksis/core/templates/core/group/full.html:71
 msgid "Average age"
 msgstr ""
 
-#: templates/core/group/full.html:76
+#: aleksis/core/templates/core/group/full.html:76
 msgid "Age range"
 msgstr ""
 
-#: templates/core/group/full.html:76
+#: aleksis/core/templates/core/group/full.html:76
 msgid "years to"
 msgstr ""
 
-#: templates/core/group/full.html:76
+#: aleksis/core/templates/core/group/full.html:76
 msgid "years "
 msgstr ""
 
-#: templates/core/group/list.html:14
+#: aleksis/core/templates/core/group/list.html:14
 msgid "Create group"
 msgstr ""
 
-#: templates/core/group/list.html:17
+#: aleksis/core/templates/core/group/list.html:17
 msgid "Filter groups"
 msgstr ""
 
-#: templates/core/group/list.html:24 templates/core/person/list.html:28
+#: aleksis/core/templates/core/group/list.html:24
+#: aleksis/core/templates/core/person/list.html:28
 msgid "Clear"
 msgstr ""
 
-#: templates/core/group/list.html:28
+#: aleksis/core/templates/core/group/list.html:28
 msgid "Selected groups"
 msgstr ""
 
-#: templates/core/group_type/edit.html:6 templates/core/group_type/edit.html:7
+#: aleksis/core/templates/core/group_type/edit.html:6
+#: aleksis/core/templates/core/group_type/edit.html:7
 msgid "Edit group type"
 msgstr ""
 
-#: templates/core/group_type/list.html:14
+#: aleksis/core/templates/core/group_type/list.html:14
 msgid "Create group type"
 msgstr ""
 
-#: templates/core/index.html:4
+#: aleksis/core/templates/core/index.html:4
 msgid "Home"
 msgstr ""
 
-#: templates/core/index.html:50
+#: aleksis/core/templates/core/index.html:49
 msgid ""
 "\n"
-"          You didn't customise your dashboard so that you see the system default. Please click on \"Edit dashboard\" to\n"
-"          customise your personal dashboard.\n"
-"        "
+"        You didn't customise your dashboard so that you see the system default. Please click on \"Edit dashboard\" to\n"
+"        customise your personal dashboard.\n"
+"      "
 msgstr ""
 
-#: templates/core/index.html:59
+#: aleksis/core/templates/core/index.html:59
 msgid "Last activities"
 msgstr ""
 
-#: templates/core/index.html:77
+#: aleksis/core/templates/core/index.html:77
 msgid "No activities available yet."
 msgstr ""
 
-#: templates/core/index.html:82
+#: aleksis/core/templates/core/index.html:82
 msgid "Recent notifications"
 msgstr ""
 
-#: templates/core/index.html:98
+#: aleksis/core/templates/core/index.html:98
+#: aleksis/core/templates/core/notifications.html:23
 msgid "More information →"
 msgstr ""
 
-#: templates/core/index.html:105
+#: aleksis/core/templates/core/index.html:105
+#: aleksis/core/templates/core/notifications.html:30
 msgid "No notifications available yet."
 msgstr ""
 
-#: templates/core/pages/about.html:6 templates/core/pages/about.html:15
+#: aleksis/core/templates/core/pages/about.html:6
+#: aleksis/core/templates/core/pages/about.html:15
 msgid "About AlekSIS"
 msgstr ""
 
-#: templates/core/pages/about.html:7
+#: aleksis/core/templates/core/pages/about.html:7
 msgid "AlekSIS – The Free School Information System"
 msgstr ""
 
-#: templates/core/pages/about.html:17
+#: aleksis/core/templates/core/pages/about.html:17
 msgid ""
 "\n"
 "              This platform is powered by AlekSIS, a web-based school information system (SIS) which can be used\n"
@@ -1236,19 +1762,19 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/pages/about.html:25
+#: aleksis/core/templates/core/pages/about.html:25
 msgid "Website of AlekSIS"
 msgstr ""
 
-#: templates/core/pages/about.html:26
+#: aleksis/core/templates/core/pages/about.html:26
 msgid "Source code"
 msgstr ""
 
-#: templates/core/pages/about.html:35
+#: aleksis/core/templates/core/pages/about.html:35
 msgid "Licence information"
 msgstr ""
 
-#: templates/core/pages/about.html:37
+#: aleksis/core/templates/core/pages/about.html:37
 msgid ""
 "\n"
 "              The core and the official apps of AlekSIS are licenced under the EUPL, version 1.2 or later. For licence\n"
@@ -1257,23 +1783,23 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/pages/about.html:45
+#: aleksis/core/templates/core/pages/about.html:45
 msgid "Free/Open Source Licence"
 msgstr ""
 
-#: templates/core/pages/about.html:46
+#: aleksis/core/templates/core/pages/about.html:46
 msgid "Other Licence"
 msgstr ""
 
-#: templates/core/pages/about.html:50
+#: aleksis/core/templates/core/pages/about.html:50
 msgid "Full licence text"
 msgstr ""
 
-#: templates/core/pages/about.html:51
+#: aleksis/core/templates/core/pages/about.html:51
 msgid "More information about the EUPL"
 msgstr ""
 
-#: templates/core/pages/about.html:90
+#: aleksis/core/templates/core/pages/about.html:90
 #, python-format
 msgid ""
 "\n"
@@ -1281,12 +1807,12 @@ msgid ""
 "                  "
 msgstr ""
 
-#: templates/core/pages/delete.html:6
+#: aleksis/core/templates/core/pages/delete.html:6
 #, python-format
 msgid "Delete %(object_name)s"
 msgstr ""
 
-#: templates/core/pages/delete.html:13
+#: aleksis/core/templates/core/pages/delete.html:13
 #, python-format
 msgid ""
 "\n"
@@ -1294,102 +1820,114 @@ msgid ""
 "    "
 msgstr ""
 
-#: templates/core/pages/progress.html:27
+#: aleksis/core/templates/core/pages/progress.html:27
 msgid ""
 "\n"
 "              Without activated JavaScript the progress status can't be updated.\n"
 "            "
 msgstr ""
 
-#: templates/core/pages/progress.html:47
-#: templates/two_factor/core/otp_required.html:19
+#: aleksis/core/templates/core/pages/progress.html:47
+#: aleksis/core/templates/two_factor/core/otp_required.html:19
 msgid "Go back"
 msgstr ""
 
-#: templates/core/pages/system_status.html:12
+#: aleksis/core/templates/core/pages/system_status.html:12
 msgid "System checks"
 msgstr ""
 
-#: templates/core/pages/system_status.html:21
+#: aleksis/core/templates/core/pages/system_status.html:21
 msgid "Maintenance mode enabled"
 msgstr ""
 
-#: templates/core/pages/system_status.html:23
+#: aleksis/core/templates/core/pages/system_status.html:23
 msgid ""
 "\n"
 "                Only admin and visitors from internal IPs can access thesite.\n"
 "              "
 msgstr ""
 
-#: templates/core/pages/system_status.html:34
+#: aleksis/core/templates/core/pages/system_status.html:34
 msgid "Maintenance mode disabled"
 msgstr ""
 
-#: templates/core/pages/system_status.html:35
+#: aleksis/core/templates/core/pages/system_status.html:35
 msgid "Everyone can access the site."
 msgstr ""
 
-#: templates/core/pages/system_status.html:45
+#: aleksis/core/templates/core/pages/system_status.html:45
 msgid "Debug mode enabled"
 msgstr ""
 
-#: templates/core/pages/system_status.html:47
+#: aleksis/core/templates/core/pages/system_status.html:47
 msgid ""
 "\n"
 "                The web server throws back debug information on errors. Do not use in production!\n"
 "              "
 msgstr ""
 
-#: templates/core/pages/system_status.html:54
+#: aleksis/core/templates/core/pages/system_status.html:54
 msgid "Debug mode disabled"
 msgstr ""
 
-#: templates/core/pages/system_status.html:56
+#: aleksis/core/templates/core/pages/system_status.html:56
 msgid ""
 "\n"
 "                Debug mode is disabled. Default error pages are displayed on errors.\n"
 "              "
 msgstr ""
 
-#: templates/core/pages/system_status.html:69
+#: aleksis/core/templates/core/pages/system_status.html:69
 msgid "System health checks"
 msgstr ""
 
-#: templates/core/pages/system_status.html:75
+#: aleksis/core/templates/core/pages/system_status.html:75
 msgid "Service"
 msgstr ""
 
-#: templates/core/pages/system_status.html:76
-#: templates/core/pages/system_status.html:115
+#: aleksis/core/templates/core/pages/system_status.html:76
+#: aleksis/core/templates/core/pages/system_status.html:115
 msgid "Status"
 msgstr ""
 
-#: templates/core/pages/system_status.html:77
+#: aleksis/core/templates/core/pages/system_status.html:77
 msgid "Time taken"
 msgstr ""
 
-#: templates/core/pages/system_status.html:96
+#: aleksis/core/templates/core/pages/system_status.html:96
 msgid "seconds"
 msgstr ""
 
-#: templates/core/pages/system_status.html:107
+#: aleksis/core/templates/core/pages/system_status.html:107
 msgid "Celery task results"
 msgstr ""
 
-#: templates/core/pages/system_status.html:112
+#: aleksis/core/templates/core/pages/system_status.html:112
 msgid "Task"
 msgstr ""
 
-#: templates/core/pages/system_status.html:113
+#: aleksis/core/templates/core/pages/system_status.html:113
 msgid "ID"
 msgstr ""
 
-#: templates/core/pages/system_status.html:114
+#: aleksis/core/templates/core/pages/system_status.html:114
 msgid "Date done"
 msgstr ""
 
-#: templates/core/partials/announcements.html:9
-#: templates/core/partials/announcements.html:36
+#: aleksis/core/templates/core/pages/test_pdf.html:7
+#: aleksis/core/templates/core/pages/test_pdf.html:8
+msgid "Test PDF generation"
+msgstr ""
+
+#: aleksis/core/templates/core/pages/test_pdf.html:14
+msgid ""
+"\n"
+"        This simple view can be used to ensure the correct function of the built-in PDF generation system.\n"
+"      "
+msgstr ""
+
+#: aleksis/core/templates/core/partials/announcements.html:8
+#: aleksis/core/templates/core/partials/announcements.html:35
 #, python-format
 msgid ""
 "\n"
@@ -1397,7 +1935,7 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/partials/announcements.html:13
+#: aleksis/core/templates/core/partials/announcements.html:12
 #, python-format
 msgid ""
 "\n"
@@ -1405,7 +1943,7 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/partials/announcements.html:40
+#: aleksis/core/templates/core/partials/announcements.html:39
 #, python-format
 msgid ""
 "\n"
@@ -1413,23 +1951,23 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/partials/crud_events.html:15
+#: aleksis/core/templates/core/partials/crud_events.html:15
 msgid "Changed by"
 msgstr ""
 
-#: templates/core/partials/crud_events.html:15
+#: aleksis/core/templates/core/partials/crud_events.html:15
 msgid "Unknown"
 msgstr ""
 
-#: templates/core/partials/language_form.html:15
+#: aleksis/core/templates/core/partials/language_form.html:15
 msgid "Language"
 msgstr ""
 
-#: templates/core/partials/language_form.html:27
+#: aleksis/core/templates/core/partials/language_form.html:27
 msgid "Select language"
 msgstr ""
 
-#: templates/core/partials/no_person.html:12
+#: aleksis/core/templates/core/partials/no_person.html:12
 msgid ""
 "\n"
 "            Your administrator account is not linked to any person. Therefore,\n"
@@ -1437,7 +1975,7 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/core/partials/no_person.html:19
+#: aleksis/core/templates/core/partials/no_person.html:19
 msgid ""
 "\n"
 "            Your user account is not linked to a person. This means you\n"
@@ -1446,105 +1984,178 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/core/person/accounts.html:12
-#: templates/core/person/accounts.html:14
-msgid "Link persons to accounts"
-msgstr ""
-
-#: templates/core/person/accounts.html:21
-msgid ""
-"\n"
-"        You can use this form to assign user accounts to persons. Use the\n"
-"        dropdowns to select existing accounts; use the text fields to create new\n"
-"        accounts on-the-fly. The latter will create a new account with the\n"
-"        entered username and copy all other details from the person.\n"
-"      "
-msgstr ""
-
-#: templates/core/person/accounts.html:36
-#: templates/core/person/accounts.html:60
-msgid "Update"
-msgstr ""
-
-#: templates/core/person/accounts.html:42
-msgid "Existing account"
-msgstr ""
-
-#: templates/core/person/accounts.html:43
-msgid "New account"
+#: aleksis/core/templates/core/person/create.html:12
+#: aleksis/core/templates/core/person/create.html:13
+#: aleksis/core/templates/core/person/list.html:17
+msgid "Create person"
 msgstr ""
 
-#: templates/core/person/edit.html:12 templates/core/person/edit.html:13
+#: aleksis/core/templates/core/person/edit.html:12
+#: aleksis/core/templates/core/person/edit.html:13
 msgid "Edit person"
 msgstr ""
 
-#: templates/core/person/full.html:44
+#: aleksis/core/templates/core/person/full.html:44
+#: aleksis/core/templates/impersonate/list_users.html:7
+#: aleksis/core/templates/impersonate/list_users.html:8
 msgid "Impersonate"
 msgstr ""
 
-#: templates/core/person/full.html:50
+#: aleksis/core/templates/core/person/full.html:50
 msgid "Contact details"
 msgstr ""
 
-#: templates/core/person/full.html:130
+#: aleksis/core/templates/core/person/full.html:130
 msgid "Children"
 msgstr ""
 
-#: templates/core/person/list.html:17
-msgid "Create person"
-msgstr ""
-
-#: templates/core/person/list.html:21
+#: aleksis/core/templates/core/person/list.html:21
 msgid "Filter persons"
 msgstr ""
 
-#: templates/core/person/list.html:32
+#: aleksis/core/templates/core/person/list.html:32
 msgid "Selected persons"
 msgstr ""
 
-#: templates/core/school_term/create.html:6
-#: templates/core/school_term/create.html:7
-#: templates/core/school_term/list.html:14
+#: aleksis/core/templates/core/school_term/create.html:6
+#: aleksis/core/templates/core/school_term/create.html:7
+#: aleksis/core/templates/core/school_term/list.html:14
 msgid "Create school term"
 msgstr ""
 
-#: templates/core/school_term/edit.html:6
-#: templates/core/school_term/edit.html:7
+#: aleksis/core/templates/core/school_term/edit.html:6
+#: aleksis/core/templates/core/school_term/edit.html:7
 msgid "Edit school term"
 msgstr ""
 
-#: templates/dynamic_preferences/form.html:9
+#: aleksis/core/templates/dynamic_preferences/form.html:9
 msgid "Site preferences"
 msgstr ""
 
-#: templates/dynamic_preferences/form.html:11
+#: aleksis/core/templates/dynamic_preferences/form.html:11
 msgid "My preferences"
 msgstr ""
 
-#: templates/dynamic_preferences/form.html:13
+#: aleksis/core/templates/dynamic_preferences/form.html:13
 #, python-format
 msgid "Preferences for %(instance)s"
 msgstr ""
 
-#: templates/dynamic_preferences/form.html:25
+#: aleksis/core/templates/dynamic_preferences/form.html:25
 msgid "Save preferences"
 msgstr ""
 
-#: templates/impersonate/list_users.html:8
-msgid "Impersonate user"
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:5
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:6
+msgid "Delete application"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:12
+#, python-format
+msgid "Are you sure to delete the application %(application_name)s?"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:24
+#: aleksis/core/templates/oauth2_provider/application_form.html:18
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:24
+#: aleksis/core/templates/two_factor/_wizard_actions.html:6
+msgid "Cancel"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:27
+msgid "Client id"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:35
+msgid "Client secret"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:43
+msgid "Client type"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:51
+msgid "Authorization Grant Type"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:59
+msgid "Redirect URIs"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_form.html:5
+msgid "Create OAuth2 Application"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_form.html:10
+msgid "Edit application"
 msgstr ""
 
-#: templates/offline.html:5
+#: aleksis/core/templates/oauth2_provider/application_list.html:8
+msgid "OAuth2 applications"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_list.html:12
+msgid "Register new applications"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_list.html:23
+msgid "No applications defined."
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorize.html:5
+#: aleksis/core/templates/oauth2_provider/authorize.html:16
+msgid "Authorize"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorize.html:18
+msgid "The application requests access to the following scopes:"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorize.html:33
+msgid "Allow"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorize.html:36
+msgid "Disallow"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-oob.html:12
+msgid "Success!"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-oob.html:14
+msgid "Please return to your application and enter this code:"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:5
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:6
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:23
+msgid "Revoke access"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:12
+msgid "Are you sure to revoke the access for this application?"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:20
+msgid "Revoke"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:33
+msgid "No authorized applications."
+msgstr ""
+
+#: aleksis/core/templates/offline.html:5
 msgid "Network error"
 msgstr ""
 
-#: templates/offline.html:8
+#: aleksis/core/templates/offline.html:8
 msgid ""
 "No internet\n"
 "    connection."
 msgstr ""
 
-#: templates/offline.html:12
+#: aleksis/core/templates/offline.html:12
 msgid ""
 "\n"
 "      There was an error accessing this page. You probably don't have an internet connection. Check to see if your WiFi\n"
@@ -1553,36 +2164,116 @@ msgid ""
 "    "
 msgstr ""
 
-#: templates/search/search.html:8
+#: aleksis/core/templates/search/search.html:8
 msgid "Global Search"
 msgstr ""
 
-#: templates/search/search.html:15
+#: aleksis/core/templates/search/search.html:15
 msgid "Search Term"
 msgstr ""
 
-#: templates/search/search.html:26
+#: aleksis/core/templates/search/search.html:26
 msgid "Results"
 msgstr ""
 
-#: templates/search/search.html:38
+#: aleksis/core/templates/search/search.html:38
 msgid "No search results could be found to your search."
 msgstr ""
 
-#: templates/search/search.html:87
+#: aleksis/core/templates/search/search.html:87
 msgid "Please enter a search term above."
 msgstr ""
 
-#: templates/templated_email/data_checks.email:4
+#: aleksis/core/templates/socialaccount/authentication_error.html:5
+#: aleksis/core/templates/socialaccount/authentication_error.html:6
+msgid "Third-party Account Login Failure"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/authentication_error.html:13
+msgid "Third-party Account Login Failure."
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/authentication_error.html:15
+msgid ""
+"\n"
+"            An error occurred while attempting to login via your third-party account.\n"
+"            Please contact one of your site administrators.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/connections.html:5
+#: aleksis/core/templates/socialaccount/connections.html:6
+msgid "Connections"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/connections.html:24
+msgid "Remove"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/connections.html:34
+msgid "You currently have no third-party accounts connected to this account."
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/connections.html:37
+msgid "Add a Third-party Account"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/login_cancelled.html:5
+#: aleksis/core/templates/socialaccount/login_cancelled.html:6
+#: aleksis/core/templates/socialaccount/login_cancelled.html:13
+msgid "Login cancelled"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/login_cancelled.html:15
+#, python-format
+msgid ""
+"\n"
+"            You decided to cancel logging in to our site using one of your existing accounts. If this was a mistake, please proceed to <a href=\"%(login_url)s\">sign in</a>.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/signup.html:12
+#, python-format
+msgid ""
+"You are about to use your %(provider_name)s account to login to\n"
+"        %(site_name)s. As a final step, please complete the following form:"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/snippets/provider_list.html:12
+#, python-format
+msgid ""
+"\n"
+"                Login with %(name)s\n"
+"              "
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/snippets/provider_list.html:21
+#, python-format
+msgid ""
+"\n"
+"            Login with %(name)s\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/snippets/provider_list.html:30
+msgid ""
+"\n"
+"          No third-party account providers available.\n"
+"        "
+msgstr ""
+
+#: aleksis/core/templates/templated_email/data_checks.email:4
 msgid "The system detected some new problems with your data."
 msgstr ""
 
-#: templates/templated_email/data_checks.email:8
-#: templates/templated_email/data_checks.email:24
+#: aleksis/core/templates/templated_email/data_checks.email:8
+#: aleksis/core/templates/templated_email/data_checks.email:24
+#: aleksis/core/templates/templated_email/person_changed.email:8
+#: aleksis/core/templates/templated_email/person_changed.email:20
 msgid "Hello,"
 msgstr ""
 
-#: templates/templated_email/data_checks.email:10
+#: aleksis/core/templates/templated_email/data_checks.email:10
 msgid ""
 "\n"
 "  the system detected some new problems with your data.\n"
@@ -1590,7 +2281,7 @@ msgid ""
 " "
 msgstr ""
 
-#: templates/templated_email/data_checks.email:26
+#: aleksis/core/templates/templated_email/data_checks.email:26
 msgid ""
 "\n"
 "   the system detected some new problems with your data.\n"
@@ -1598,35 +2289,35 @@ msgid ""
 "  "
 msgstr ""
 
-#: templates/templated_email/data_checks.email:34
+#: aleksis/core/templates/templated_email/data_checks.email:34
 msgid "Problem description"
 msgstr ""
 
-#: templates/templated_email/data_checks.email:35
+#: aleksis/core/templates/templated_email/data_checks.email:35
 msgid "Count of objects with new problems"
 msgstr ""
 
-#: templates/templated_email/notification.email:3
+#: aleksis/core/templates/templated_email/notification.email:3
 msgid "New notification for"
 msgstr ""
 
-#: templates/templated_email/notification.email:6
-#: templates/templated_email/notification.email:27
+#: aleksis/core/templates/templated_email/notification.email:6
+#: aleksis/core/templates/templated_email/notification.email:27
 #, python-format
 msgid "Dear %(notification_user)s,"
 msgstr ""
 
-#: templates/templated_email/notification.email:8
-#: templates/templated_email/notification.email:29
+#: aleksis/core/templates/templated_email/notification.email:8
+#: aleksis/core/templates/templated_email/notification.email:29
 msgid "we got a new notification for you:"
 msgstr ""
 
-#: templates/templated_email/notification.email:15
-#: templates/templated_email/notification.email:35
+#: aleksis/core/templates/templated_email/notification.email:15
+#: aleksis/core/templates/templated_email/notification.email:35
 msgid "More information"
 msgstr ""
 
-#: templates/templated_email/notification.email:18
+#: aleksis/core/templates/templated_email/notification.email:18
 #, python-format
 msgid ""
 "\n"
@@ -1634,12 +2325,7 @@ msgid ""
 "    "
 msgstr ""
 
-#: templates/templated_email/notification.email:22
-#: templates/templated_email/notification.email:46
-msgid "Your AlekSIS team"
-msgstr ""
-
-#: templates/templated_email/notification.email:40
+#: aleksis/core/templates/templated_email/notification.email:40
 #, python-format
 msgid ""
 "\n"
@@ -1647,24 +2333,41 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/two_factor/_base_focus.html:6
-#: templates/two_factor/core/otp_required.html:22
-#: templates/two_factor/core/setup.html:5
-#: templates/two_factor/profile/profile.html:87
-msgid "Enable Two-Factor Authentication"
+#: aleksis/core/templates/templated_email/person_changed.email:4
+#, python-format
+msgid "%(person)s changed their data!"
 msgstr ""
 
-#: templates/two_factor/_wizard_actions.html:6
-msgid "Cancel"
+#: aleksis/core/templates/templated_email/person_changed.email:10
+#, python-format
+msgid ""
+"\n"
+"   the person %(person)s recently changed the following fields:\n"
+" "
+msgstr ""
+
+#: aleksis/core/templates/templated_email/person_changed.email:22
+#, python-format
+msgid ""
+"\n"
+"    the person %(person)s recently changed the following fields:\n"
+"  "
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:5
-#: templates/two_factor/core/backup_tokens.html:9
-#: templates/two_factor/profile/profile.html:46
+#: aleksis/core/templates/two_factor/_base_focus.html:6
+#: aleksis/core/templates/two_factor/core/otp_required.html:22
+#: aleksis/core/templates/two_factor/core/setup.html:5
+#: aleksis/core/templates/two_factor/profile/profile.html:87
+msgid "Enable Two-Factor Authentication"
+msgstr ""
+
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:5
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:9
+#: aleksis/core/templates/two_factor/profile/profile.html:46
 msgid "Backup Tokens"
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:14
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:14
 msgid ""
 "\n"
 "        Backup tokens can be used when your primary and backup\n"
@@ -1675,115 +2378,129 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:33
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:33
 msgid ""
 "\n"
 "          Print these tokens and keep them somewhere safe.\n"
 "        "
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:39
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:39
 msgid "You don't have any backup codes yet."
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:45
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:45
 msgid "Back to Account Security"
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:49
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:49
 msgid "Generate Tokens"
 msgstr ""
 
-#: templates/two_factor/core/login.html:16
-msgid "You have no permission to view this page. Please login with an other account."
+#: aleksis/core/templates/two_factor/core/login.html:20
+msgid "Login with username and password"
 msgstr ""
 
-#: templates/two_factor/core/login.html:24
-msgid "Please login to see this page."
+#: aleksis/core/templates/two_factor/core/login.html:28
+msgid ""
+"You have no permission to view this page. Please login with an other\n"
+"                    account."
 msgstr ""
 
-#: templates/two_factor/core/login.html:30
-msgid "Login with username and password"
+#: aleksis/core/templates/two_factor/core/login.html:36
+msgid "Please login to see this page."
 msgstr ""
 
-#: templates/two_factor/core/login.html:40
+#: aleksis/core/templates/two_factor/core/login.html:46
 msgid ""
-"We are calling your phone right now, please enter the\n"
-"              digits you hear."
+"\n"
+"                        We are calling your phone right now, please enter the\n"
+"                        digits you hear.\n"
+"                      "
 msgstr ""
 
-#: templates/two_factor/core/login.html:43
+#: aleksis/core/templates/two_factor/core/login.html:51
 msgid ""
-"We sent you a text message, please enter the tokens we\n"
-"              sent."
+"\n"
+"                        We sent you a text message, please enter the tokens we\n"
+"                        sent.\n"
+"                      "
 msgstr ""
 
-#: templates/two_factor/core/login.html:46
+#: aleksis/core/templates/two_factor/core/login.html:56
 msgid ""
-"Please enter the tokens generated by your token\n"
-"              generator."
+"\n"
+"                        Please enter the tokens generated by your token\n"
+"                        generator.\n"
+"                      "
 msgstr ""
 
-#: templates/two_factor/core/login.html:50
+#: aleksis/core/templates/two_factor/core/login.html:62
 msgid ""
-"Use this form for entering backup tokens for logging in.\n"
-"            These tokens have been generated for you to print and keep safe. Please\n"
-"            enter one of these backup tokens to login to your account."
+"\n"
+"                      Use this form for entering backup tokens for logging in.\n"
+"                      These tokens have been generated for you to print and keep safe. Please\n"
+"                      enter one of these backup tokens to login to your account.\n"
+"                    "
 msgstr ""
 
-#: templates/two_factor/core/login.html:68
+#: aleksis/core/templates/two_factor/core/login.html:90
+msgid "Device currently not available?"
+msgstr ""
+
+#: aleksis/core/templates/two_factor/core/login.html:92
 msgid "Or, alternatively, use one of your backup phones:"
 msgstr ""
 
-#: templates/two_factor/core/login.html:78
+#: aleksis/core/templates/two_factor/core/login.html:102
 msgid "As a last resort, you can use a backup token:"
 msgstr ""
 
-#: templates/two_factor/core/login.html:81
+#: aleksis/core/templates/two_factor/core/login.html:105
 msgid "Use Backup Token"
 msgstr ""
 
-#: templates/two_factor/core/login.html:93
+#: aleksis/core/templates/two_factor/core/login.html:116
 msgid "Use alternative login options"
 msgstr ""
 
-#: templates/two_factor/core/otp_required.html:9
+#: aleksis/core/templates/two_factor/core/otp_required.html:9
 msgid "Permission Denied"
 msgstr ""
 
-#: templates/two_factor/core/otp_required.html:10
+#: aleksis/core/templates/two_factor/core/otp_required.html:10
 msgid ""
 "The page you requested, enforces users to verify using\n"
 "          two-factor authentication for security reasons. You need to enable these\n"
 "          security features in order to access this page."
 msgstr ""
 
-#: templates/two_factor/core/otp_required.html:14
+#: aleksis/core/templates/two_factor/core/otp_required.html:14
 msgid ""
 "Two-factor authentication is not enabled for your\n"
 "          account. Enable two-factor authentication for enhanced account\n"
 "          security."
 msgstr ""
 
-#: templates/two_factor/core/phone_register.html:5
-#: templates/two_factor/core/phone_register.html:9
+#: aleksis/core/templates/two_factor/core/phone_register.html:5
+#: aleksis/core/templates/two_factor/core/phone_register.html:9
 msgid "Add Backup Phone"
 msgstr ""
 
-#: templates/two_factor/core/phone_register.html:12
+#: aleksis/core/templates/two_factor/core/phone_register.html:12
 msgid ""
 "You'll be adding a backup phone number to your\n"
 "      account. This number will be used if your primary method of\n"
 "      registration is not available."
 msgstr ""
 
-#: templates/two_factor/core/phone_register.html:16
+#: aleksis/core/templates/two_factor/core/phone_register.html:16
 msgid ""
 "We've sent a token to your phone number. Please\n"
 "      enter the token you've received."
 msgstr ""
 
-#: templates/two_factor/core/setup.html:9
+#: aleksis/core/templates/two_factor/core/setup.html:9
 msgid ""
 "\n"
 "        You are about to take your account security to the\n"
@@ -1792,14 +2509,14 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:17
+#: aleksis/core/templates/two_factor/core/setup.html:17
 msgid ""
 "\n"
 "        Please select which authentication method you would like to use:\n"
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:23
+#: aleksis/core/templates/two_factor/core/setup.html:23
 msgid ""
 "\n"
 "        To start using a token generator, please use your\n"
@@ -1808,7 +2525,7 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:34
+#: aleksis/core/templates/two_factor/core/setup.html:34
 msgid ""
 "\n"
 "        Please enter the phone number you wish to receive the\n"
@@ -1816,7 +2533,7 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:41
+#: aleksis/core/templates/two_factor/core/setup.html:41
 msgid ""
 "\n"
 "        Please enter the phone number you wish to be called on.\n"
@@ -1824,21 +2541,21 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:50
+#: aleksis/core/templates/two_factor/core/setup.html:50
 msgid ""
 "\n"
 "            We are calling your phone right now, please enter the digits you hear.\n"
 "          "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:56
+#: aleksis/core/templates/two_factor/core/setup.html:56
 msgid ""
 "\n"
 "            We sent you a text message, please enter the tokens we sent.\n"
 "          "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:63
+#: aleksis/core/templates/two_factor/core/setup.html:63
 msgid ""
 "\n"
 "          We've encountered an issue with the selected authentication method. Please\n"
@@ -1848,7 +2565,7 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:73
+#: aleksis/core/templates/two_factor/core/setup.html:73
 msgid ""
 "\n"
 "        To identify and verify your YubiKey, please insert a\n"
@@ -1857,29 +2574,29 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:5
-#: templates/two_factor/core/setup_complete.html:9
+#: aleksis/core/templates/two_factor/core/setup_complete.html:5
+#: aleksis/core/templates/two_factor/core/setup_complete.html:9
 msgid "Two-Factor Authentication successfully enabled"
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:14
+#: aleksis/core/templates/two_factor/core/setup_complete.html:14
 msgid ""
 "\n"
 "        Congratulations, you've successfully enabled two-factor authentication.\n"
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:24
-#: templates/two_factor/core/setup_complete.html:44
+#: aleksis/core/templates/two_factor/core/setup_complete.html:24
+#: aleksis/core/templates/two_factor/core/setup_complete.html:44
 msgid "Back to Profile"
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:28
-#: templates/two_factor/core/setup_complete.html:48
+#: aleksis/core/templates/two_factor/core/setup_complete.html:28
+#: aleksis/core/templates/two_factor/core/setup_complete.html:48
 msgid "Generate backup codes"
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:34
+#: aleksis/core/templates/two_factor/core/setup_complete.html:34
 msgid ""
 "\n"
 "          However, it might happen that you don't have access to\n"
@@ -1888,65 +2605,65 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:52
-#: templates/two_factor/profile/profile.html:41
+#: aleksis/core/templates/two_factor/core/setup_complete.html:52
+#: aleksis/core/templates/two_factor/profile/profile.html:41
 msgid "Add Phone Number"
 msgstr ""
 
-#: templates/two_factor/profile/disable.html:5
-#: templates/two_factor/profile/disable.html:9
-#: templates/two_factor/profile/profile.html:63
-#: templates/two_factor/profile/profile.html:73
+#: aleksis/core/templates/two_factor/profile/disable.html:5
+#: aleksis/core/templates/two_factor/profile/disable.html:9
+#: aleksis/core/templates/two_factor/profile/profile.html:63
+#: aleksis/core/templates/two_factor/profile/profile.html:73
 msgid "Disable Two-Factor Authentication"
 msgstr ""
 
-#: templates/two_factor/profile/disable.html:12
+#: aleksis/core/templates/two_factor/profile/disable.html:12
 msgid "You are about to disable two-factor authentication. This weakens your account security, are you sure?"
 msgstr ""
 
-#: templates/two_factor/profile/disable.html:26
+#: aleksis/core/templates/two_factor/profile/disable.html:26
 msgid "Disable"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:5
-#: templates/two_factor/profile/profile.html:10
+#: aleksis/core/templates/two_factor/profile/profile.html:5
+#: aleksis/core/templates/two_factor/profile/profile.html:10
 msgid "Account Security"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:15
+#: aleksis/core/templates/two_factor/profile/profile.html:15
 msgid "Tokens will be generated by your token generator."
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:17
+#: aleksis/core/templates/two_factor/profile/profile.html:17
 #, python-format
 msgid "Primary method: %(primary)s"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:19
+#: aleksis/core/templates/two_factor/profile/profile.html:19
 msgid "Tokens will be generated by your YubiKey."
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:23
+#: aleksis/core/templates/two_factor/profile/profile.html:23
 msgid "Backup Phone Numbers"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:24
+#: aleksis/core/templates/two_factor/profile/profile.html:24
 msgid ""
 "If your primary method is not available, we are able to\n"
 "        send backup tokens to the phone numbers listed below."
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:33
+#: aleksis/core/templates/two_factor/profile/profile.html:33
 msgid "Unregister"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:48
+#: aleksis/core/templates/two_factor/profile/profile.html:48
 msgid ""
 "If you don't have any device with you, you can access\n"
 "        your account using backup tokens."
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:50
+#: aleksis/core/templates/two_factor/profile/profile.html:50
 #, python-format
 msgid ""
 "\n"
@@ -1963,11 +2680,11 @@ msgstr[3] ""
 msgstr[4] ""
 msgstr[5] ""
 
-#: templates/two_factor/profile/profile.html:59
+#: aleksis/core/templates/two_factor/profile/profile.html:59
 msgid "Show Codes"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:65
+#: aleksis/core/templates/two_factor/profile/profile.html:65
 msgid ""
 "\n"
 "        However we strongly discourage you to do so, you can\n"
@@ -1975,7 +2692,7 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:78
+#: aleksis/core/templates/two_factor/profile/profile.html:78
 msgid ""
 "\n"
 "        Two-factor authentication is not enabled for your\n"
@@ -1984,99 +2701,135 @@ msgid ""
 "      "
 msgstr ""
 
-#: util/notifications.py:65
+#: aleksis/core/util/notifications.py:65
 msgid "E-Mail"
 msgstr ""
 
-#: util/notifications.py:66
+#: aleksis/core/util/notifications.py:66
 msgid "SMS"
 msgstr ""
 
-#: views.py:141
+#: aleksis/core/util/pdf.py:105
+msgid "Progress: Generate PDF file"
+msgstr ""
+
+#: aleksis/core/util/pdf.py:106
+msgid "Generating PDF file …"
+msgstr ""
+
+#: aleksis/core/util/pdf.py:107
+msgid "The PDF file has been generated successfully."
+msgstr ""
+
+#: aleksis/core/util/pdf.py:108
+msgid "There was a problem while generating the PDF file."
+msgstr ""
+
+#: aleksis/core/util/pdf.py:111
+msgid "Download PDF"
+msgstr ""
+
+#: aleksis/core/views.py:251
 msgid "The school term has been created."
 msgstr ""
 
-#: views.py:153
+#: aleksis/core/views.py:263
 msgid "The school term has been saved."
 msgstr ""
 
-#: views.py:298
+#: aleksis/core/views.py:387
 msgid "The child groups were successfully saved."
 msgstr ""
 
-#: views.py:336
+#: aleksis/core/views.py:406 aleksis/core/views.py:416
 msgid "The person has been saved."
 msgstr ""
 
-#: views.py:375
+#: aleksis/core/views.py:466
 msgid "The group has been saved."
 msgstr ""
 
-#: views.py:467
+#: aleksis/core/views.py:563
 msgid "The announcement has been saved."
 msgstr ""
 
-#: views.py:483
+#: aleksis/core/views.py:579
 msgid "The announcement has been deleted."
 msgstr ""
 
-#: views.py:562
+#: aleksis/core/views.py:663
 msgid "The preferences have been saved successfully."
 msgstr ""
 
-#: views.py:586
+#: aleksis/core/views.py:687
 msgid "The person has been deleted."
 msgstr ""
 
-#: views.py:600
+#: aleksis/core/views.py:701
 msgid "The group has been deleted."
 msgstr ""
 
-#: views.py:632
+#: aleksis/core/views.py:733
 msgid "The additional_field has been saved."
 msgstr ""
 
-#: views.py:666
+#: aleksis/core/views.py:767
 msgid "The additional field has been deleted."
 msgstr ""
 
-#: views.py:691
+#: aleksis/core/views.py:792
 msgid "The group type has been saved."
 msgstr ""
 
-#: views.py:721
+#: aleksis/core/views.py:822
 msgid "The group type has been deleted."
 msgstr ""
 
-#: views.py:749
-msgid "The data check has been started. Please note that it may take a while before you are able to fetch the data on this page."
+#: aleksis/core/views.py:855
+msgid "Progress: Run data checks"
+msgstr ""
+
+#: aleksis/core/views.py:856
+msgid "Run data checks …"
 msgstr ""
 
-#: views.py:754
-msgid "The data check has finished."
+#: aleksis/core/views.py:857
+msgid "The data checks were run successfully."
 msgstr ""
 
-#: views.py:769
+#: aleksis/core/views.py:858
+msgid "There was a problem while running data checks."
+msgstr ""
+
+#: aleksis/core/views.py:874
 #, python-brace-format
 msgid "The solve option '{solve_option_obj.verbose_name}' "
 msgstr ""
 
-#: views.py:811
+#: aleksis/core/views.py:916
 msgid "The dashboard widget has been saved."
 msgstr ""
 
-#: views.py:841
+#: aleksis/core/views.py:946
 msgid "The dashboard widget has been created."
 msgstr ""
 
-#: views.py:851
+#: aleksis/core/views.py:956
 msgid "The dashboard widget has been deleted."
 msgstr ""
 
-#: views.py:914
+#: aleksis/core/views.py:1023
 msgid "Your dashboard configuration has been saved successfully."
 msgstr ""
 
-#: views.py:916
+#: aleksis/core/views.py:1025
 msgid "The configuration of the default dashboard has been saved successfully."
 msgstr ""
+
+#: aleksis/core/views.py:1153
+msgid "The third-party account could not be disconnected because it is the only login method available."
+msgstr ""
+
+#: aleksis/core/views.py:1160
+msgid "The third-party account has been successfully disconnected."
+msgstr ""
diff --git a/aleksis/core/locale/ar/LC_MESSAGES/djangojs.po b/aleksis/core/locale/ar/LC_MESSAGES/djangojs.po
index 126b0ef841fd5c5732a533ae0604af89fcfb093b..58d7ae38a2326acb79508cd05c32cab673c58fe9 100644
--- a/aleksis/core/locale/ar/LC_MESSAGES/djangojs.po
+++ b/aleksis/core/locale/ar/LC_MESSAGES/djangojs.po
@@ -8,7 +8,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2021-01-11 21:30+0100\n"
+"POT-Creation-Date: 2021-10-28 16:18+0200\n"
 "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
@@ -18,18 +18,18 @@ msgstr ""
 "Content-Transfer-Encoding: 8bit\n"
 "Plural-Forms: nplurals=6; plural=n==0 ? 0 : n==1 ? 1 : n==2 ? 2 : n%100>=3 && n%100<=10 ? 3 : n%100>=11 && n%100<=99 ? 4 : 5;\n"
 
-#: static/js/main.js:15
+#: aleksis/core/static/js/main.js:15
 msgid "Today"
 msgstr ""
 
-#: static/js/main.js:16
+#: aleksis/core/static/js/main.js:16
 msgid "Cancel"
 msgstr ""
 
-#: static/js/main.js:17
+#: aleksis/core/static/js/main.js:17
 msgid "OK"
 msgstr ""
 
-#: static/js/main.js:118
+#: aleksis/core/static/js/main.js:127
 msgid "This page may contain outdated information since there is no internet connection."
 msgstr ""
diff --git a/aleksis/core/locale/de_DE/LC_MESSAGES/django.po b/aleksis/core/locale/de_DE/LC_MESSAGES/django.po
index d8a3b4e5a53a776cd9fb0433267539bde11efe99..f392000d5c400048743b02d15cfca5dbb4efd644 100644
--- a/aleksis/core/locale/de_DE/LC_MESSAGES/django.po
+++ b/aleksis/core/locale/de_DE/LC_MESSAGES/django.po
@@ -7,828 +7,1119 @@ msgid ""
 msgstr ""
 "Project-Id-Version: AlekSIS (School Information System) 0.1\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2021-01-11 21:30+0100\n"
-"PO-Revision-Date: 2021-03-14 11:51+0000\n"
+"POT-Creation-Date: 2021-10-28 16:18+0200\n"
+"PO-Revision-Date: 2021-10-29 14:28+0000\n"
 "Last-Translator: Jonathan Weth <teckids@jonathanweth.de>\n"
-"Language-Team: German <https://translate.edugit.org/projects/aleksis/aleksis/"
-"de/>\n"
+"Language-Team: German <https://translate.edugit.org/projects/aleksis/"
+"aleksis-core/de/>\n"
 "Language: de_DE\n"
 "MIME-Version: 1.0\n"
 "Content-Type: text/plain; charset=UTF-8\n"
 "Content-Transfer-Encoding: 8bit\n"
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
-"X-Generator: Weblate 4.4\n"
+"X-Generator: Weblate 4.8\n"
 
-#: data_checks.py:53
+#: aleksis/core/apps.py:150
+msgid "OpenID Connect scope"
+msgstr "OpenID-Connect-Scope"
+
+#: aleksis/core/apps.py:151
+msgid "Given name, family name, link to profile and picture if existing."
+msgstr "Vorname, Nachname, Link zum Profil und Bild falls vorhanden"
+
+#: aleksis/core/apps.py:152
+msgid "Full home postal address"
+msgstr "Vollständige Postanschrift"
+
+#: aleksis/core/apps.py:153
+msgid "Email address"
+msgstr "E-Mail-Adresse"
+
+#: aleksis/core/apps.py:154
+msgid "Home and mobile phone"
+msgstr "Festnetz- und Mobilfunknummer"
+
+#: aleksis/core/data_checks.py:55
 msgid "Ignore problem"
 msgstr "Problem ignorieren"
 
-#: data_checks.py:174
+#: aleksis/core/data_checks.py:184
 #, python-brace-format
 msgid "Solve option '{solve_option_obj.verbose_name}' "
 msgstr "Lösungsoption \"{solve_option_obj.verbose_name}\" "
 
-#: filters.py:37 templates/core/base.html:77 templates/core/group/list.html:20
-#: templates/core/person/list.html:24 templates/search/search.html:7
-#: templates/search/search.html:22
+#: aleksis/core/data_checks.py:291
+msgid "Deactivate DashboardWidget"
+msgstr "Dashboard-Widget deaktivieren"
+
+#: aleksis/core/data_checks.py:303
+msgid "Ensure that there are no broken DashboardWidgets."
+msgstr "Sicherstellen, dass es keine kaputten Dashboard-Widgets gibt."
+
+#: aleksis/core/data_checks.py:304
+msgid "The DashboardWidget was reported broken automatically."
+msgstr "Das Dashboard-Widget wurde automatisch als kaputt gemeldet."
+
+#: aleksis/core/filters.py:37 aleksis/core/templates/core/base.html:83
+#: aleksis/core/templates/core/base.html:84
+#: aleksis/core/templates/core/group/list.html:20
+#: aleksis/core/templates/core/person/list.html:24
+#: aleksis/core/templates/search/search.html:7
+#: aleksis/core/templates/search/search.html:22
 msgid "Search"
 msgstr "Suchen"
 
-#: filters.py:53
+#: aleksis/core/filters.py:53
 msgid "Search by name"
 msgstr "Nach Namen suchen"
 
-#: filters.py:65
+#: aleksis/core/filters.py:65
 msgid "Search by contact details"
 msgstr "Nach Kontaktdetails suchen"
 
-#: forms.py:54
-msgid "You cannot set a new username when also selecting an existing user."
-msgstr "Sie können keine neuen Benutzer erstellen, wenn Sie gleichzeitig einen existierenden Benutzer auswählen."
-
-#: forms.py:58
-msgid "This username is already in use."
-msgstr "Dieser Benutzername wird bereits genutzt."
-
-#: forms.py:82
+#: aleksis/core/forms.py:41 aleksis/core/forms.py:387
 msgid "Base data"
 msgstr "Basisdaten"
 
-#: forms.py:88
+#: aleksis/core/forms.py:47
 msgid "Address"
 msgstr "Adresse"
 
-#: forms.py:89
+#: aleksis/core/forms.py:48
 msgid "Contact data"
 msgstr "Kontaktdaten"
 
-#: forms.py:91
+#: aleksis/core/forms.py:50
 msgid "Advanced personal data"
 msgstr "Zusätzliche persönliche Daten"
 
-#: forms.py:134
+#: aleksis/core/forms.py:93
 msgid "New user"
 msgstr "Neuer Benutzer"
 
-#: forms.py:134
+#: aleksis/core/forms.py:93
 msgid "Create a new account"
 msgstr "Neues Benutzerkonto erstellen"
 
-#: forms.py:146 models.py:102
+#: aleksis/core/forms.py:124
+msgid "You cannot set a new username when also selecting an existing user."
+msgstr "Sie können keine neuen Benutzer erstellen, wenn Sie gleichzeitig einen existierenden Benutzer auswählen."
+
+#: aleksis/core/forms.py:128
+msgid "This username is already in use."
+msgstr "Dieser Benutzername wird bereits genutzt."
+
+#: aleksis/core/forms.py:145 aleksis/core/models.py:117
 msgid "School term"
 msgstr "Schuljahr"
 
-#: forms.py:147
+#: aleksis/core/forms.py:146
 msgid "Common data"
 msgstr "Allgemeine Daten"
 
-#: forms.py:148 forms.py:197 menus.py:169 models.py:116
-#: templates/core/person/list.html:8 templates/core/person/list.html:9
+#: aleksis/core/forms.py:147 aleksis/core/forms.py:196
+#: aleksis/core/menus.py:238 aleksis/core/models.py:140
+#: aleksis/core/templates/core/person/list.html:8
+#: aleksis/core/templates/core/person/list.html:9
 msgid "Persons"
 msgstr "Personen"
 
-#: forms.py:149
+#: aleksis/core/forms.py:148
 msgid "Additional data"
 msgstr "Zusätzliche Datne"
 
-#: forms.py:189 forms.py:192 models.py:45
+#: aleksis/core/forms.py:188 aleksis/core/forms.py:191
+#: aleksis/core/models.py:60
 msgid "Date"
 msgstr "Datum"
 
-#: forms.py:190 forms.py:193 models.py:53
+#: aleksis/core/forms.py:189 aleksis/core/forms.py:192
+#: aleksis/core/models.py:68
 msgid "Time"
 msgstr "Zeit"
 
-#: forms.py:210 menus.py:177 models.py:338 templates/core/group/list.html:8
-#: templates/core/group/list.html:9 templates/core/person/full.html:144
+#: aleksis/core/forms.py:209 aleksis/core/menus.py:249
+#: aleksis/core/models.py:398 aleksis/core/templates/core/group/list.html:8
+#: aleksis/core/templates/core/group/list.html:9
+#: aleksis/core/templates/core/person/full.html:144
 msgid "Groups"
 msgstr "Gruppen"
 
-#: forms.py:220
+#: aleksis/core/forms.py:219
 msgid "From when until when should the announcement be displayed?"
 msgstr "Von wann bis wann soll die Ankündigung angezeigt werden?"
 
-#: forms.py:223
+#: aleksis/core/forms.py:222
 msgid "Who should see the announcement?"
 msgstr "Wer soll die Ankündigung sehen?"
 
-#: forms.py:224
+#: aleksis/core/forms.py:223
 msgid "Write your announcement:"
 msgstr "Schreiben Sie ihre Ankündigung:"
 
-#: forms.py:263
+#: aleksis/core/forms.py:262
 msgid "You are not allowed to create announcements which are only valid in the past."
 msgstr "Sie dürfen keine Ankündigungen erstellen, die nur für die Vergangenheit gültig sind."
 
-#: forms.py:267
+#: aleksis/core/forms.py:266
 msgid "The from date and time must be earlier then the until date and time."
 msgstr "Das Startdatum und die Startzeit müssen vor dem Enddatum und der Endzeit sein."
 
-#: forms.py:276
+#: aleksis/core/forms.py:275
 msgid "You need at least one recipient."
 msgstr "Sie benötigen mindestens einen Empfänger."
 
-#: health_checks.py:15
+#: aleksis/core/forms.py:389
+msgid "Account data"
+msgstr "Kontodaten"
+
+#: aleksis/core/forms.py:391
+msgid "Consents"
+msgstr "Zustimmungen"
+
+#: aleksis/core/forms.py:396
+msgid "Password"
+msgstr "Passwort"
+
+#: aleksis/core/forms.py:402
+msgid "Password (again)"
+msgstr "Passwort wiederholen"
+
+#: aleksis/core/forms.py:411
+#, python-brace-format
+msgid "I have read the <a href='{privacy_policy}'>Privacy policy</a> and agree with them."
+msgstr "Ich habe die <a href='{privacy_policy}'>Datenschutzerklärung</a> gelesen und stimme ihr zu."
+
+#: aleksis/core/forms.py:435
+msgid "You must type the same password each time."
+msgstr "Sie müssen zweimal das gleiche Passwort eingeben."
+
+#: aleksis/core/forms.py:580
+msgid "No valid selection."
+msgstr "Keine gültige Auswahl."
+
+#: aleksis/core/health_checks.py:21
 msgid "There are unresolved data problems."
 msgstr "Es gibt ungelöste Datenprobleme."
 
-#: menus.py:7 templates/two_factor/core/login.html:6
-#: templates/two_factor/core/login.html:10
-#: templates/two_factor/core/login.html:86
+#: aleksis/core/health_checks.py:38
+msgid "The backup folder doesn't exist."
+msgstr "Der Backup-Ordner existiert nicht."
+
+#: aleksis/core/health_checks.py:47
+#, python-brace-format
+msgid "Last backup {time_gone_since_backup}!"
+msgstr "Letztes Backup: {time_gone_since_backup}!"
+
+#: aleksis/core/health_checks.py:49
+msgid "No backup found!"
+msgstr "Kein Backup gefunden!"
+
+#: aleksis/core/health_checks.py:76
+msgid "No backup result found!"
+msgstr "Kein Backupergebnis gefunden!"
+
+#: aleksis/core/health_checks.py:78
+#, python-brace-format
+msgid "{task.status} - {task.result}"
+msgstr "{task.status} - {task.result}"
+
+#: aleksis/core/menus.py:9 aleksis/core/templates/two_factor/core/login.html:6
+#: aleksis/core/templates/two_factor/core/login.html:22
+#: aleksis/core/templates/two_factor/core/login.html:76
 msgid "Login"
 msgstr "Anmelden"
 
-#: menus.py:13
+#: aleksis/core/menus.py:15 aleksis/core/templates/account/signup.html:20
+#: aleksis/core/templates/socialaccount/signup.html:23
+msgid "Sign up"
+msgstr "Registrieren"
+
+#: aleksis/core/menus.py:24
 msgid "Dashboard"
 msgstr "Dashboard"
 
-#: menus.py:19
+#: aleksis/core/menus.py:32 aleksis/core/models.py:605
+#: aleksis/core/preferences.py:26
+#: aleksis/core/templates/core/notifications.html:4
+#: aleksis/core/templates/core/notifications.html:5
+msgid "Notifications"
+msgstr "Benachrichtigungen"
+
+#: aleksis/core/menus.py:41
 msgid "Account"
 msgstr "Konto"
 
-#: menus.py:26
+#: aleksis/core/menus.py:48
 msgid "Stop impersonation"
 msgstr "Verkleidung beenden"
 
-#: menus.py:35 templates/core/base.html:56
+#: aleksis/core/menus.py:57 aleksis/core/templates/core/base.html:62
 msgid "Logout"
 msgstr "Abmelden"
 
-#: menus.py:41
+#: aleksis/core/menus.py:63
 msgid "2FA"
 msgstr "2FA"
 
-#: menus.py:47
+#: aleksis/core/menus.py:69
+#: aleksis/core/templates/account/password_change.html:5
+#: aleksis/core/templates/account/password_change.html:6
+#: aleksis/core/templates/account/password_change.html:19
+#: aleksis/core/templates/account/password_reset_from_key.html:5
+#: aleksis/core/templates/account/password_reset_from_key.html:42
+#: aleksis/core/templates/account/password_reset_from_key.html:46
+#: aleksis/core/templates/account/password_reset_from_key_done.html:5
+#: aleksis/core/templates/account/password_reset_from_key_done.html:6
+msgid "Change password"
+msgstr "Passwort ändern"
+
+#: aleksis/core/menus.py:81
 msgid "Me"
 msgstr "Ich"
 
-#: menus.py:56 templates/dynamic_preferences/form.html:5
+#: aleksis/core/menus.py:90
+#: aleksis/core/templates/dynamic_preferences/form.html:5
 msgid "Preferences"
 msgstr "Einstellungen"
 
-#: menus.py:67
+#: aleksis/core/menus.py:99
+msgid "Third-party accounts"
+msgstr "Drittanbieter-Konten"
+
+#: aleksis/core/menus.py:108
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:5
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:6
+msgid "Authorized applications"
+msgstr "Autorisierte Anwendungen"
+
+#: aleksis/core/menus.py:119
 msgid "Admin"
 msgstr "Admin"
 
-#: menus.py:75 models.py:602 templates/core/announcement/list.html:7
-#: templates/core/announcement/list.html:8
+#: aleksis/core/menus.py:127 aleksis/core/models.py:704
+#: aleksis/core/templates/core/announcement/list.html:7
+#: aleksis/core/templates/core/announcement/list.html:8
 msgid "Announcements"
 msgstr "Ankündigungen"
 
-#: menus.py:86 models.py:103 templates/core/school_term/list.html:8
-#: templates/core/school_term/list.html:9
+#: aleksis/core/menus.py:138 aleksis/core/models.py:118
+#: aleksis/core/templates/core/school_term/list.html:8
+#: aleksis/core/templates/core/school_term/list.html:9
 msgid "School terms"
 msgstr "Schuljahre"
 
-#: menus.py:97 templates/core/dashboard_widget/list.html:8
-#: templates/core/dashboard_widget/list.html:9
+#: aleksis/core/menus.py:149
+#: aleksis/core/templates/core/dashboard_widget/list.html:8
+#: aleksis/core/templates/core/dashboard_widget/list.html:9
 msgid "Dashboard widgets"
 msgstr "Dashboard-Widgets"
 
-#: menus.py:108 templates/core/management/data_management.html:6
-#: templates/core/management/data_management.html:7
+#: aleksis/core/menus.py:160
+#: aleksis/core/templates/core/management/data_management.html:6
+#: aleksis/core/templates/core/management/data_management.html:7
 msgid "Data management"
 msgstr "Datenverwaltung"
 
-#: menus.py:116 templates/core/pages/system_status.html:5
-#: templates/core/pages/system_status.html:7
+#: aleksis/core/menus.py:171
+#: aleksis/core/templates/core/pages/system_status.html:5
+#: aleksis/core/templates/core/pages/system_status.html:7
 msgid "System status"
 msgstr "Systemstatus"
 
-#: menus.py:127
+#: aleksis/core/menus.py:182
 msgid "Impersonation"
 msgstr "Verkleidung"
 
-#: menus.py:135
+#: aleksis/core/menus.py:193
 msgid "Configuration"
 msgstr "Konfiguration"
 
-#: menus.py:146 templates/core/data_check/list.html:9
-#: templates/core/data_check/list.html:10
+#: aleksis/core/menus.py:204 aleksis/core/templates/core/data_check/list.html:9
+#: aleksis/core/templates/core/data_check/list.html:10
 msgid "Data checks"
 msgstr "Datenprüfungen"
 
-#: menus.py:152
+#: aleksis/core/menus.py:210
 msgid "Backend Admin"
 msgstr "Backend-Administration"
 
-#: menus.py:160
+#: aleksis/core/menus.py:216
+#: aleksis/core/templates/oauth2_provider/application_detail.html:5
+#: aleksis/core/templates/oauth2_provider/application_list.html:5
+msgid "OAuth2 Applications"
+msgstr "OAuth2-Anwendungen"
+
+#: aleksis/core/menus.py:229
 msgid "People"
 msgstr "Leute"
 
-#: menus.py:185 models.py:812 templates/core/group_type/list.html:8
-#: templates/core/group_type/list.html:9
+#: aleksis/core/menus.py:260 aleksis/core/models.py:958
+#: aleksis/core/templates/core/group_type/list.html:8
+#: aleksis/core/templates/core/group_type/list.html:9
 msgid "Group types"
 msgstr "Gruppentypen"
 
-#: menus.py:196
-msgid "Persons and accounts"
-msgstr "Personen und Konten"
-
-#: menus.py:207
+#: aleksis/core/menus.py:271
 msgid "Groups and child groups"
 msgstr "Gruppen und Kindgruppen"
 
-#: menus.py:218 models.py:385 templates/core/additional_field/list.html:8
-#: templates/core/additional_field/list.html:9
+#: aleksis/core/menus.py:282 aleksis/core/models.py:446
+#: aleksis/core/templates/core/additional_field/list.html:8
+#: aleksis/core/templates/core/additional_field/list.html:9
 msgid "Additional fields"
 msgstr "Zusätzliche Felder"
 
-#: menus.py:233 templates/core/group/child_groups.html:7
-#: templates/core/group/child_groups.html:9
+#: aleksis/core/menus.py:297
+#: aleksis/core/templates/core/group/child_groups.html:7
+#: aleksis/core/templates/core/group/child_groups.html:9
 msgid "Assign child groups to groups"
 msgstr "Kindgruppen zu Gruppen zuordnen"
 
-#: mixins.py:384
+#: aleksis/core/mixins.py:498
 msgid "Linked school term"
 msgstr "Zugeordnetes Schuljahr"
 
-#: models.py:43
+#: aleksis/core/models.py:58
 msgid "Boolean (Yes/No)"
 msgstr "Boolean (Ja/Nein)"
 
-#: models.py:44
+#: aleksis/core/models.py:59
 msgid "Text (one line)"
 msgstr "Text (eine Zeile)"
 
-#: models.py:46
+#: aleksis/core/models.py:61
 msgid "Date and time"
 msgstr "Datum und Uhrzeit"
 
-#: models.py:47
+#: aleksis/core/models.py:62
 msgid "Decimal number"
 msgstr "Dezimalzahl"
 
-#: models.py:48 models.py:157
+#: aleksis/core/models.py:63 aleksis/core/models.py:186
 msgid "E-mail address"
 msgstr "E-Mail-Adresse"
 
-#: models.py:49
+#: aleksis/core/models.py:64
 msgid "Integer"
 msgstr "Ganze Zahl"
 
-#: models.py:50
+#: aleksis/core/models.py:65
 msgid "IP address"
 msgstr "IP-Adresse"
 
-#: models.py:51
+#: aleksis/core/models.py:66
 msgid "Boolean or empty (Yes/No/Neither)"
 msgstr "Boolean oder leer (Ja/Nein/weder)"
 
-#: models.py:52
+#: aleksis/core/models.py:67
 msgid "Text (multi-line)"
 msgstr "Text (mehrzeilig)"
 
-#: models.py:54
+#: aleksis/core/models.py:69
 msgid "URL / Link"
 msgstr "URL / Link"
 
-#: models.py:66 models.py:785
+#: aleksis/core/models.py:81 aleksis/core/models.py:927
 msgid "Name"
 msgstr "Name"
 
-#: models.py:68
+#: aleksis/core/models.py:83
 msgid "Start date"
 msgstr "Startdatum"
 
-#: models.py:69
+#: aleksis/core/models.py:84
 msgid "End date"
 msgstr "Enddatum"
 
-#: models.py:88
+#: aleksis/core/models.py:103
 msgid "The start date must be earlier than the end date."
 msgstr "Das Startdatum muss vor dem Enddatum liegen."
 
-#: models.py:95
+#: aleksis/core/models.py:110
 msgid "There is already a school term for this time or a part of this time."
 msgstr "Es gibt bereits ein Schuljahr für diesen Zeitraum oder einen Teilzeitraum."
 
-#: models.py:115 models.py:744 templates/core/person/accounts.html:41
+#: aleksis/core/models.py:139 aleksis/core/models.py:876
 msgid "Person"
 msgstr "Person"
 
-#: models.py:118
+#: aleksis/core/models.py:142
 msgid "Can view address"
 msgstr "Kann Adresse sehen"
 
-#: models.py:119
+#: aleksis/core/models.py:143
 msgid "Can view contact details"
 msgstr "Kann Kontaktdetails sehen"
 
-#: models.py:120
+#: aleksis/core/models.py:144
 msgid "Can view photo"
 msgstr "Kann Foto sehen"
 
-#: models.py:121
+#: aleksis/core/models.py:145
 msgid "Can view persons groups"
 msgstr "Kann Gruppen einer Person sehen"
 
-#: models.py:122
+#: aleksis/core/models.py:146
 msgid "Can view personal details"
 msgstr "Kann persönliche Daten sehen"
 
-#: models.py:127
+#: aleksis/core/models.py:156
 msgid "female"
 msgstr "weiblich"
 
-#: models.py:127
+#: aleksis/core/models.py:156
 msgid "male"
 msgstr "männlich"
 
-#: models.py:135
+#: aleksis/core/models.py:164
 msgid "Linked user"
 msgstr "Verknüpfter Benutzer"
 
-#: models.py:137
+#: aleksis/core/models.py:166
 msgid "Is person active?"
 msgstr "Ist die Person aktiv?"
 
-#: models.py:139
+#: aleksis/core/models.py:168
 msgid "First name"
 msgstr "Vorname"
 
-#: models.py:140
+#: aleksis/core/models.py:169
 msgid "Last name"
 msgstr "Nachname"
 
-#: models.py:142
+#: aleksis/core/models.py:171
 msgid "Additional name(s)"
 msgstr "Zusätzliche Namen"
 
-#: models.py:146 models.py:354
+#: aleksis/core/models.py:175 aleksis/core/models.py:415
 msgid "Short name"
 msgstr "Kurzname"
 
-#: models.py:149
+#: aleksis/core/models.py:178
 msgid "Street"
 msgstr "Straße"
 
-#: models.py:150
+#: aleksis/core/models.py:179
 msgid "Street number"
 msgstr "Hausnummer"
 
-#: models.py:151
+#: aleksis/core/models.py:180
 msgid "Postal code"
 msgstr "Postleitzahl"
 
-#: models.py:152
+#: aleksis/core/models.py:181
 msgid "Place"
 msgstr "Ort"
 
-#: models.py:154
+#: aleksis/core/models.py:183
 msgid "Home phone"
 msgstr "Festnetz"
 
-#: models.py:155
+#: aleksis/core/models.py:184
 msgid "Mobile phone"
 msgstr "Handy"
 
-#: models.py:159
+#: aleksis/core/models.py:188
 msgid "Date of birth"
 msgstr "Geburtsdatum"
 
-#: models.py:160
+#: aleksis/core/models.py:189
 msgid "Sex"
 msgstr "Geschlecht"
 
-#: models.py:162
+#: aleksis/core/models.py:191
 msgid "Photo"
 msgstr "Foto"
 
-#: models.py:166 templates/core/person/full.html:137
+#: aleksis/core/models.py:195 aleksis/core/templates/core/person/full.html:137
 msgid "Guardians / Parents"
 msgstr "Erziehungsberechtigte / Eltern"
 
-#: models.py:173
+#: aleksis/core/models.py:202
 msgid "Primary group"
 msgstr "Primärgruppe"
 
-#: models.py:176 models.py:461 models.py:485 models.py:570 models.py:805
-#: templates/core/person/full.html:120
+#: aleksis/core/models.py:205 aleksis/core/models.py:563
+#: aleksis/core/models.py:587 aleksis/core/models.py:672
+#: aleksis/core/models.py:951 aleksis/core/templates/core/person/full.html:120
 msgid "Description"
 msgstr "Beschreibung"
 
-#: models.py:313
+#: aleksis/core/models.py:370
 msgid "Title of field"
 msgstr "Feldtitel"
 
-#: models.py:315
+#: aleksis/core/models.py:372
 msgid "Type of field"
 msgstr "Feldtyp"
 
-#: models.py:322
+#: aleksis/core/models.py:379
 msgid "Addtitional field for groups"
 msgstr "Zusätzliche Felder für Gruppen"
 
-#: models.py:323
+#: aleksis/core/models.py:380
 msgid "Addtitional fields for groups"
 msgstr "Zusätzliche Felder für Gruppen"
 
-#: models.py:337
+#: aleksis/core/models.py:397
 msgid "Group"
 msgstr "Gruppe"
 
-#: models.py:340
+#: aleksis/core/models.py:400
 msgid "Can assign child groups to groups"
 msgstr "Kann Kindgruppen zu Gruppen zuordnen"
 
-#: models.py:341
+#: aleksis/core/models.py:401
 msgid "Can view statistics about group."
 msgstr "Kann Statistiken über Gruppen sehen."
 
-#: models.py:352
+#: aleksis/core/models.py:413
 msgid "Long name"
 msgstr "Langname"
 
-#: models.py:362 templates/core/group/full.html:85
+#: aleksis/core/models.py:423 aleksis/core/templates/core/group/full.html:85
 msgid "Members"
 msgstr "Mitglieder"
 
-#: models.py:365 templates/core/group/full.html:82
+#: aleksis/core/models.py:426 aleksis/core/templates/core/group/full.html:82
 msgid "Owners"
 msgstr "Leiter/-innen"
 
-#: models.py:372 templates/core/group/full.html:55
+#: aleksis/core/models.py:433 aleksis/core/templates/core/group/full.html:55
 msgid "Parent groups"
 msgstr "Ãœbergeordnete Gruppen"
 
-#: models.py:380
+#: aleksis/core/models.py:441
 msgid "Type of group"
 msgstr "Gruppentyp"
 
-#: models.py:457
+#: aleksis/core/models.py:559
 msgid "User"
 msgstr "Benutzer"
 
-#: models.py:460 models.py:484 models.py:569
-#: templates/core/announcement/list.html:18
+#: aleksis/core/models.py:562 aleksis/core/models.py:586
+#: aleksis/core/models.py:671
+#: aleksis/core/templates/core/announcement/list.html:18
 msgid "Title"
 msgstr "Titel"
 
-#: models.py:463
+#: aleksis/core/models.py:565
 msgid "Application"
 msgstr "Anwendung"
 
-#: models.py:469
+#: aleksis/core/models.py:571
 msgid "Activity"
 msgstr "Aktivität"
 
-#: models.py:470
+#: aleksis/core/models.py:572
 msgid "Activities"
 msgstr "Aktivitäten"
 
-#: models.py:476
+#: aleksis/core/models.py:578
 msgid "Sender"
 msgstr "Absender"
 
-#: models.py:481
+#: aleksis/core/models.py:583
 msgid "Recipient"
 msgstr "Empfänger"
 
-#: models.py:486 models.py:786
+#: aleksis/core/models.py:588 aleksis/core/models.py:928
 msgid "Link"
 msgstr "Link"
 
-#: models.py:488
+#: aleksis/core/models.py:590
 msgid "Read"
 msgstr "Gelesen"
 
-#: models.py:489
+#: aleksis/core/models.py:591
 msgid "Sent"
 msgstr "Versandt"
 
-#: models.py:502
+#: aleksis/core/models.py:604
 msgid "Notification"
 msgstr "Benachrichtigung"
 
-#: models.py:503
-msgid "Notifications"
-msgstr "Benachrichtigungen"
-
-#: models.py:571
+#: aleksis/core/models.py:673
 msgid "Link to detailed view"
 msgstr "Link zur detaillierten Ansicht"
 
-#: models.py:574
+#: aleksis/core/models.py:676
 msgid "Date and time from when to show"
 msgstr "Datum und Uhrzeit des Anzeigestarts"
 
-#: models.py:577
+#: aleksis/core/models.py:679
 msgid "Date and time until when to show"
 msgstr "Anzeigezeitraum"
 
-#: models.py:601
+#: aleksis/core/models.py:703
 msgid "Announcement"
 msgstr "Ankündigung"
 
-#: models.py:639
+#: aleksis/core/models.py:741
 msgid "Announcement recipient"
 msgstr "Empfänger der Ankündigung"
 
-#: models.py:640
+#: aleksis/core/models.py:742
 msgid "Announcement recipients"
 msgstr "Empfänger der Ankündigung"
 
-#: models.py:690
+#: aleksis/core/models.py:797
 msgid "Widget Title"
 msgstr "Widget-Titel"
 
-#: models.py:691
+#: aleksis/core/models.py:798
 msgid "Activate Widget"
 msgstr "Widget aktivieren"
 
-#: models.py:694
+#: aleksis/core/models.py:799
+msgid "Widget is broken"
+msgstr "Widget ist kaputt"
+
+#: aleksis/core/models.py:802
 msgid "Size on mobile devices"
 msgstr "Größe auf Mobilgeräten"
 
-#: models.py:695
+#: aleksis/core/models.py:803
 msgid "<= 600 px, 12 columns"
 msgstr "<= 600 px, 12 Spalten"
 
-#: models.py:700
+#: aleksis/core/models.py:808
 msgid "Size on tablet devices"
 msgstr "Größe auf Tablets"
 
-#: models.py:701
+#: aleksis/core/models.py:809
 msgid "> 600 px, 12 columns"
 msgstr "> 600px, 12 Spalten"
 
-#: models.py:706
+#: aleksis/core/models.py:814
 msgid "Size on desktop devices"
 msgstr "Größe auf Desktopgeräten"
 
-#: models.py:707
+#: aleksis/core/models.py:815
 msgid "> 992 px, 12 columns"
 msgstr "> 992 px, 12 Spalten"
 
-#: models.py:712
+#: aleksis/core/models.py:820
 msgid "Size on large desktop devices"
 msgstr "Größe auf großen Desktopgeräten"
 
-#: models.py:713
+#: aleksis/core/models.py:821
 msgid "> 1200 px>, 12 columns"
 msgstr "> 1200 px, 12 Spalten"
 
-#: models.py:734
+#: aleksis/core/models.py:852
 msgid "Can edit default dashboard"
 msgstr "Kann Standarddashboard bearbeiten"
 
-#: models.py:735
+#: aleksis/core/models.py:853
 msgid "Dashboard Widget"
 msgstr "Dashboard-Widget"
 
-#: models.py:736
+#: aleksis/core/models.py:854
 msgid "Dashboard Widgets"
 msgstr "Dashboard-Widgets"
 
-#: models.py:741
+#: aleksis/core/models.py:860
+msgid "URL"
+msgstr "URL"
+
+#: aleksis/core/models.py:861
+msgid "Icon URL"
+msgstr "Symbol-URL"
+
+#: aleksis/core/models.py:867
+msgid "External link widget"
+msgstr "Externer-Link-Widget"
+
+#: aleksis/core/models.py:868
+msgid "External link widgets"
+msgstr "Externer-Link-Widgets"
+
+#: aleksis/core/models.py:873
 msgid "Dashboard widget"
 msgstr "Dashboard-Widget"
 
-#: models.py:746
+#: aleksis/core/models.py:878
 msgid "Order"
 msgstr "Reihenfolge"
 
-#: models.py:747
+#: aleksis/core/models.py:879
 msgid "Part of the default dashboard"
 msgstr "Teil des Standarddashboards"
 
-#: models.py:755
+#: aleksis/core/models.py:894
 msgid "Dashboard widget order"
 msgstr "Reihenfolge der Dashboard-Widgets"
 
-#: models.py:756
+#: aleksis/core/models.py:895
 msgid "Dashboard widget orders"
 msgstr "Reihenfolgen der Dashboard-Widgets"
 
-#: models.py:762
+#: aleksis/core/models.py:901
 msgid "Menu ID"
 msgstr "Menü-ID"
 
-#: models.py:775
+#: aleksis/core/models.py:914
 msgid "Custom menu"
 msgstr "Benutzerdefiniertes Menü"
 
-#: models.py:776
+#: aleksis/core/models.py:915
 msgid "Custom menus"
 msgstr "Benutzerdefinierte Menüs"
 
-#: models.py:783
+#: aleksis/core/models.py:925
 msgid "Menu"
 msgstr "Menü"
 
-#: models.py:787
+#: aleksis/core/models.py:929
 msgid "Icon"
 msgstr "Symbol"
 
-#: models.py:793
+#: aleksis/core/models.py:935
 msgid "Custom menu item"
 msgstr "Benutzerdefiniertes Menüelement"
 
-#: models.py:794
+#: aleksis/core/models.py:936
 msgid "Custom menu items"
 msgstr "Benutzerdefinierte Menüelemente"
 
-#: models.py:804
+#: aleksis/core/models.py:950
 msgid "Title of type"
 msgstr "Titel des Typs"
 
-#: models.py:811 templates/core/group/full.html:47
+#: aleksis/core/models.py:957 aleksis/core/templates/core/group/full.html:47
 msgid "Group type"
 msgstr "Gruppentyp"
 
-#: models.py:821
+#: aleksis/core/models.py:971
 msgid "Can view system status"
 msgstr "Kann Systemstatus sehen"
 
-#: models.py:822
-msgid "Can link persons to accounts"
-msgstr "Kann Personen mit Benutzerkonten verknüpfen"
-
-#: models.py:823
+#: aleksis/core/models.py:972
 msgid "Can manage data"
 msgstr "Kann Daten verwalten"
 
-#: models.py:824
+#: aleksis/core/models.py:973
 msgid "Can impersonate"
 msgstr "Kann sich verkleiden"
 
-#: models.py:825
+#: aleksis/core/models.py:974
 msgid "Can use search"
 msgstr "Kann Suche benutzen"
 
-#: models.py:826
+#: aleksis/core/models.py:975
 msgid "Can change site preferences"
 msgstr "Kann Konfiguration ändern"
 
-#: models.py:827
+#: aleksis/core/models.py:976
 msgid "Can change person preferences"
 msgstr "Kann Einstellungen einer Person verändern"
 
-#: models.py:828
+#: aleksis/core/models.py:977
 msgid "Can change group preferences"
 msgstr "Kann Einstellungen einer Gruppe verändern"
 
-#: models.py:864
+#: aleksis/core/models.py:978
+msgid "Can add oauth applications"
+msgstr "Kann OAuth-Anwendungen hinzufügen"
+
+#: aleksis/core/models.py:979
+msgid "Can list oauth applications"
+msgstr "Can OAuth-Anwendungen auflisten"
+
+#: aleksis/core/models.py:980
+msgid "Can view oauth applications"
+msgstr "Kann OAuth-Anwendungen sehen"
+
+#: aleksis/core/models.py:981
+msgid "Can update oauth applications"
+msgstr "Kann OAuth-Anwendungen aktualisieren"
+
+#: aleksis/core/models.py:982
+msgid "Can delete oauth applications"
+msgstr "Kann OAuth-Anwendungen löschen"
+
+#: aleksis/core/models.py:983
+msgid "Can test PDF generation"
+msgstr "Kann die PDF-Generierung testen"
+
+#: aleksis/core/models.py:1019
 msgid "Related data check task"
 msgstr "Zugehörige Datenprüfungsaufgabe"
 
-#: models.py:872
+#: aleksis/core/models.py:1027
 msgid "Issue solved"
 msgstr "Problem gelöst"
 
-#: models.py:873
+#: aleksis/core/models.py:1028
 msgid "Notification sent"
 msgstr "Benachrichtigung gesendet"
 
-#: models.py:886
+#: aleksis/core/models.py:1041
 msgid "Data check result"
 msgstr "Datenprüfungsergebnis"
 
-#: models.py:887
+#: aleksis/core/models.py:1042
 msgid "Data check results"
 msgstr "Datenprüfungsergebnisse"
 
-#: models.py:889
+#: aleksis/core/models.py:1044
 msgid "Can run data checks"
 msgstr "Kann Datenprüfungen ausführen"
 
-#: models.py:890
+#: aleksis/core/models.py:1045
 msgid "Can solve data check problems"
 msgstr "Kann Datenprüfungsprobleme lösen"
 
-#: preferences.py:27
+#: aleksis/core/models.py:1060
+msgid "Owner"
+msgstr "Leiter"
+
+#: aleksis/core/models.py:1064
+msgid "File expires at"
+msgstr "Datei abgelaufen am"
+
+#: aleksis/core/models.py:1066
+msgid "Generated HTML file"
+msgstr "Generierte HTML-Datei"
+
+#: aleksis/core/models.py:1068
+msgid "Generated PDF file"
+msgstr "Generierte PDF-Datei"
+
+#: aleksis/core/models.py:1075
+msgid "PDF file"
+msgstr "PDF-Datei"
+
+#: aleksis/core/models.py:1076
+msgid "PDF files"
+msgstr "PDF-Dateien"
+
+#: aleksis/core/models.py:1081
+msgid "Task result"
+msgstr "Task-Ergebnis"
+
+#: aleksis/core/models.py:1084
+msgid "Task user"
+msgstr "Task-Benutzer"
+
+#: aleksis/core/models.py:1096
+msgid "Task user assignment"
+msgstr "Task-Benutzer-Zuordnung"
+
+#: aleksis/core/models.py:1097
+msgid "Task user assignments"
+msgstr "Task-Benutzer-Zuordnungen"
+
+#: aleksis/core/preferences.py:22
+msgid "General"
+msgstr "Allgemein"
+
+#: aleksis/core/preferences.py:23
+msgid "School"
+msgstr "Schule"
+
+#: aleksis/core/preferences.py:24
+msgid "Theme"
+msgstr "Theme"
+
+#: aleksis/core/preferences.py:25
+msgid "Mail"
+msgstr "E-Mail"
+
+#: aleksis/core/preferences.py:27
+msgid "Footer"
+msgstr "Fußbereich"
+
+#: aleksis/core/preferences.py:28
+msgid "Accounts"
+msgstr "Konten"
+
+#: aleksis/core/preferences.py:29
 msgid "Authentication"
 msgstr "Authentifizierung"
 
-#: preferences.py:28
+#: aleksis/core/preferences.py:30
 msgid "Internationalisation"
 msgstr "Internationalisierung"
 
-#: preferences.py:37
+#: aleksis/core/preferences.py:41
 msgid "Site title"
 msgstr "Seitentitel"
 
-#: preferences.py:46
+#: aleksis/core/preferences.py:52
 msgid "Site description"
 msgstr "Seitenbeschreibung"
 
-#: preferences.py:55
+#: aleksis/core/preferences.py:63
 msgid "Primary colour"
 msgstr "Primärfarbe"
 
-#: preferences.py:64
+#: aleksis/core/preferences.py:75
 msgid "Secondary colour"
 msgstr "Akzentfarbe"
 
-#: preferences.py:72
+#: aleksis/core/preferences.py:86
 msgid "Logo"
 msgstr "Logo"
 
-#: preferences.py:80
+#: aleksis/core/preferences.py:96
 msgid "Favicon"
 msgstr "Favicon"
 
-#: preferences.py:88
+#: aleksis/core/preferences.py:106
 msgid "PWA-Icon"
 msgstr "PWA-Icon"
 
-#: preferences.py:97
+#: aleksis/core/preferences.py:117
 msgid "Mail out name"
 msgstr "Ausgangsmailname"
 
-#: preferences.py:106
+#: aleksis/core/preferences.py:128
 msgid "Mail out address"
 msgstr "E-Mail-Ausgangsadresse"
 
-#: preferences.py:116
+#: aleksis/core/preferences.py:140
 msgid "Link to privacy policy"
 msgstr "Link zur Datenschutzerklärung"
 
-#: preferences.py:126
+#: aleksis/core/preferences.py:152
 msgid "Link to imprint"
 msgstr "Link zum Impressum"
 
-#: preferences.py:136
+#: aleksis/core/preferences.py:164
 msgid "Name format for addressing"
 msgstr "Namensformat für Anreden"
 
-#: preferences.py:150
+#: aleksis/core/preferences.py:180
 msgid "Channels to use for notifications"
 msgstr "Aktivierte Benachrichtungskanäle"
 
-#: preferences.py:160
+#: aleksis/core/preferences.py:192
 msgid "Regular expression to match primary group, e.g. '^Class .*'"
 msgstr "Regulärer Ausdruck um Primärgruppen zu finden, z. B.  '^Class .*'"
 
-#: preferences.py:169
+#: aleksis/core/preferences.py:203
 msgid "Field on person to match primary group against"
 msgstr "Feld um Primärgruppen zu finden"
 
-#: preferences.py:181
+#: aleksis/core/preferences.py:215
+msgid "Automatically create new persons for new users"
+msgstr "Erstelle automatisch neue Personen für neue Benutzer"
+
+#: aleksis/core/preferences.py:224
+msgid "Automatically link existing persons to new users by their e-mail address"
+msgstr "Verknüpfe existierende Personen automatisch mit neuen Personen anhand ihrer E-Mail-Adresse"
+
+#: aleksis/core/preferences.py:235
 msgid "Display name of the school"
 msgstr "Sichtbarer Name der Schule"
 
-#: preferences.py:190
+#: aleksis/core/preferences.py:246
 msgid "Official name of the school, e.g. as given by supervisory authority"
 msgstr "Offizieller Name der Schule, wie er z.B. von der Behörde vorgegeben ist"
 
-#: preferences.py:198
-msgid "Enabled custom authentication backends"
-msgstr "Benutzerdefinierte Authentifizierungsbackends aktivieren"
+#: aleksis/core/preferences.py:254
+msgid "Allow users to change their passwords"
+msgstr "Erlaube Benutzern, ihr Passwort zu ändern"
 
-#: preferences.py:211
+#: aleksis/core/preferences.py:262
+msgid "Enable signup"
+msgstr "Registrierung aktivieren"
+
+#: aleksis/core/preferences.py:273
 msgid "Available languages"
 msgstr "Verfügbare Sprachen"
 
-#: preferences.py:223
+#: aleksis/core/preferences.py:285
 msgid "Send emails if data checks detect problems"
 msgstr "E-Mails versenden, wenn Datenprüfungen Probleme finden"
 
-#: preferences.py:234
+#: aleksis/core/preferences.py:296
 msgid "Email recipients for data checks problem emails"
 msgstr "E-Mailempfänger für Datenprüfungsproblem-E-Mails"
 
-#: preferences.py:245
+#: aleksis/core/preferences.py:307
 msgid "Email recipient groups for data checks problem emails"
 msgstr "E-Mail-Empfängergruppen für Datenprüfungsproblem-E-Mails"
 
-#: settings.py:322
+#: aleksis/core/preferences.py:316
+msgid "Show dashboard to users without login"
+msgstr "Zeige Dashboard für Benutzer ohne Login"
+
+#: aleksis/core/preferences.py:325
+msgid "Allow users to edit their dashboard"
+msgstr "Erlaube Benutzern, ihr Dashboard zu bearbeiten"
+
+#: aleksis/core/preferences.py:336
+msgid "Fields on person model which are editable by themselves."
+msgstr "Felder des Personen-Models welche von ihnen selbst editierbar sind."
+
+#: aleksis/core/preferences.py:350
+msgid "Editable fields on person model which should trigger a notification on change"
+msgstr "Editierbare Felder des Personen-Models welche eine Benachrichtigung für Änderungen auslösen soll"
+
+#: aleksis/core/preferences.py:363
+msgid "Contact for notification if a person changes their data"
+msgstr "Kontakt für Benachrichtigung, wenn eine Person ihre Daten ändert"
+
+#: aleksis/core/preferences.py:373
+msgid "PDF file expiration duration"
+msgstr "PDF-Datei-Ablaufdauer"
+
+#: aleksis/core/preferences.py:374
+msgid "in minutes"
+msgstr "in Minuten"
+
+#: aleksis/core/preferences.py:384
+msgid "Automatically update the dashboard and its widgets"
+msgstr "Automatisch das Dashboard und seine Widgets aktualisieren"
+
+#: aleksis/core/preferences.py:394
+msgid "Automatically update the dashboard and its widgets sitewide"
+msgstr "Automatisch das Dashboard und seine Widgets aktualisieren (auf der ganzen Seite)"
+
+#: aleksis/core/settings.py:452
 msgid "English"
 msgstr "Englisch"
 
-#: settings.py:323
+#: aleksis/core/settings.py:453
 msgid "German"
 msgstr "Deutsch"
 
-#: settings.py:324
-msgid "French"
-msgstr "Französisch"
-
-#: settings.py:325
-msgid "Norwegian (bokmål)"
-msgstr "Norwegisch (bokmål)"
-
-#: tables.py:19 templates/core/announcement/list.html:36
-#: templates/core/group/full.html:24 templates/core/person/full.html:23
+#: aleksis/core/tables.py:19
+#: aleksis/core/templates/core/announcement/list.html:36
+#: aleksis/core/templates/core/group/full.html:24
+#: aleksis/core/templates/core/person/full.html:23
+#: aleksis/core/templates/oauth2_provider/application_detail.html:17
 msgid "Edit"
 msgstr "Bearbeiten"
 
-#: tables.py:21 tables.py:89 templates/core/announcement/list.html:22
+#: aleksis/core/tables.py:21 aleksis/core/tables.py:89
+#: aleksis/core/templates/core/announcement/list.html:22
 msgid "Actions"
 msgstr "Aktionen"
 
-#: tables.py:56 tables.py:57 tables.py:71 tables.py:87
-#: templates/core/announcement/list.html:42 templates/core/group/full.html:31
-#: templates/core/pages/delete.html:22 templates/core/person/full.html:30
+#: aleksis/core/tables.py:56 aleksis/core/tables.py:57
+#: aleksis/core/tables.py:71 aleksis/core/tables.py:87
+#: aleksis/core/templates/core/announcement/list.html:42
+#: aleksis/core/templates/core/group/full.html:31
+#: aleksis/core/templates/core/pages/delete.html:22
+#: aleksis/core/templates/core/person/full.html:30
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:20
+#: aleksis/core/templates/oauth2_provider/application_detail.html:21
 msgid "Delete"
 msgstr "Löschen"
 
-#: templates/403.html:14 templates/404.html:10 templates/500.html:10
+#: aleksis/core/templates/403.html:14 aleksis/core/templates/404.html:10
+#: aleksis/core/templates/500.html:10
+#: aleksis/core/templates/oauth2_provider/authorize.html:47
+#: aleksis/core/templates/oauth2_provider/authorized-oob.html:24
 msgid "Error"
 msgstr "Fehler"
 
-#: templates/403.html:14
+#: aleksis/core/templates/403.html:14
 msgid ""
 "You are not allowed to access the requested page or\n"
 "          object."
@@ -836,7 +1127,7 @@ msgstr ""
 "Es ist Ihnen nicht erlaubt, auf die angefragte Seite oder das angefragte\n"
 "             Objekt zuzugreifen."
 
-#: templates/403.html:19 templates/404.html:17
+#: aleksis/core/templates/403.html:19 aleksis/core/templates/404.html:17
 msgid ""
 "\n"
 "            If you think this is an error in AlekSIS, please contact your site\n"
@@ -848,7 +1139,7 @@ msgstr ""
 "     Systemadministratoren:\n"
 "          "
 
-#: templates/404.html:10
+#: aleksis/core/templates/404.html:10
 msgid ""
 "The requested page or object was not\n"
 "          found."
@@ -856,7 +1147,7 @@ msgstr ""
 "Die angefragte Seite oder das angefragte Objekt wurde nicht\n"
 "            gefunden."
 
-#: templates/404.html:13
+#: aleksis/core/templates/404.html:13
 msgid ""
 "\n"
 "            If you were redirected by a link on an external page,\n"
@@ -868,7 +1159,7 @@ msgstr ""
 "      ist es möglich, dass dieser veraltet war.\n"
 "          "
 
-#: templates/500.html:10
+#: aleksis/core/templates/500.html:10
 msgid ""
 "An unexpected error has\n"
 "          occured."
@@ -876,7 +1167,7 @@ msgstr ""
 "Ein unerwarteter Fehler ist\n"
 "            aufgetreten."
 
-#: templates/500.html:13
+#: aleksis/core/templates/500.html:13
 msgid ""
 "\n"
 "            Your site administrators will automatically be notified about this\n"
@@ -888,7 +1179,7 @@ msgstr ""
 "      Sie können diese auch direkt kontaktieren:\n"
 "          "
 
-#: templates/503.html:10
+#: aleksis/core/templates/503.html:10
 msgid ""
 "The maintenance mode is currently enabled. Please try again\n"
 "          later."
@@ -896,7 +1187,7 @@ msgstr ""
 "Der Wartungsmodus ist aktuell aktiviert. Bitte versuchen Sie es\n"
 "            später erneut."
 
-#: templates/503.html:13
+#: aleksis/core/templates/503.html:13
 msgid ""
 "\n"
 "            This page is currently unavailable. If this error persists, contact your site administrators:\n"
@@ -906,100 +1197,367 @@ msgstr ""
 "            Diese Seite ist aktuell nicht erreichbar. Wenn dieser Fehler bestehen bleibt, kontaktieren Sie bitte einen Ihrer Systemadministratoren:\n"
 "          "
 
-#: templates/core/additional_field/edit.html:6
-#: templates/core/additional_field/edit.html:7
+#: aleksis/core/templates/account/account_inactive.html:5
+#: aleksis/core/templates/account/account_inactive.html:6
+msgid "Account inactive"
+msgstr "Konto inaktiv"
+
+#: aleksis/core/templates/account/account_inactive.html:13
+msgid "Account inactive."
+msgstr "Konto inaktiv."
+
+#: aleksis/core/templates/account/account_inactive.html:15
+msgid ""
+"\n"
+"            This account is currently inactive. If you think this is an\n"
+"            error, please contact one of your site administrators.\n"
+"          "
+msgstr ""
+"\n"
+"            Dieses Konto ist aktuell inaktiv. Wenn Sie denken,\n"
+"dass dies ein Fehler ist, kontaktieren Sie einen der Administratoren:\n"
+"          "
+
+#: aleksis/core/templates/account/email/base_message.txt:5
+msgid "Hello!"
+msgstr "Hallo!"
+
+#: aleksis/core/templates/account/email/base_message.txt:9
+#: aleksis/core/templates/templated_email/notification.email:22
+#: aleksis/core/templates/templated_email/notification.email:46
+msgid "Your AlekSIS team"
+msgstr "Ihr AlekSIS-Team"
+
+#: aleksis/core/templates/account/email_confirm.html:5
+#: aleksis/core/templates/account/email_confirm.html:6
+#: aleksis/core/templates/account/email_confirm.html:17
+msgid "Confirm"
+msgstr "Bestätigen"
+
+#: aleksis/core/templates/account/email_confirm.html:12
+#, python-format
+msgid "Please confirm that <a href=\"mailto:%(email)s\">%(email)s</a> is an e-mail address for user %(user_display)s."
+msgstr "Bitte bestätigen Sie, dass <a href=\"mailto:%(email)s\">%(email)s</a> eine E-Mail-Adresse für den Benutzer %(user_display)s ist."
+
+#: aleksis/core/templates/account/email_confirm.html:25
+#, python-format
+msgid "This e-mail confirmation link expired or is invalid. Please <a href=\"%(email_url)s\">issue a new e-mail confirmation request</a>."
+msgstr "Dieser E-Mail-Bestätigungslink ist abgelaufen oder nicht gültig. Bitte <a href=\"%(email_url)s\">fragen Sie eine neue E-Mail-Bestätigung an</a>."
+
+#: aleksis/core/templates/account/password_change.html:12
+msgid "Forgot your current password? Click here to reset it:"
+msgstr "Haben Sie Ihr aktuelles Passwort vergessen? Klicken Sie hier, um es zurückzusetzen:"
+
+#: aleksis/core/templates/account/password_change.html:12
+msgid "Forgot Password?"
+msgstr "Passwort vergessen?"
+
+#: aleksis/core/templates/account/password_change_disabled.html:5
+#: aleksis/core/templates/account/password_change_disabled.html:6
+msgid "Changing of password disabled"
+msgstr "Passwortänderung deaktiviert"
+
+#: aleksis/core/templates/account/password_change_disabled.html:13
+msgid "Changing of password disabled."
+msgstr "Passwortänderung deaktiviert."
+
+#: aleksis/core/templates/account/password_change_disabled.html:15
+msgid ""
+"\n"
+"            Users are not allowed to edit their own passwords. If you think\n"
+"            this is an error please contact one of your site administrators.\n"
+"          "
+msgstr ""
+"\n"
+"            Benutzer dürfen ihre eigenen Passwörter nicht ändern. Wenn Sie denken, \n"
+"dass dies ein Fehler ist, kontaktieren Sie bitte einen der Administratoren:\n"
+"          "
+
+#: aleksis/core/templates/account/password_reset.html:5
+#: aleksis/core/templates/account/password_reset.html:15
+#: aleksis/core/templates/account/password_reset.html:23
+#: aleksis/core/templates/account/password_reset_done.html:5
+#: aleksis/core/templates/account/verification_email_required.html:5
+#: aleksis/core/templates/account/verification_email_required.html:6
+#: aleksis/core/templates/two_factor/core/login.html:81
+msgid "Reset password"
+msgstr "Passwort zurücksetzen"
+
+#: aleksis/core/templates/account/password_reset.html:17
+msgid "Forgotten your password? Enter your e-mail address below, and we'll send you an e-mail allowing you to reset it."
+msgstr "Passwort vergessen? Geben Sie Ihre E-Mail-Adresse hier ein und wir werden Ihnen eine E-Mail zum Zurücksetzen des Passwortes schicken."
+
+#: aleksis/core/templates/account/password_reset.html:30
+msgid ""
+"Please contact one of your site administrators, if you\n"
+"        have any trouble resetting your password:"
+msgstr ""
+"Bitte kontaktieren Sie einen der Administratoren, \n"
+"wenn Sie irgendwelche Probleme beim Zurücksetzen des Passwortes haben:"
+
+#: aleksis/core/templates/account/password_reset_done.html:15
+msgid "Password reset mail sent"
+msgstr "E-Mail zum Zurücksetzen des Passwortes versendet"
+
+#: aleksis/core/templates/account/password_reset_done.html:18
+#: aleksis/core/templates/account/verification_email_required.html:16
+msgid ""
+"\n"
+"            We have sent you an e-mail. Please contact one of your site\n"
+"            administrators if you do not receive it within a few minutes.\n"
+"          "
+msgstr ""
+"\n"
+"            Wir haben Ihnen eine E-Mail gesendet. Bitte kontaktieren Sie einen der Administratoren,\n"
+"wenn Sie diese nicht innerhalb weniger Minuten erhalten.\n"
+"          "
+
+#: aleksis/core/templates/account/password_reset_from_key.html:15
+msgid "Bad token"
+msgstr "Ungültiges Token"
+
+#: aleksis/core/templates/account/password_reset_from_key.html:19
+#, python-format
+msgid ""
+"\n"
+"              The password reset link was invalid, possibly because it has already been used. Please request a <a href=\"%(passwd_reset_url)s\"\n"
+"              class=\"blue-text text-lighten-2\">new password reset</a>.\n"
+"            "
+msgstr ""
+"\n"
+"              Der Link zum Zurücksetzen des Passwortes war falsch, wahrscheinlich, weil er bereits benutzt wurde. Bitte starten Sie eine neue Anfrage <a href=\"%(passwd_reset_url)s\"              class=\"blue-text text-lighten-2\">zum Zurücksetzen des Passwortes</a>.\n"
+"            "
+
+#: aleksis/core/templates/account/password_reset_from_key.html:25
+msgid ""
+"\n"
+"              If this issue persists, please contact one of your site\n"
+"              administrators\n"
+"            "
+msgstr ""
+"\n"
+"              Wenn dieser Fehler bestehen bleibt,\n"
+"kontaktieren Sie bitte einen der Administratoren:\n"
+"            "
+
+#: aleksis/core/templates/account/password_reset_from_key.html:56
+#: aleksis/core/templates/account/password_reset_from_key_done.html:15
+msgid ""
+"\n"
+"            Your password is now changed!\n"
+"          "
+msgstr ""
+"\n"
+"            Ihr Password wurde geändert!\n"
+"          "
+
+#: aleksis/core/templates/account/password_reset_from_key.html:61
+msgid "Back to login"
+msgstr "Zurück zur Anmeldung"
+
+#: aleksis/core/templates/account/password_reset_from_key_done.html:13
+msgid "Password changed!"
+msgstr "Passwort geändert!"
+
+#: aleksis/core/templates/account/password_set.html:5
+#: aleksis/core/templates/account/password_set.html:6
+#: aleksis/core/templates/account/password_set.html:12
+msgid "Set password"
+msgstr "Password setzen"
+
+#: aleksis/core/templates/account/signup.html:5
+#: aleksis/core/templates/account/signup.html:6
+#: aleksis/core/templates/socialaccount/signup.html:5
+#: aleksis/core/templates/socialaccount/signup.html:6
+msgid "Signup"
+msgstr "Registrierung"
+
+#: aleksis/core/templates/account/signup.html:12
+#, python-format
+msgid "Already have an account? Then please <a href=\"%(login_url)s\">sign in</a>."
+msgstr "Haben Sie bereits ein Konto? Dann <a href=\"%(login_url)s\">melden Sie sich bitte an</a>."
+
+#: aleksis/core/templates/account/signup_closed.html:5
+#: aleksis/core/templates/account/signup_closed.html:6
+msgid "Signup closed"
+msgstr "Registrierung geschlossen"
+
+#: aleksis/core/templates/account/signup_closed.html:13
+msgid "Signup closed."
+msgstr "Registrierung geschlossen."
+
+#: aleksis/core/templates/account/signup_closed.html:15
+msgid ""
+"\n"
+"            This sign up is currently closed. If you think this is an\n"
+"            error, please contact one of your site administrators.\n"
+"          "
+msgstr ""
+"\n"
+"            Die Registrierung ist aktuell geschlossen. Wenn Sie denken, dass dies ein Fehler ist,\n"
+" kontaktieren Sie bitte einen Ihrer Systemadministratoren.\n"
+"          "
+
+#: aleksis/core/templates/account/verification_email_required.html:14
+msgid "Password reset mail sent!"
+msgstr "E-Mail zum Zurücksetzen des Passwortes versendet!"
+
+#: aleksis/core/templates/account/verification_sent.html:5
+#: aleksis/core/templates/account/verification_sent.html:6
+msgid "Verify your email address"
+msgstr "Verifizieren Sie Ihre E-Mail-Adresse"
+
+#: aleksis/core/templates/account/verification_sent.html:14
+msgid "Verify your email!"
+msgstr "Verifizieren Sie Ihre E-Mail!"
+
+#: aleksis/core/templates/account/verification_sent.html:16
+msgid ""
+"\n"
+"            This part of the site requires us to verify that you are who you claim to be.\n"
+"            For this purpose, we require that you verify ownership of your e-mail address.\n"
+"          "
+msgstr ""
+"\n"
+"            Dieser Teil der Anwendung setzt voraus, dass wir verifizieren, dass Sie die Person sind, die sie vorgeben, zu sein.\n"
+"Zu diesem Zweck setzen wir voraus, dass Sie die Inhaberschaft Ihrer E-Mail-Adresse bestätigen.\n"
+"          "
+
+#: aleksis/core/templates/account/verification_sent.html:22
+msgid ""
+"\n"
+"            We have sent an e-mail to you for verification.\n"
+"            Please click on the link inside this e-mail. Please\n"
+"            contact us if you do not receive it within a few minutes.\n"
+"          "
+msgstr ""
+"\n"
+"            Wir haben Ihnen eine E-Mail zur Verifizierung geschickt.\n"
+"Bitte klicken Sie auf den Link in dieser E-Mail.\n"
+"Bitte kontaktieren Sie uns, wenn Sie diese nicht binnen weniger Minuten erhalten.\n"
+"          "
+
+#: aleksis/core/templates/account/verification_sent.html:30
+#, python-format
+msgid "<strong>Note:</strong> you can still <a href=\"%(email_url)s\">change your e-mail address</a>"
+msgstr "<strong>Hinweis:</strong> Sie können immer noch <a href=\"%(email_url)s\"> Ihre E-Mail-Adresse ändern</a>"
+
+#: aleksis/core/templates/core/additional_field/edit.html:6
+#: aleksis/core/templates/core/additional_field/edit.html:7
 msgid "Edit additional field"
 msgstr "Zusätzliches Feld bearbeiten"
 
-#: templates/core/additional_field/list.html:14
+#: aleksis/core/templates/core/additional_field/list.html:14
 msgid "Create additional field"
 msgstr "Zusätzliches Feld erstellen"
 
-#: templates/core/announcement/form.html:14
-#: templates/core/announcement/form.html:21
+#: aleksis/core/templates/core/announcement/form.html:14
+#: aleksis/core/templates/core/announcement/form.html:21
 msgid "Edit announcement"
 msgstr "Ankündigung bearbeiten"
 
-#: templates/core/announcement/form.html:16
+#: aleksis/core/templates/core/announcement/form.html:16
 msgid "Publish announcement"
 msgstr "Ankündigung veröffentlichen"
 
-#: templates/core/announcement/form.html:23
-#: templates/core/announcement/list.html:13
+#: aleksis/core/templates/core/announcement/form.html:23
+#: aleksis/core/templates/core/announcement/list.html:13
 msgid "Publish new announcement"
 msgstr "Neue Ankündigung veröffentlichen"
 
-#: templates/core/announcement/form.html:34
+#: aleksis/core/templates/core/announcement/form.html:34
 msgid "Save und publish announcement"
 msgstr "Ankündigung speichern und veröffentlichen"
 
-#: templates/core/announcement/list.html:19
+#: aleksis/core/templates/core/announcement/list.html:19
 msgid "Valid from"
 msgstr "Gültig von"
 
-#: templates/core/announcement/list.html:20
+#: aleksis/core/templates/core/announcement/list.html:20
 msgid "Valid until"
 msgstr "Gültig bis"
 
-#: templates/core/announcement/list.html:21
+#: aleksis/core/templates/core/announcement/list.html:21
 msgid "Recipients"
 msgstr "Empfänger"
 
-#: templates/core/announcement/list.html:50
+#: aleksis/core/templates/core/announcement/list.html:50
 msgid "There are no announcements."
 msgstr "Es gibt aktuell keine Ankündigungen."
 
-#: templates/core/base.html:54
+#: aleksis/core/templates/core/base.html:60
 msgid "Logged in as"
 msgstr "Angemeldet als"
 
-#: templates/core/base.html:147
+#: aleksis/core/templates/core/base.html:154
 msgid "About AlekSIS — The Free School Information System"
 msgstr "Über AlekSIS — The Free School Information System"
 
-#: templates/core/base.html:155
+#: aleksis/core/templates/core/base.html:162
 msgid "Impress"
 msgstr "Impressum"
 
-#: templates/core/base.html:163
+#: aleksis/core/templates/core/base.html:170
 msgid "Privacy Policy"
 msgstr "Datenschutzerklärung"
 
-#: templates/core/base_print.html:64
+#: aleksis/core/templates/core/base_print.html:72
 msgid "Powered by AlekSIS"
 msgstr "Betrieben mit AlekSIS"
 
-#: templates/core/dashboard_widget/create.html:8
-#: templates/core/dashboard_widget/create.html:12
+#: aleksis/core/templates/core/dashboard_widget/create.html:8
+#: aleksis/core/templates/core/dashboard_widget/create.html:12
 #, python-format
 msgid "Create %(widget)s"
 msgstr "%(widget)s erstellen"
 
-#: templates/core/dashboard_widget/edit.html:8
-#: templates/core/dashboard_widget/edit.html:12
+#: aleksis/core/templates/core/dashboard_widget/dashboardwidget_broken.html:10
+msgid "This widget is currently not available."
+msgstr "Das Widget ist aktuell nicht verfügbar."
+
+#: aleksis/core/templates/core/dashboard_widget/dashboardwidget_broken.html:14
+#, python-format
+msgid ""
+"\n"
+"            There is a problem getting the widget \"%(title)s\".\n"
+"            There is no need for you to take any action.\n"
+"          "
+msgstr ""
+"\n"
+"            Es ist ein Problem dabei aufgetreten, das Widget \"%(title)s\" zu laden.\n"
+"Sie brauchen nichts weiter machen.\n"
+"          "
+
+#: aleksis/core/templates/core/dashboard_widget/edit.html:8
+#: aleksis/core/templates/core/dashboard_widget/edit.html:12
 #, python-format
 msgid "Edit %(widget)s"
 msgstr "%(widget)s bearbeiten"
 
-#: templates/core/dashboard_widget/list.html:17
+#: aleksis/core/templates/core/dashboard_widget/list.html:15
+msgid "Create dashboard widget"
+msgstr "Dashboard-Widget erstellen"
+
+#: aleksis/core/templates/core/dashboard_widget/list.html:22
 #, python-format
 msgid "Create %(name)s"
 msgstr "%(name)s erstellen"
 
-#: templates/core/dashboard_widget/list.html:25
-#: templates/core/edit_dashboard.html:8 templates/core/edit_dashboard.html:15
+#: aleksis/core/templates/core/dashboard_widget/list.html:32
+#: aleksis/core/templates/core/edit_dashboard.html:8
+#: aleksis/core/templates/core/edit_dashboard.html:15
 msgid "Edit default dashboard"
 msgstr "Standard-Dashboard bearbeiten"
 
-#: templates/core/data_check/list.html:15
+#: aleksis/core/templates/core/data_check/list.html:15
 msgid "Check data again"
 msgstr "Daten erneut prüfen"
 
-#: templates/core/data_check/list.html:22
+#: aleksis/core/templates/core/data_check/list.html:22
 msgid "The system detected some problems with your data."
 msgstr "Das System hat einige Problemen mit Ihren Daten gefunden."
 
-#: templates/core/data_check/list.html:23
+#: aleksis/core/templates/core/data_check/list.html:23
 msgid ""
 "Please go through all data and check whether some extra action is\n"
 "          needed."
@@ -1007,43 +1565,43 @@ msgstr ""
 "Bitte gehen Sie alle Daten durch und prüfen Sie, ob weitere Aktionen\n"
 "notwendig sind."
 
-#: templates/core/data_check/list.html:31
+#: aleksis/core/templates/core/data_check/list.html:31
 msgid "Everything is fine."
 msgstr "Alles ist gut."
 
-#: templates/core/data_check/list.html:32
+#: aleksis/core/templates/core/data_check/list.html:32
 msgid "The system hasn't detected any problems with your data."
 msgstr "Das System hat keine Probleme mit Ihren Daten entdeckt."
 
-#: templates/core/data_check/list.html:40
+#: aleksis/core/templates/core/data_check/list.html:40
 msgid "Detected problems"
 msgstr "Gefundene Probleme"
 
-#: templates/core/data_check/list.html:45
+#: aleksis/core/templates/core/data_check/list.html:45
 msgid "Affected object"
 msgstr "Betroffenes Objekt"
 
-#: templates/core/data_check/list.html:46
+#: aleksis/core/templates/core/data_check/list.html:46
 msgid "Detected problem"
 msgstr "Entdecktes Problem"
 
-#: templates/core/data_check/list.html:47
+#: aleksis/core/templates/core/data_check/list.html:47
 msgid "Show details"
 msgstr "Details anzeigen"
 
-#: templates/core/data_check/list.html:48
+#: aleksis/core/templates/core/data_check/list.html:48
 msgid "Options to solve the problem"
 msgstr "Optionen, das Problem zu lösen"
 
-#: templates/core/data_check/list.html:62
+#: aleksis/core/templates/core/data_check/list.html:62
 msgid "Show object"
 msgstr "Objekt anzeigen"
 
-#: templates/core/data_check/list.html:84
+#: aleksis/core/templates/core/data_check/list.html:84
 msgid "Registered checks"
 msgstr "Registrierte Prüfungen"
 
-#: templates/core/data_check/list.html:88
+#: aleksis/core/templates/core/data_check/list.html:88
 msgid ""
 "\n"
 "            The system will check for the following problems:\n"
@@ -1053,12 +1611,13 @@ msgstr ""
 "            Das System wird nach folgenden Problemen suchen:\n"
 "          "
 
-#: templates/core/edit_dashboard.html:6 templates/core/edit_dashboard.html:13
-#: templates/core/index.html:14
+#: aleksis/core/templates/core/edit_dashboard.html:6
+#: aleksis/core/templates/core/edit_dashboard.html:13
+#: aleksis/core/templates/core/index.html:17
 msgid "Edit dashboard"
 msgstr "Dashboard bearbeiten"
 
-#: templates/core/edit_dashboard.html:24
+#: aleksis/core/templates/core/edit_dashboard.html:24
 msgid ""
 "\n"
 "          On this page you can arrange your personal dashboard. You can drag any items from \"Available widgets\" to \"Your\n"
@@ -1067,15 +1626,12 @@ msgid ""
 "        "
 msgstr ""
 "\n"
-"          Auf dieser Seite können Sie Ihr persönliches Dashboard "
-"zusammenstallen. Sie können beliebige Elemente von den \"Verfügbaren "
-"Widgets\" \n"
-"in \"Ihr Dashboard\" ziehen oder die Reihenfolge verändern, indem Sie die "
-"Widgets bewegen. Wenn Sie fertig sind, vergessen Sie bitte nicht, \n"
+"          Auf dieser Seite können Sie Ihr persönliches Dashboard zusammenstallen. Sie können beliebige Elemente von den \"Verfügbaren Widgets\" \n"
+"in \"Ihr Dashboard\" ziehen oder die Reihenfolge verändern, indem Sie die Widgets bewegen. Wenn Sie fertig sind, vergessen Sie bitte nicht, \n"
 "auf \"Speichern\" zu drücken.\n"
 "        "
 
-#: templates/core/edit_dashboard.html:30
+#: aleksis/core/templates/core/edit_dashboard.html:30
 msgid ""
 "\n"
 "          On this page you can arrange the default dashboard which is shown when a user doesn't arrange his own\n"
@@ -1084,29 +1640,24 @@ msgid ""
 "        "
 msgstr ""
 "\n"
-"          Auf dieser Seite können Sie Ihr das Standard-Dashboard "
-"zusammenstallen, welches angezeigt wird, wenn ein Nutzer kein eigenes "
-"definiert. \n"
-"Sie können beliebige Elemente von den \"Verfügbaren Widgets\" in \"Standard-"
-"Dashboard\" ziehen oder die Reihenfolge verändern, indem Sie die Widgets "
-"bewegen. \n"
-"Wenn Sie fertig sind, vergessen Sie bitte nicht, auf \"Speichern\" zu "
-"drücken.\n"
+"          Auf dieser Seite können Sie Ihr das Standard-Dashboard zusammenstallen, welches angezeigt wird, wenn ein Benutzer kein eigenes definiert. \n"
+"Sie können beliebige Elemente von den \"Verfügbaren Widgets\" in \"Standard-Dashboard\" ziehen oder die Reihenfolge verändern, indem Sie die Widgets bewegen. \n"
+"Wenn Sie fertig sind, vergessen Sie bitte nicht, auf \"Speichern\" zu drücken.\n"
 "        "
 
-#: templates/core/edit_dashboard.html:48
+#: aleksis/core/templates/core/edit_dashboard.html:48
 msgid "Available widgets"
 msgstr "Verfügbare Widgets"
 
-#: templates/core/edit_dashboard.html:57
+#: aleksis/core/templates/core/edit_dashboard.html:57
 msgid "Your dashboard"
 msgstr "Ihr Dashboard"
 
-#: templates/core/edit_dashboard.html:59
+#: aleksis/core/templates/core/edit_dashboard.html:59
 msgid "Default dashboard"
 msgstr "Standard-Dashboard"
 
-#: templates/core/group/child_groups.html:18
+#: aleksis/core/templates/core/group/child_groups.html:18
 msgid ""
 "\n"
 "          You can use this to assign child groups to groups. Please use the filters below to select groups you want to\n"
@@ -1118,23 +1669,23 @@ msgstr ""
 "          ändern möchten und klicken auf \"Weiter\".\n"
 "        "
 
-#: templates/core/group/child_groups.html:31
+#: aleksis/core/templates/core/group/child_groups.html:31
 msgid "Update selection"
 msgstr "Auswahl aktualisieren"
 
-#: templates/core/group/child_groups.html:35
+#: aleksis/core/templates/core/group/child_groups.html:35
 msgid "Clear all filters"
-msgstr "Alle Filter leeren"
+msgstr "Alle Filter zurücksetzen"
 
-#: templates/core/group/child_groups.html:39
+#: aleksis/core/templates/core/group/child_groups.html:39
 msgid "Currently selected groups"
 msgstr "Aktuell ausgewählte Gruppen"
 
-#: templates/core/group/child_groups.html:52
+#: aleksis/core/templates/core/group/child_groups.html:52
 msgid "Start assigning child groups for this groups"
 msgstr "Zuordnung von Kindgruppen zu Gruppen starten"
 
-#: templates/core/group/child_groups.html:61
+#: aleksis/core/templates/core/group/child_groups.html:61
 msgid ""
 "\n"
 "            Please select some groups in order to go on with assigning.\n"
@@ -1144,15 +1695,15 @@ msgstr ""
 "            Bitte wählen Sie Gruppen aus, um Gruppen zuzuordnen.\n"
 "          "
 
-#: templates/core/group/child_groups.html:72
+#: aleksis/core/templates/core/group/child_groups.html:72
 msgid "Current group:"
 msgstr "Aktuelle Gruppe:"
 
-#: templates/core/group/child_groups.html:78
+#: aleksis/core/templates/core/group/child_groups.html:78
 msgid "Please be careful!"
 msgstr "Bitte seien Sie vorsichtig!"
 
-#: templates/core/group/child_groups.html:79
+#: aleksis/core/templates/core/group/child_groups.html:79
 msgid ""
 "\n"
 "            If you click \"Back\" or \"Next\" the current group assignments are not saved.\n"
@@ -1166,133 +1717,139 @@ msgstr ""
 "mit dem überschrieben, was Sie auf dieser Seite ausgewählt haben.\n"
 "          "
 
-#: templates/core/group/child_groups.html:93
-#: templates/core/group/child_groups.html:128
-#: templates/two_factor/_wizard_actions.html:15
-#: templates/two_factor/_wizard_actions.html:20
+#: aleksis/core/templates/core/group/child_groups.html:93
+#: aleksis/core/templates/core/group/child_groups.html:128
+#: aleksis/core/templates/oauth2_provider/application_detail.html:9
+#: aleksis/core/templates/two_factor/_wizard_actions.html:15
+#: aleksis/core/templates/two_factor/_wizard_actions.html:20
 msgid "Back"
 msgstr "Zurück"
 
-#: templates/core/group/child_groups.html:99
-#: templates/core/group/child_groups.html:134
-#: templates/two_factor/_wizard_actions.html:26
+#: aleksis/core/templates/core/group/child_groups.html:99
+#: aleksis/core/templates/core/group/child_groups.html:134
+#: aleksis/core/templates/two_factor/_wizard_actions.html:26
 msgid "Next"
 msgstr "Weiter"
 
-#: templates/core/group/child_groups.html:106
-#: templates/core/group/child_groups.html:141
-#: templates/core/partials/save_button.html:3
+#: aleksis/core/templates/core/group/child_groups.html:106
+#: aleksis/core/templates/core/group/child_groups.html:141
+#: aleksis/core/templates/core/partials/save_button.html:3
 msgid "Save"
 msgstr "Speichern"
 
-#: templates/core/group/child_groups.html:112
-#: templates/core/group/child_groups.html:147
+#: aleksis/core/templates/core/group/child_groups.html:112
+#: aleksis/core/templates/core/group/child_groups.html:147
 msgid "Save and next"
 msgstr "Speichern und weiter"
 
-#: templates/core/group/edit.html:11 templates/core/group/edit.html:12
+#: aleksis/core/templates/core/group/edit.html:11
+#: aleksis/core/templates/core/group/edit.html:12
 msgid "Edit group"
 msgstr "Gruppe editieren"
 
-#: templates/core/group/full.html:38 templates/core/person/full.html:37
+#: aleksis/core/templates/core/group/full.html:38
+#: aleksis/core/templates/core/person/full.html:37
 msgid "Change preferences"
 msgstr "Einstellungen ändern"
 
-#: templates/core/group/full.html:64
+#: aleksis/core/templates/core/group/full.html:64
 msgid "Statistics"
 msgstr "Statistiken"
 
-#: templates/core/group/full.html:67
+#: aleksis/core/templates/core/group/full.html:67
 msgid "Count of members"
 msgstr "Anzahl der Mitglieder"
 
-#: templates/core/group/full.html:71
+#: aleksis/core/templates/core/group/full.html:71
 msgid "Average age"
 msgstr "Durchschnittsalter"
 
-#: templates/core/group/full.html:76
+#: aleksis/core/templates/core/group/full.html:76
 msgid "Age range"
 msgstr "Altersbereich"
 
-#: templates/core/group/full.html:76
+#: aleksis/core/templates/core/group/full.html:76
 msgid "years to"
 msgstr "Jahre bis"
 
-#: templates/core/group/full.html:76
+#: aleksis/core/templates/core/group/full.html:76
 msgid "years "
 msgstr "Jahre "
 
-#: templates/core/group/list.html:14
+#: aleksis/core/templates/core/group/list.html:14
 msgid "Create group"
 msgstr "Gruppe erstellen"
 
-#: templates/core/group/list.html:17
+#: aleksis/core/templates/core/group/list.html:17
 msgid "Filter groups"
 msgstr "Gruppen filtern"
 
-#: templates/core/group/list.html:24 templates/core/person/list.html:28
+#: aleksis/core/templates/core/group/list.html:24
+#: aleksis/core/templates/core/person/list.html:28
 msgid "Clear"
 msgstr "Zurücksetzen"
 
-#: templates/core/group/list.html:28
+#: aleksis/core/templates/core/group/list.html:28
 msgid "Selected groups"
 msgstr "Ausgewählte Gruppen"
 
-#: templates/core/group_type/edit.html:6 templates/core/group_type/edit.html:7
+#: aleksis/core/templates/core/group_type/edit.html:6
+#: aleksis/core/templates/core/group_type/edit.html:7
 msgid "Edit group type"
 msgstr "Gruppentyp editieren"
 
-#: templates/core/group_type/list.html:14
+#: aleksis/core/templates/core/group_type/list.html:14
 msgid "Create group type"
 msgstr "Gruppentyp erstellen"
 
-#: templates/core/index.html:4
+#: aleksis/core/templates/core/index.html:4
 msgid "Home"
 msgstr "Startseite"
 
-#: templates/core/index.html:50
+#: aleksis/core/templates/core/index.html:49
 msgid ""
 "\n"
-"          You didn't customise your dashboard so that you see the system default. Please click on \"Edit dashboard\" to\n"
-"          customise your personal dashboard.\n"
-"        "
+"        You didn't customise your dashboard so that you see the system default. Please click on \"Edit dashboard\" to\n"
+"        customise your personal dashboard.\n"
+"      "
 msgstr ""
 "\n"
-"          Sie haben Ihr Dashboard nicht angepasst, sodass Sie das Standard-"
-"Dashboard sehen.\n"
-"Bitte klicken Sie auf \"Dashboard bearbeiten\", um Ihr persönliches "
-"Dashboard anzupassen.\n"
-"        "
+"        Sie haben Ihr Dashboard nicht angepasst, sodass Sie das Standard-Dashboard sehen.\n"
+"Bitte klicken Sie auf \"Dashboard bearbeiten\", um Ihr persönliches Dashboard anzupassen.\n"
+"      "
 
-#: templates/core/index.html:59
+#: aleksis/core/templates/core/index.html:59
 msgid "Last activities"
 msgstr "Letzte Aktivitäten"
 
-#: templates/core/index.html:77
+#: aleksis/core/templates/core/index.html:77
 msgid "No activities available yet."
 msgstr "Aktuell keine Aktivitäten verfügbar."
 
-#: templates/core/index.html:82
+#: aleksis/core/templates/core/index.html:82
 msgid "Recent notifications"
 msgstr "Letzte Benachrichtigungen"
 
-#: templates/core/index.html:98
+#: aleksis/core/templates/core/index.html:98
+#: aleksis/core/templates/core/notifications.html:23
 msgid "More information →"
 msgstr "Mehr Informationen →"
 
-#: templates/core/index.html:105
+#: aleksis/core/templates/core/index.html:105
+#: aleksis/core/templates/core/notifications.html:30
 msgid "No notifications available yet."
 msgstr "Aktuell keine Benachrichtigungen verfügbar."
 
-#: templates/core/pages/about.html:6 templates/core/pages/about.html:15
+#: aleksis/core/templates/core/pages/about.html:6
+#: aleksis/core/templates/core/pages/about.html:15
 msgid "About AlekSIS"
 msgstr "Ãœber AlekSIS"
 
-#: templates/core/pages/about.html:7
+#: aleksis/core/templates/core/pages/about.html:7
 msgid "AlekSIS – The Free School Information System"
 msgstr "AlekSIS – The Free School Information System"
 
-#: templates/core/pages/about.html:17
+#: aleksis/core/templates/core/pages/about.html:17
 msgid ""
 "\n"
 "              This platform is powered by AlekSIS, a web-based school information system (SIS) which can be used\n"
@@ -1306,19 +1863,19 @@ msgstr ""
 "AlekSIS ist freie Software und kann von jedem benutzt werden.\n"
 "            "
 
-#: templates/core/pages/about.html:25
+#: aleksis/core/templates/core/pages/about.html:25
 msgid "Website of AlekSIS"
 msgstr "Website von AlekSIS"
 
-#: templates/core/pages/about.html:26
+#: aleksis/core/templates/core/pages/about.html:26
 msgid "Source code"
 msgstr "Quellcode"
 
-#: templates/core/pages/about.html:35
+#: aleksis/core/templates/core/pages/about.html:35
 msgid "Licence information"
 msgstr "Lizenzinformationen"
 
-#: templates/core/pages/about.html:37
+#: aleksis/core/templates/core/pages/about.html:37
 msgid ""
 "\n"
 "              The core and the official apps of AlekSIS are licenced under the EUPL, version 1.2 or later. For licence\n"
@@ -1332,23 +1889,23 @@ msgstr ""
 "sind wie folgt markiert:\n"
 "            "
 
-#: templates/core/pages/about.html:45
+#: aleksis/core/templates/core/pages/about.html:45
 msgid "Free/Open Source Licence"
 msgstr "Freie/Open Source Lizenz"
 
-#: templates/core/pages/about.html:46
+#: aleksis/core/templates/core/pages/about.html:46
 msgid "Other Licence"
 msgstr "Andere Lizenz"
 
-#: templates/core/pages/about.html:50
+#: aleksis/core/templates/core/pages/about.html:50
 msgid "Full licence text"
 msgstr "Kompletter Lizenztext"
 
-#: templates/core/pages/about.html:51
+#: aleksis/core/templates/core/pages/about.html:51
 msgid "More information about the EUPL"
 msgstr "Weitere Informationen über die EUPL"
 
-#: templates/core/pages/about.html:90
+#: aleksis/core/templates/core/pages/about.html:90
 #, python-format
 msgid ""
 "\n"
@@ -1359,12 +1916,12 @@ msgstr ""
 "                    Diese App ist unter %(licence)s lizenziert.\n"
 "                  "
 
-#: templates/core/pages/delete.html:6
+#: aleksis/core/templates/core/pages/delete.html:6
 #, python-format
 msgid "Delete %(object_name)s"
 msgstr "%(object_name)s löschen"
 
-#: templates/core/pages/delete.html:13
+#: aleksis/core/templates/core/pages/delete.html:13
 #, python-format
 msgid ""
 "\n"
@@ -1375,7 +1932,7 @@ msgstr ""
 "      Möchten Sie wirklich %(object_name)s \"%(object)s\" löschen?\n"
 "    "
 
-#: templates/core/pages/progress.html:27
+#: aleksis/core/templates/core/pages/progress.html:27
 msgid ""
 "\n"
 "              Without activated JavaScript the progress status can't be updated.\n"
@@ -1385,20 +1942,20 @@ msgstr ""
 "              Ohne aktiviertes JavaScript kann der Fortschritt leider nicht aktualisiert werden.\n"
 "            "
 
-#: templates/core/pages/progress.html:47
-#: templates/two_factor/core/otp_required.html:19
+#: aleksis/core/templates/core/pages/progress.html:47
+#: aleksis/core/templates/two_factor/core/otp_required.html:19
 msgid "Go back"
 msgstr "Zurück"
 
-#: templates/core/pages/system_status.html:12
+#: aleksis/core/templates/core/pages/system_status.html:12
 msgid "System checks"
 msgstr "Systemprüfungen"
 
-#: templates/core/pages/system_status.html:21
+#: aleksis/core/templates/core/pages/system_status.html:21
 msgid "Maintenance mode enabled"
 msgstr "Wartungsmodus aktiviert"
 
-#: templates/core/pages/system_status.html:23
+#: aleksis/core/templates/core/pages/system_status.html:23
 msgid ""
 "\n"
 "                Only admin and visitors from internal IPs can access thesite.\n"
@@ -1408,19 +1965,19 @@ msgstr ""
 "                Nur Administratoren und Besucher von internen IP-Adressen können die Seite aufrufen.\n"
 "              "
 
-#: templates/core/pages/system_status.html:34
+#: aleksis/core/templates/core/pages/system_status.html:34
 msgid "Maintenance mode disabled"
 msgstr "Wartungsmodus deaktiviert"
 
-#: templates/core/pages/system_status.html:35
+#: aleksis/core/templates/core/pages/system_status.html:35
 msgid "Everyone can access the site."
 msgstr "Jeder kann die Seite aufrufen."
 
-#: templates/core/pages/system_status.html:45
+#: aleksis/core/templates/core/pages/system_status.html:45
 msgid "Debug mode enabled"
 msgstr "Debug-Modus aktiviert"
 
-#: templates/core/pages/system_status.html:47
+#: aleksis/core/templates/core/pages/system_status.html:47
 msgid ""
 "\n"
 "                The web server throws back debug information on errors. Do not use in production!\n"
@@ -1430,11 +1987,11 @@ msgstr ""
 "                Der Server gibt Debug-Informationen bei Fehlern zurück. Nicht im Produktivbetrieb nutzen!\n"
 "              "
 
-#: templates/core/pages/system_status.html:54
+#: aleksis/core/templates/core/pages/system_status.html:54
 msgid "Debug mode disabled"
 msgstr "Debug-Modus deaktivert"
 
-#: templates/core/pages/system_status.html:56
+#: aleksis/core/templates/core/pages/system_status.html:56
 msgid ""
 "\n"
 "                Debug mode is disabled. Default error pages are displayed on errors.\n"
@@ -1444,45 +2001,60 @@ msgstr ""
 "                Debug-Modus ist deaktiviert. Standard-Fehlerseiten werden bei Fehlern angezeigt.\n"
 "              "
 
-#: templates/core/pages/system_status.html:69
+#: aleksis/core/templates/core/pages/system_status.html:69
 msgid "System health checks"
 msgstr "Systemprüfungen"
 
-#: templates/core/pages/system_status.html:75
+#: aleksis/core/templates/core/pages/system_status.html:75
 msgid "Service"
 msgstr "Dienst"
 
-#: templates/core/pages/system_status.html:76
-#: templates/core/pages/system_status.html:115
+#: aleksis/core/templates/core/pages/system_status.html:76
+#: aleksis/core/templates/core/pages/system_status.html:115
 msgid "Status"
 msgstr "Status"
 
-#: templates/core/pages/system_status.html:77
+#: aleksis/core/templates/core/pages/system_status.html:77
 msgid "Time taken"
 msgstr "Dauer"
 
-#: templates/core/pages/system_status.html:96
+#: aleksis/core/templates/core/pages/system_status.html:96
 msgid "seconds"
 msgstr "Sekunden"
 
-#: templates/core/pages/system_status.html:107
+#: aleksis/core/templates/core/pages/system_status.html:107
 msgid "Celery task results"
 msgstr "Celery Task-Ergebnisse"
 
-#: templates/core/pages/system_status.html:112
+#: aleksis/core/templates/core/pages/system_status.html:112
 msgid "Task"
 msgstr "Task"
 
-#: templates/core/pages/system_status.html:113
+#: aleksis/core/templates/core/pages/system_status.html:113
 msgid "ID"
 msgstr "ID"
 
-#: templates/core/pages/system_status.html:114
+#: aleksis/core/templates/core/pages/system_status.html:114
 msgid "Date done"
 msgstr "Erledigungszeitpunkt"
 
-#: templates/core/partials/announcements.html:9
-#: templates/core/partials/announcements.html:36
+#: aleksis/core/templates/core/pages/test_pdf.html:7
+#: aleksis/core/templates/core/pages/test_pdf.html:8
+msgid "Test PDF generation"
+msgstr "PDF-Generierung testen"
+
+#: aleksis/core/templates/core/pages/test_pdf.html:14
+msgid ""
+"\n"
+"        This simple view can be used to ensure the correct function of the built-in PDF generation system.\n"
+"      "
+msgstr ""
+"\n"
+"        Diese einfache Seite kann genutzt werden, um die korrekte Funktionalität des eingebauten PDF-Generierungssystem zu testen.\n"
+"      "
+
+#: aleksis/core/templates/core/partials/announcements.html:8
+#: aleksis/core/templates/core/partials/announcements.html:35
 #, python-format
 msgid ""
 "\n"
@@ -1493,7 +2065,7 @@ msgstr ""
 "              Gültig für %(from)s\n"
 "            "
 
-#: templates/core/partials/announcements.html:13
+#: aleksis/core/templates/core/partials/announcements.html:12
 #, python-format
 msgid ""
 "\n"
@@ -1504,7 +2076,7 @@ msgstr ""
 "              Gültig von %(from)s bis %(until)s\n"
 "            "
 
-#: templates/core/partials/announcements.html:40
+#: aleksis/core/templates/core/partials/announcements.html:39
 #, python-format
 msgid ""
 "\n"
@@ -1515,23 +2087,23 @@ msgstr ""
 "              Gültig von %(from)s – %(until)s\n"
 "            "
 
-#: templates/core/partials/crud_events.html:15
+#: aleksis/core/templates/core/partials/crud_events.html:15
 msgid "Changed by"
 msgstr "Verändert von"
 
-#: templates/core/partials/crud_events.html:15
+#: aleksis/core/templates/core/partials/crud_events.html:15
 msgid "Unknown"
 msgstr "Unbekannt"
 
-#: templates/core/partials/language_form.html:15
+#: aleksis/core/templates/core/partials/language_form.html:15
 msgid "Language"
 msgstr "Sprache"
 
-#: templates/core/partials/language_form.html:27
+#: aleksis/core/templates/core/partials/language_form.html:27
 msgid "Select language"
 msgstr "Sprache auswählen"
 
-#: templates/core/partials/no_person.html:12
+#: aleksis/core/templates/core/partials/no_person.html:12
 msgid ""
 "\n"
 "            Your administrator account is not linked to any person. Therefore,\n"
@@ -1543,7 +2115,7 @@ msgstr ""
 "            wurde Ihr Konto mit einer Dummyperson verknüpft.\n"
 "          "
 
-#: templates/core/partials/no_person.html:19
+#: aleksis/core/templates/core/partials/no_person.html:19
 msgid ""
 "\n"
 "            Your user account is not linked to a person. This means you\n"
@@ -1557,105 +2129,172 @@ msgstr ""
 "        die Verwaltenden von AlekSIS an Ihrer Schule.\n"
 "          "
 
-#: templates/core/person/accounts.html:12
-#: templates/core/person/accounts.html:14
-msgid "Link persons to accounts"
-msgstr "Personen mit Benutzerkonten verknüpfen"
-
-#: templates/core/person/accounts.html:21
-msgid ""
-"\n"
-"        You can use this form to assign user accounts to persons. Use the\n"
-"        dropdowns to select existing accounts; use the text fields to create new\n"
-"        accounts on-the-fly. The latter will create a new account with the\n"
-"        entered username and copy all other details from the person.\n"
-"      "
-msgstr ""
-"\n"
-"        Sie können dieses Formular nutzen, um Benutzerkonten Personen zuzuweisen. Nutzen Sie das\n"
-"    Auswahlfeld um ein existierendes Benutzerkonto auszuwählen; nutzen Sie das Textfeld, um einen neuen Benutzer zu\n"
-"    erstellen. Letzteres erstellt ein neues Benutzerkonto mit dem\n"
-"    eingegebenen Benutzernamen und kopiert alle anderen Daten der Person.\n"
-"      "
-
-#: templates/core/person/accounts.html:36
-#: templates/core/person/accounts.html:60
-msgid "Update"
-msgstr "Aktualisieren"
-
-#: templates/core/person/accounts.html:42
-msgid "Existing account"
-msgstr "Existierendes Konto"
-
-#: templates/core/person/accounts.html:43
-msgid "New account"
-msgstr "Neues Konto"
+#: aleksis/core/templates/core/person/create.html:12
+#: aleksis/core/templates/core/person/create.html:13
+#: aleksis/core/templates/core/person/list.html:17
+msgid "Create person"
+msgstr "Person erstellen"
 
-#: templates/core/person/edit.html:12 templates/core/person/edit.html:13
+#: aleksis/core/templates/core/person/edit.html:12
+#: aleksis/core/templates/core/person/edit.html:13
 msgid "Edit person"
 msgstr "Person editieren"
 
-#: templates/core/person/full.html:44
+#: aleksis/core/templates/core/person/full.html:44
+#: aleksis/core/templates/impersonate/list_users.html:7
+#: aleksis/core/templates/impersonate/list_users.html:8
 msgid "Impersonate"
 msgstr "Verkleiden"
 
-#: templates/core/person/full.html:50
+#: aleksis/core/templates/core/person/full.html:50
 msgid "Contact details"
 msgstr "Kontaktdetails"
 
-#: templates/core/person/full.html:130
+#: aleksis/core/templates/core/person/full.html:130
 msgid "Children"
 msgstr "Kinder"
 
-#: templates/core/person/list.html:17
-msgid "Create person"
-msgstr "Person erstellen"
-
-#: templates/core/person/list.html:21
+#: aleksis/core/templates/core/person/list.html:21
 msgid "Filter persons"
 msgstr "Personen filtern"
 
-#: templates/core/person/list.html:32
+#: aleksis/core/templates/core/person/list.html:32
 msgid "Selected persons"
 msgstr "Ausgewählte Personen"
 
-#: templates/core/school_term/create.html:6
-#: templates/core/school_term/create.html:7
-#: templates/core/school_term/list.html:14
+#: aleksis/core/templates/core/school_term/create.html:6
+#: aleksis/core/templates/core/school_term/create.html:7
+#: aleksis/core/templates/core/school_term/list.html:14
 msgid "Create school term"
 msgstr "Schuljahr erstellen"
 
-#: templates/core/school_term/edit.html:6
-#: templates/core/school_term/edit.html:7
+#: aleksis/core/templates/core/school_term/edit.html:6
+#: aleksis/core/templates/core/school_term/edit.html:7
 msgid "Edit school term"
 msgstr "Schuljahr bearbeiten"
 
-#: templates/dynamic_preferences/form.html:9
+#: aleksis/core/templates/dynamic_preferences/form.html:9
 msgid "Site preferences"
 msgstr "Seiteneinstellungen ändern"
 
-#: templates/dynamic_preferences/form.html:11
+#: aleksis/core/templates/dynamic_preferences/form.html:11
 msgid "My preferences"
 msgstr "Meine Einstellungen"
 
-#: templates/dynamic_preferences/form.html:13
+#: aleksis/core/templates/dynamic_preferences/form.html:13
 #, python-format
 msgid "Preferences for %(instance)s"
 msgstr "Einstellungen für %(instance)s"
 
-#: templates/dynamic_preferences/form.html:25
+#: aleksis/core/templates/dynamic_preferences/form.html:25
 msgid "Save preferences"
 msgstr "Einstellungen speichern"
 
-#: templates/impersonate/list_users.html:8
-msgid "Impersonate user"
-msgstr "Als Benutzer verkleiden"
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:5
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:6
+msgid "Delete application"
+msgstr "Anwendung löschen"
+
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:12
+#, python-format
+msgid "Are you sure to delete the application %(application_name)s?"
+msgstr "Sind Sie sicher, dass Sie die Anwendung %(application_name)s löschen möchten?"
+
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:24
+#: aleksis/core/templates/oauth2_provider/application_form.html:18
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:24
+#: aleksis/core/templates/two_factor/_wizard_actions.html:6
+msgid "Cancel"
+msgstr "Abbrechen"
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:27
+msgid "Client id"
+msgstr "Client-ID"
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:35
+msgid "Client secret"
+msgstr "Client-Secret"
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:43
+msgid "Client type"
+msgstr "Client-Typ"
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:51
+msgid "Authorization Grant Type"
+msgstr "Authorization Grant-Typ"
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:59
+msgid "Redirect URIs"
+msgstr "Weiterleitungs-URLs"
+
+#: aleksis/core/templates/oauth2_provider/application_form.html:5
+msgid "Create OAuth2 Application"
+msgstr "OAuth2-Anwendung erstellen"
+
+#: aleksis/core/templates/oauth2_provider/application_form.html:10
+msgid "Edit application"
+msgstr "Anwendung bearbeiten"
+
+#: aleksis/core/templates/oauth2_provider/application_list.html:8
+msgid "OAuth2 applications"
+msgstr "OAuth2-Anwendungen"
+
+#: aleksis/core/templates/oauth2_provider/application_list.html:12
+msgid "Register new applications"
+msgstr "Neue Anwendungen registrieren"
+
+#: aleksis/core/templates/oauth2_provider/application_list.html:23
+msgid "No applications defined."
+msgstr "Keine Anwendungen definiert."
 
-#: templates/offline.html:5
+#: aleksis/core/templates/oauth2_provider/authorize.html:5
+#: aleksis/core/templates/oauth2_provider/authorize.html:16
+msgid "Authorize"
+msgstr "Autorisieren"
+
+#: aleksis/core/templates/oauth2_provider/authorize.html:18
+msgid "The application requests access to the following scopes:"
+msgstr "Die Anwendung fordert Zugriff auf die folgenden Bereiche an:"
+
+#: aleksis/core/templates/oauth2_provider/authorize.html:33
+msgid "Allow"
+msgstr "Erlauben"
+
+#: aleksis/core/templates/oauth2_provider/authorize.html:36
+msgid "Disallow"
+msgstr "Verbieten"
+
+#: aleksis/core/templates/oauth2_provider/authorized-oob.html:12
+msgid "Success!"
+msgstr "Erfolg!"
+
+#: aleksis/core/templates/oauth2_provider/authorized-oob.html:14
+msgid "Please return to your application and enter this code:"
+msgstr "Bitte gehen Sie zurück in Ihre Anwendung und geben Sie diesen Code ein:"
+
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:5
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:6
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:23
+msgid "Revoke access"
+msgstr "Zugriff zurückziehen"
+
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:12
+msgid "Are you sure to revoke the access for this application?"
+msgstr "Sind Sie sicher, dass Sie den Zugriff für diese Anwendung zurückziehen möchten?"
+
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:20
+msgid "Revoke"
+msgstr "Zurückziehen"
+
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:33
+msgid "No authorized applications."
+msgstr "Keine autorisierten Anwendungen."
+
+#: aleksis/core/templates/offline.html:5
 msgid "Network error"
 msgstr "Netzwerkfehler"
 
-#: templates/offline.html:8
+#: aleksis/core/templates/offline.html:8
 msgid ""
 "No internet\n"
 "    connection."
@@ -1663,7 +2302,7 @@ msgstr ""
 "Keine\n"
 "    Internetverbindung."
 
-#: templates/offline.html:12
+#: aleksis/core/templates/offline.html:12
 msgid ""
 "\n"
 "      There was an error accessing this page. You probably don't have an internet connection. Check to see if your WiFi\n"
@@ -1677,36 +2316,134 @@ msgstr ""
 "      Systemadministratoren:\n"
 "    "
 
-#: templates/search/search.html:8
+#: aleksis/core/templates/search/search.html:8
 msgid "Global Search"
 msgstr "Globale Suche"
 
-#: templates/search/search.html:15
+#: aleksis/core/templates/search/search.html:15
 msgid "Search Term"
 msgstr "Suchausdruck"
 
-#: templates/search/search.html:26
+#: aleksis/core/templates/search/search.html:26
 msgid "Results"
 msgstr "Ergebnisse"
 
-#: templates/search/search.html:38
+#: aleksis/core/templates/search/search.html:38
 msgid "No search results could be found to your search."
 msgstr "Es konnten keine Suchergebnisse zu Ihrem Suchausdruck gefunden werden."
 
-#: templates/search/search.html:87
+#: aleksis/core/templates/search/search.html:87
 msgid "Please enter a search term above."
 msgstr "Bitte geben Sie einen Suchausdruck ein."
 
-#: templates/templated_email/data_checks.email:4
+#: aleksis/core/templates/socialaccount/authentication_error.html:5
+#: aleksis/core/templates/socialaccount/authentication_error.html:6
+msgid "Third-party Account Login Failure"
+msgstr "Anmeldung über Drittanbieter-Konto fehlgeschlagen"
+
+#: aleksis/core/templates/socialaccount/authentication_error.html:13
+msgid "Third-party Account Login Failure."
+msgstr "Anmeldung über Drittanbieter-Konto fehlgeschlagen."
+
+#: aleksis/core/templates/socialaccount/authentication_error.html:15
+msgid ""
+"\n"
+"            An error occurred while attempting to login via your third-party account.\n"
+"            Please contact one of your site administrators.\n"
+"          "
+msgstr ""
+"\n"
+"            Beim dem Versuch, die Anmeldung über Ihr Drittanbieter-Konto durchzuführen, ist ein Fehler aufgetreten.\n"
+"            Kontaktieren Sie bitte einen Ihrer Systemadministratoren:\n"
+"          "
+
+#: aleksis/core/templates/socialaccount/connections.html:5
+#: aleksis/core/templates/socialaccount/connections.html:6
+msgid "Connections"
+msgstr "Verbindungen"
+
+#: aleksis/core/templates/socialaccount/connections.html:24
+msgid "Remove"
+msgstr "Löschen"
+
+#: aleksis/core/templates/socialaccount/connections.html:34
+msgid "You currently have no third-party accounts connected to this account."
+msgstr "Sie haben aktuell keine Drittanbieter-Konten mit Ihrem Konto verbunden."
+
+#: aleksis/core/templates/socialaccount/connections.html:37
+msgid "Add a Third-party Account"
+msgstr "Ein Drittanbieter-Konto hinzufügen"
+
+#: aleksis/core/templates/socialaccount/login_cancelled.html:5
+#: aleksis/core/templates/socialaccount/login_cancelled.html:6
+#: aleksis/core/templates/socialaccount/login_cancelled.html:13
+msgid "Login cancelled"
+msgstr "Login abgebrochen"
+
+#: aleksis/core/templates/socialaccount/login_cancelled.html:15
+#, python-format
+msgid ""
+"\n"
+"            You decided to cancel logging in to our site using one of your existing accounts. If this was a mistake, please proceed to <a href=\"%(login_url)s\">sign in</a>.\n"
+"          "
+msgstr ""
+"\n"
+"            Sie haben sich entschieden, die Anmeldung mit einem Ihrer bestehenden Konten bei uns abzubrechen. Wenn dies ein Fehler war, <a href=\"%(login_url)s\">fahren Sie bitte mit dem Login fort</a>.\n"
+"          "
+
+#: aleksis/core/templates/socialaccount/signup.html:12
+#, python-format
+msgid ""
+"You are about to use your %(provider_name)s account to login to\n"
+"        %(site_name)s. As a final step, please complete the following form:"
+msgstr ""
+"Sie sind dabei, Ihr %(provider_name)s-Konto zur Anmeldung bei %(site_name)s zu nutzen. \n"
+"Als ein letzter Schritt vervollständigen Sie bitte das folgende Formular:"
+
+#: aleksis/core/templates/socialaccount/snippets/provider_list.html:12
+#, python-format
+msgid ""
+"\n"
+"                Login with %(name)s\n"
+"              "
+msgstr ""
+"\n"
+"                Anmelden mit %(name)s\n"
+"              "
+
+#: aleksis/core/templates/socialaccount/snippets/provider_list.html:21
+#, python-format
+msgid ""
+"\n"
+"            Login with %(name)s\n"
+"          "
+msgstr ""
+"\n"
+"            Anmelden mit %(name)s\n"
+"          "
+
+#: aleksis/core/templates/socialaccount/snippets/provider_list.html:30
+msgid ""
+"\n"
+"          No third-party account providers available.\n"
+"        "
+msgstr ""
+"\n"
+"          Keine Drittanbieter verfügbar.\n"
+"        "
+
+#: aleksis/core/templates/templated_email/data_checks.email:4
 msgid "The system detected some new problems with your data."
 msgstr "Das System hat einige neue Probleme mit Ihren Daten entdeckt."
 
-#: templates/templated_email/data_checks.email:8
-#: templates/templated_email/data_checks.email:24
+#: aleksis/core/templates/templated_email/data_checks.email:8
+#: aleksis/core/templates/templated_email/data_checks.email:24
+#: aleksis/core/templates/templated_email/person_changed.email:8
+#: aleksis/core/templates/templated_email/person_changed.email:20
 msgid "Hello,"
 msgstr "Hallo,"
 
-#: templates/templated_email/data_checks.email:10
+#: aleksis/core/templates/templated_email/data_checks.email:10
 msgid ""
 "\n"
 "  the system detected some new problems with your data.\n"
@@ -1718,7 +2455,7 @@ msgstr ""
 "Bitte nehmen Sie sich etwas Zeit, diese zu überprüfen und sie zu lösen oder als ignoriert zu markieren.\n"
 " "
 
-#: templates/templated_email/data_checks.email:26
+#: aleksis/core/templates/templated_email/data_checks.email:26
 msgid ""
 "\n"
 "   the system detected some new problems with your data.\n"
@@ -1730,35 +2467,35 @@ msgstr ""
 "Bitte nehmen Sie sich etwas Zeit, diese zu überprüfen und sie zu lösen oder als ignoriert zu markieren.\n"
 "  "
 
-#: templates/templated_email/data_checks.email:34
+#: aleksis/core/templates/templated_email/data_checks.email:34
 msgid "Problem description"
 msgstr "Problembeschreibung"
 
-#: templates/templated_email/data_checks.email:35
+#: aleksis/core/templates/templated_email/data_checks.email:35
 msgid "Count of objects with new problems"
 msgstr "Anzahl der Objekte mit neuen Problemen"
 
-#: templates/templated_email/notification.email:3
+#: aleksis/core/templates/templated_email/notification.email:3
 msgid "New notification for"
 msgstr "Neue Benachrichtigung für"
 
-#: templates/templated_email/notification.email:6
-#: templates/templated_email/notification.email:27
+#: aleksis/core/templates/templated_email/notification.email:6
+#: aleksis/core/templates/templated_email/notification.email:27
 #, python-format
 msgid "Dear %(notification_user)s,"
 msgstr "Liebe(r) %(notification_user)s,"
 
-#: templates/templated_email/notification.email:8
-#: templates/templated_email/notification.email:29
+#: aleksis/core/templates/templated_email/notification.email:8
+#: aleksis/core/templates/templated_email/notification.email:29
 msgid "we got a new notification for you:"
 msgstr "wir haben eine neue Benachrichtigung für Sie:"
 
-#: templates/templated_email/notification.email:15
-#: templates/templated_email/notification.email:35
+#: aleksis/core/templates/templated_email/notification.email:15
+#: aleksis/core/templates/templated_email/notification.email:35
 msgid "More information"
 msgstr "Mehr Informationen"
 
-#: templates/templated_email/notification.email:18
+#: aleksis/core/templates/templated_email/notification.email:18
 #, python-format
 msgid ""
 "\n"
@@ -1769,12 +2506,7 @@ msgstr ""
 "        Von %(trans_sender)s am %(trans_created_at)s\n"
 "    "
 
-#: templates/templated_email/notification.email:22
-#: templates/templated_email/notification.email:46
-msgid "Your AlekSIS team"
-msgstr "Ihr AlekSIS-Team"
-
-#: templates/templated_email/notification.email:40
+#: aleksis/core/templates/templated_email/notification.email:40
 #, python-format
 msgid ""
 "\n"
@@ -1785,24 +2517,47 @@ msgstr ""
 "            Von %(trans_sender)s um %(trans_created_at)s\n"
 "        "
 
-#: templates/two_factor/_base_focus.html:6
-#: templates/two_factor/core/otp_required.html:22
-#: templates/two_factor/core/setup.html:5
-#: templates/two_factor/profile/profile.html:87
+#: aleksis/core/templates/templated_email/person_changed.email:4
+#, python-format
+msgid "%(person)s changed their data!"
+msgstr "%(person)s hat Daten verändert!"
+
+#: aleksis/core/templates/templated_email/person_changed.email:10
+#, python-format
+msgid ""
+"\n"
+"   the person %(person)s recently changed the following fields:\n"
+" "
+msgstr ""
+"\n"
+"   die Person %(person)s hat kürzlich die folgenden Felder geändert:\n"
+" "
+
+#: aleksis/core/templates/templated_email/person_changed.email:22
+#, python-format
+msgid ""
+"\n"
+"    the person %(person)s recently changed the following fields:\n"
+"  "
+msgstr ""
+"\n"
+"    die Person %(person)s hat kürzlich die folgenden Felder geändert:\n"
+"  "
+
+#: aleksis/core/templates/two_factor/_base_focus.html:6
+#: aleksis/core/templates/two_factor/core/otp_required.html:22
+#: aleksis/core/templates/two_factor/core/setup.html:5
+#: aleksis/core/templates/two_factor/profile/profile.html:87
 msgid "Enable Two-Factor Authentication"
 msgstr "Zwei-Faktor-Authentifizierung aktivieren"
 
-#: templates/two_factor/_wizard_actions.html:6
-msgid "Cancel"
-msgstr "Abbrechen"
-
-#: templates/two_factor/core/backup_tokens.html:5
-#: templates/two_factor/core/backup_tokens.html:9
-#: templates/two_factor/profile/profile.html:46
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:5
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:9
+#: aleksis/core/templates/two_factor/profile/profile.html:46
 msgid "Backup Tokens"
 msgstr "Backup-Token"
 
-#: templates/two_factor/core/backup_tokens.html:14
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:14
 msgid ""
 "\n"
 "        Backup tokens can be used when your primary and backup\n"
@@ -1819,7 +2574,7 @@ msgstr ""
 "        müssen Sie neue generieren. Nur gültige Backup-Tokens werden angezeigt.\n"
 "      "
 
-#: templates/two_factor/core/backup_tokens.html:33
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:33
 msgid ""
 "\n"
 "          Print these tokens and keep them somewhere safe.\n"
@@ -1829,85 +2584,111 @@ msgstr ""
 "          Drucken Sie diese Tokens aus und bewahren Sie sie gut auf.\n"
 "        "
 
-#: templates/two_factor/core/backup_tokens.html:39
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:39
 msgid "You don't have any backup codes yet."
 msgstr "Sie haben aktuell keine Backup-Codes."
 
-#: templates/two_factor/core/backup_tokens.html:45
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:45
 msgid "Back to Account Security"
 msgstr "Zurück zur Kontosicherheit"
 
-#: templates/two_factor/core/backup_tokens.html:49
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:49
 msgid "Generate Tokens"
 msgstr "Tokens generieren"
 
-#: templates/two_factor/core/login.html:16
-msgid "You have no permission to view this page. Please login with an other account."
-msgstr "Sie haben nicht die nötigen Berechtigungen, um diese Seite aufzurufen. Bitte loggen Sie sich mit einem anderen Account ein."
+#: aleksis/core/templates/two_factor/core/login.html:20
+msgid "Login with username and password"
+msgstr "Anmeldung mit Benutzername und Passwort"
+
+#: aleksis/core/templates/two_factor/core/login.html:28
+msgid ""
+"You have no permission to view this page. Please login with an other\n"
+"                    account."
+msgstr ""
+"Sie haben keine Berichtigung, um diese Seite aufzurufen. \n"
+"Bitte loggen Sie sich mit einem anderen Account ein."
 
-#: templates/two_factor/core/login.html:24
+#: aleksis/core/templates/two_factor/core/login.html:36
 msgid "Please login to see this page."
 msgstr "Bitte melden Sie sich an, um diese Seite zu sehen."
 
-#: templates/two_factor/core/login.html:30
-msgid "Login with username and password"
-msgstr "Anmeldung mit Nutzername und Passwort"
-
-#: templates/two_factor/core/login.html:40
+#: aleksis/core/templates/two_factor/core/login.html:46
 msgid ""
-"We are calling your phone right now, please enter the\n"
-"              digits you hear."
+"\n"
+"                        We are calling your phone right now, please enter the\n"
+"                        digits you hear.\n"
+"                      "
 msgstr ""
-"Wir rufen Ihr Telefon jetzt an, bitte geben Sie die\n"
-"                Zahlen ein, die Sie hören."
+"\n"
+"                        Wir rufen Ihr Telefon jetzt an, \n"
+"            bitte geben Sie die Zahlen ein, die Sie hören.\n"
+"                      "
 
-#: templates/two_factor/core/login.html:43
+#: aleksis/core/templates/two_factor/core/login.html:51
 msgid ""
-"We sent you a text message, please enter the tokens we\n"
-"              sent."
+"\n"
+"                        We sent you a text message, please enter the tokens we\n"
+"                        sent.\n"
+"                      "
 msgstr ""
-"Wir haben Ihnen eine Textnachricht geschickt. Bitte geben Sie die Tokens ein,\n"
-"              die wir Ihnen geschickt haben."
+"\n"
+"                        Wir haben Ihnen eine SMS geschickt, \n"
+"            bitte geben Sie die Token ein, die wir geschickt haben.\n"
+"                      "
 
-#: templates/two_factor/core/login.html:46
+#: aleksis/core/templates/two_factor/core/login.html:56
 msgid ""
-"Please enter the tokens generated by your token\n"
-"              generator."
+"\n"
+"                        Please enter the tokens generated by your token\n"
+"                        generator.\n"
+"                      "
 msgstr ""
-"Bitte geben Sie den von Ihrem Token-Generator\n"
-"              generierten Token ein."
+"\n"
+"                        Bitte geben Sie den von Ihrem Token-Generator\n"
+"              generierten Token ein.\n"
+"                      "
 
-#: templates/two_factor/core/login.html:50
+#: aleksis/core/templates/two_factor/core/login.html:62
 msgid ""
-"Use this form for entering backup tokens for logging in.\n"
-"            These tokens have been generated for you to print and keep safe. Please\n"
-"            enter one of these backup tokens to login to your account."
+"\n"
+"                      Use this form for entering backup tokens for logging in.\n"
+"                      These tokens have been generated for you to print and keep safe. Please\n"
+"                      enter one of these backup tokens to login to your account.\n"
+"                    "
 msgstr ""
-"Nutzen Sie dieses Formular um Ihre Backup-Tokens zum Anmelden einzugeben.\n"
-"                Diese Tokens wurden für Sie generiert, um diese gut aufzubewahren. Bitte\n"
-"                geben Sie einen dieser Tokens ein, um sich einzuloggen."
+"\n"
+"                      Nutzen Sie dieses Formular um Ihre Backup-Tokens zum "
+"Anmelden einzugeben.\n"
+"                Diese Tokens wurden für Sie generiert, um diese gut "
+"aufzubewahren. Bitte\n"
+"                geben Sie einen dieser Tokens ein, um sich einzuloggen.\n"
+"                    "
 
-#: templates/two_factor/core/login.html:68
+#: aleksis/core/templates/two_factor/core/login.html:90
+msgid "Device currently not available?"
+msgstr "Gerät aktuell nicht verfügbar?"
+
+#: aleksis/core/templates/two_factor/core/login.html:92
 msgid "Or, alternatively, use one of your backup phones:"
 msgstr "Oder, alternativ, nutzen Sie eins Ihrer Backup-Telefone:"
 
-#: templates/two_factor/core/login.html:78
+#: aleksis/core/templates/two_factor/core/login.html:102
 msgid "As a last resort, you can use a backup token:"
 msgstr "Als letzte Möglichkeit können Sie einen Backup-Token nutzen:"
 
-#: templates/two_factor/core/login.html:81
+#: aleksis/core/templates/two_factor/core/login.html:105
 msgid "Use Backup Token"
 msgstr "Backup-Token nutzen"
 
-#: templates/two_factor/core/login.html:93
+#: aleksis/core/templates/two_factor/core/login.html:116
 msgid "Use alternative login options"
 msgstr "Alternative Anmeldemöglichkeiten nutzen"
 
-#: templates/two_factor/core/otp_required.html:9
+#: aleksis/core/templates/two_factor/core/otp_required.html:9
 msgid "Permission Denied"
 msgstr "Zugriff verwehrt"
 
-#: templates/two_factor/core/otp_required.html:10
+#: aleksis/core/templates/two_factor/core/otp_required.html:10
 msgid ""
 "The page you requested, enforces users to verify using\n"
 "          two-factor authentication for security reasons. You need to enable these\n"
@@ -1917,7 +2698,7 @@ msgstr ""
 "          eine Verifizierung durch Zwei-Faktor-Authentifizierung. Sie müssen diese\n"
 "          Sicherheitsfunktion aktivieren, um diese Seite aufzurufen."
 
-#: templates/two_factor/core/otp_required.html:14
+#: aleksis/core/templates/two_factor/core/otp_required.html:14
 msgid ""
 "Two-factor authentication is not enabled for your\n"
 "          account. Enable two-factor authentication for enhanced account\n"
@@ -1927,12 +2708,12 @@ msgstr ""
 "          Aktivieren Sie Zwei-Faktor-Authentifizierung für eine verbesserte\n"
 "          Accountsicherheit."
 
-#: templates/two_factor/core/phone_register.html:5
-#: templates/two_factor/core/phone_register.html:9
+#: aleksis/core/templates/two_factor/core/phone_register.html:5
+#: aleksis/core/templates/two_factor/core/phone_register.html:9
 msgid "Add Backup Phone"
 msgstr "Backup-Telefon hinzufügen"
 
-#: templates/two_factor/core/phone_register.html:12
+#: aleksis/core/templates/two_factor/core/phone_register.html:12
 msgid ""
 "You'll be adding a backup phone number to your\n"
 "      account. This number will be used if your primary method of\n"
@@ -1942,7 +2723,7 @@ msgstr ""
 "      Account hinzufügen. Diese Nummer wird genutzt, wenn Ihre primäre\n"
 "      Authentifikationsmethode nicht verfügbar ist."
 
-#: templates/two_factor/core/phone_register.html:16
+#: aleksis/core/templates/two_factor/core/phone_register.html:16
 msgid ""
 "We've sent a token to your phone number. Please\n"
 "      enter the token you've received."
@@ -1950,7 +2731,7 @@ msgstr ""
 "Wir haben Ihnen einen Token an Ihre Telefonnummer gesendet.\n"
 "      Bitte geben Sie diesen Token ein."
 
-#: templates/two_factor/core/setup.html:9
+#: aleksis/core/templates/two_factor/core/setup.html:9
 msgid ""
 "\n"
 "        You are about to take your account security to the\n"
@@ -1964,7 +2745,7 @@ msgstr ""
 "       Zwei-Faktor-Authentifizierug zu aktivieren.\n"
 "      "
 
-#: templates/two_factor/core/setup.html:17
+#: aleksis/core/templates/two_factor/core/setup.html:17
 msgid ""
 "\n"
 "        Please select which authentication method you would like to use:\n"
@@ -1974,7 +2755,7 @@ msgstr ""
 "        Bitte wählen Sie aus, welche Authentifikationsmethode Sie nutzen wollen:\n"
 "      "
 
-#: templates/two_factor/core/setup.html:23
+#: aleksis/core/templates/two_factor/core/setup.html:23
 msgid ""
 "\n"
 "        To start using a token generator, please use your\n"
@@ -1988,7 +2769,7 @@ msgstr ""
 "Dann geben Sie den in der App angezeigten Code an:\n"
 "      "
 
-#: templates/two_factor/core/setup.html:34
+#: aleksis/core/templates/two_factor/core/setup.html:34
 msgid ""
 "\n"
 "        Please enter the phone number you wish to receive the\n"
@@ -2000,7 +2781,7 @@ msgstr ""
 "        an die die SMS-Nachrichten geschickt werden sollen. Diese Nummer wird im nächsten Schritt überprüft.\n"
 "      "
 
-#: templates/two_factor/core/setup.html:41
+#: aleksis/core/templates/two_factor/core/setup.html:41
 msgid ""
 "\n"
 "        Please enter the phone number you wish to be called on.\n"
@@ -2012,7 +2793,7 @@ msgstr ""
 "Diese Nummer wird im nächsten Schritt überprüft.\n"
 "      "
 
-#: templates/two_factor/core/setup.html:50
+#: aleksis/core/templates/two_factor/core/setup.html:50
 msgid ""
 "\n"
 "            We are calling your phone right now, please enter the digits you hear.\n"
@@ -2022,17 +2803,18 @@ msgstr ""
 "            Wir rufen Ihr Telefon jetzt an, bitte geben Sie die Zahlen ein, die Sie hören.\n"
 "          "
 
-#: templates/two_factor/core/setup.html:56
+#: aleksis/core/templates/two_factor/core/setup.html:56
 msgid ""
 "\n"
 "            We sent you a text message, please enter the tokens we sent.\n"
 "          "
 msgstr ""
 "\n"
-"            Wir haben Ihnen per SMS einen Token geschickt, bitte geben Sie diesen ein.\n"
+"            Wir haben Ihnen eine SMS geschickt, bitte geben Sie die Tokens "
+"ein, die wir geschickt haben.\n"
 "          "
 
-#: templates/two_factor/core/setup.html:63
+#: aleksis/core/templates/two_factor/core/setup.html:63
 msgid ""
 "\n"
 "          We've encountered an issue with the selected authentication method. Please\n"
@@ -2048,7 +2830,7 @@ msgstr ""
 "Wenn der Fehler bestehen bleibt, kontaktieren Sie bitte einen der Administratoren.\n"
 "        "
 
-#: templates/two_factor/core/setup.html:73
+#: aleksis/core/templates/two_factor/core/setup.html:73
 msgid ""
 "\n"
 "        To identify and verify your YubiKey, please insert a\n"
@@ -2062,12 +2844,12 @@ msgstr ""
 "Dann wird Ihr YubiKey mit Ihrem Konto verknüpft.\n"
 "      "
 
-#: templates/two_factor/core/setup_complete.html:5
-#: templates/two_factor/core/setup_complete.html:9
+#: aleksis/core/templates/two_factor/core/setup_complete.html:5
+#: aleksis/core/templates/two_factor/core/setup_complete.html:9
 msgid "Two-Factor Authentication successfully enabled"
 msgstr "Zwei-Faktor-Authentifizierung erfolgreich aktiviert"
 
-#: templates/two_factor/core/setup_complete.html:14
+#: aleksis/core/templates/two_factor/core/setup_complete.html:14
 msgid ""
 "\n"
 "        Congratulations, you've successfully enabled two-factor authentication.\n"
@@ -2077,17 +2859,17 @@ msgstr ""
 "        Gratulation, Sie haben die Zwei-Faktor-Authentifizierung erfolgreich aktiviert.\n"
 "      "
 
-#: templates/two_factor/core/setup_complete.html:24
-#: templates/two_factor/core/setup_complete.html:44
+#: aleksis/core/templates/two_factor/core/setup_complete.html:24
+#: aleksis/core/templates/two_factor/core/setup_complete.html:44
 msgid "Back to Profile"
 msgstr "Zurück zum Profil"
 
-#: templates/two_factor/core/setup_complete.html:28
-#: templates/two_factor/core/setup_complete.html:48
+#: aleksis/core/templates/two_factor/core/setup_complete.html:28
+#: aleksis/core/templates/two_factor/core/setup_complete.html:48
 msgid "Generate backup codes"
 msgstr "Backup-Codes generieren"
 
-#: templates/two_factor/core/setup_complete.html:34
+#: aleksis/core/templates/two_factor/core/setup_complete.html:34
 msgid ""
 "\n"
 "          However, it might happen that you don't have access to\n"
@@ -2101,49 +2883,49 @@ msgstr ""
 "generieren Sie Backupcodes oder fügen eine Telefonnummer hinzu.\n"
 "        "
 
-#: templates/two_factor/core/setup_complete.html:52
-#: templates/two_factor/profile/profile.html:41
+#: aleksis/core/templates/two_factor/core/setup_complete.html:52
+#: aleksis/core/templates/two_factor/profile/profile.html:41
 msgid "Add Phone Number"
 msgstr "Telefonnummer hinzufügen"
 
-#: templates/two_factor/profile/disable.html:5
-#: templates/two_factor/profile/disable.html:9
-#: templates/two_factor/profile/profile.html:63
-#: templates/two_factor/profile/profile.html:73
+#: aleksis/core/templates/two_factor/profile/disable.html:5
+#: aleksis/core/templates/two_factor/profile/disable.html:9
+#: aleksis/core/templates/two_factor/profile/profile.html:63
+#: aleksis/core/templates/two_factor/profile/profile.html:73
 msgid "Disable Two-Factor Authentication"
 msgstr "Zwei-Faktor-Authentifizierung deaktiveren"
 
-#: templates/two_factor/profile/disable.html:12
+#: aleksis/core/templates/two_factor/profile/disable.html:12
 msgid "You are about to disable two-factor authentication. This weakens your account security, are you sure?"
 msgstr "Sie sind dabei, Zwei-Faktor-Authentifizierung zu deaktivieren. Das verschlechtert Ihre Kontosicherheit. Sind Sie sicher?"
 
-#: templates/two_factor/profile/disable.html:26
+#: aleksis/core/templates/two_factor/profile/disable.html:26
 msgid "Disable"
 msgstr "Deaktivieren"
 
-#: templates/two_factor/profile/profile.html:5
-#: templates/two_factor/profile/profile.html:10
+#: aleksis/core/templates/two_factor/profile/profile.html:5
+#: aleksis/core/templates/two_factor/profile/profile.html:10
 msgid "Account Security"
 msgstr "Kontosicherheit"
 
-#: templates/two_factor/profile/profile.html:15
+#: aleksis/core/templates/two_factor/profile/profile.html:15
 msgid "Tokens will be generated by your token generator."
 msgstr "Tokens werden von Ihrem Token-Generator generiert."
 
-#: templates/two_factor/profile/profile.html:17
+#: aleksis/core/templates/two_factor/profile/profile.html:17
 #, python-format
 msgid "Primary method: %(primary)s"
 msgstr "Primäre Methode: %(primary)s"
 
-#: templates/two_factor/profile/profile.html:19
+#: aleksis/core/templates/two_factor/profile/profile.html:19
 msgid "Tokens will be generated by your YubiKey."
 msgstr "Tokens werden von Ihrem YubiKey generiert."
 
-#: templates/two_factor/profile/profile.html:23
+#: aleksis/core/templates/two_factor/profile/profile.html:23
 msgid "Backup Phone Numbers"
 msgstr "Backup-Telefonnummern"
 
-#: templates/two_factor/profile/profile.html:24
+#: aleksis/core/templates/two_factor/profile/profile.html:24
 msgid ""
 "If your primary method is not available, we are able to\n"
 "        send backup tokens to the phone numbers listed below."
@@ -2151,11 +2933,11 @@ msgstr ""
 "Wenn Ihre primäre Methode nicht verfügbar ist, sind wir in der Lage,\n"
 "        Backup-Tokens zu den unten aufgelisteten Telefonnummern zu schicken."
 
-#: templates/two_factor/profile/profile.html:33
+#: aleksis/core/templates/two_factor/profile/profile.html:33
 msgid "Unregister"
 msgstr "Abmelden"
 
-#: templates/two_factor/profile/profile.html:48
+#: aleksis/core/templates/two_factor/profile/profile.html:48
 msgid ""
 "If you don't have any device with you, you can access\n"
 "        your account using backup tokens."
@@ -2163,7 +2945,7 @@ msgstr ""
 "Wenn Sie keines Ihrer Geräte dabei haben, können Sie sich\n"
 "      mit Backup-Tokens einloggen."
 
-#: templates/two_factor/profile/profile.html:50
+#: aleksis/core/templates/two_factor/profile/profile.html:50
 #, python-format
 msgid ""
 "\n"
@@ -2182,11 +2964,11 @@ msgstr[1] ""
 "        Sie haben %(counter)s Backup-Tokens übrig.\n"
 "      "
 
-#: templates/two_factor/profile/profile.html:59
+#: aleksis/core/templates/two_factor/profile/profile.html:59
 msgid "Show Codes"
 msgstr "Codes anzeigen"
 
-#: templates/two_factor/profile/profile.html:65
+#: aleksis/core/templates/two_factor/profile/profile.html:65
 msgid ""
 "\n"
 "        However we strongly discourage you to do so, you can\n"
@@ -2198,7 +2980,7 @@ msgstr ""
 "        Sie können jedoch auch die Zwei-Faktor-Authentifizierung für Ihr Konto deaktivieren.\n"
 "      "
 
-#: templates/two_factor/profile/profile.html:78
+#: aleksis/core/templates/two_factor/profile/profile.html:78
 msgid ""
 "\n"
 "        Two-factor authentication is not enabled for your\n"
@@ -2212,123 +2994,223 @@ msgstr ""
 "          Accountsicherheit.\n"
 "      "
 
-#: util/notifications.py:65
+#: aleksis/core/util/notifications.py:65
 msgid "E-Mail"
 msgstr "E-Mail"
 
-#: util/notifications.py:66
+#: aleksis/core/util/notifications.py:66
 msgid "SMS"
 msgstr "SMS"
 
-#: views.py:141
+#: aleksis/core/util/pdf.py:105
+msgid "Progress: Generate PDF file"
+msgstr "Fortschritt: PDF-Datei generieren"
+
+#: aleksis/core/util/pdf.py:106
+msgid "Generating PDF file …"
+msgstr "PDF-Datei wird generiert …"
+
+#: aleksis/core/util/pdf.py:107
+msgid "The PDF file has been generated successfully."
+msgstr "Die PDF-Datei wurde erfolgreich generiert."
+
+#: aleksis/core/util/pdf.py:108
+msgid "There was a problem while generating the PDF file."
+msgstr "Es ist ein Fehler beim Generieren der PDF-Datei aufgetreten."
+
+#: aleksis/core/util/pdf.py:111
+msgid "Download PDF"
+msgstr "PDF herunterladen"
+
+#: aleksis/core/views.py:251
 msgid "The school term has been created."
 msgstr "Das Schuljahr wurde erstellt."
 
-#: views.py:153
+#: aleksis/core/views.py:263
 msgid "The school term has been saved."
 msgstr "Das Schuljahr wurde gespeichert."
 
-#: views.py:298
+#: aleksis/core/views.py:387
 msgid "The child groups were successfully saved."
 msgstr "Die Untergruppen wurden gespeichert."
 
-#: views.py:336
+#: aleksis/core/views.py:406 aleksis/core/views.py:416
 msgid "The person has been saved."
 msgstr "Die Person wurde gespeichert."
 
-#: views.py:375
+#: aleksis/core/views.py:466
 msgid "The group has been saved."
 msgstr "Die Gruppe wurde gespeichert."
 
-#: views.py:467
+#: aleksis/core/views.py:563
 msgid "The announcement has been saved."
 msgstr "Die Ankündigung wurde gespeichert."
 
-#: views.py:483
+#: aleksis/core/views.py:579
 msgid "The announcement has been deleted."
 msgstr "Ankündigung wurde gelöscht."
 
-#: views.py:562
+#: aleksis/core/views.py:663
 msgid "The preferences have been saved successfully."
 msgstr "Die Einstellungen wurde gespeichert."
 
-#: views.py:586
+#: aleksis/core/views.py:687
 msgid "The person has been deleted."
 msgstr "Die Person wurde gelöscht."
 
-#: views.py:600
+#: aleksis/core/views.py:701
 msgid "The group has been deleted."
 msgstr "Die Gruppe wurde gelöscht."
 
-#: views.py:632
+#: aleksis/core/views.py:733
 msgid "The additional_field has been saved."
 msgstr "Das zusätzliche Feld wurde gespeichert."
 
-#: views.py:666
+#: aleksis/core/views.py:767
 msgid "The additional field has been deleted."
 msgstr "Das zusätzliche Feld wurde gelöscht."
 
-#: views.py:691
+#: aleksis/core/views.py:792
 msgid "The group type has been saved."
 msgstr "Der Gruppentyp wurde gespeichert."
 
-#: views.py:721
+#: aleksis/core/views.py:822
 msgid "The group type has been deleted."
 msgstr "Der Gruppentyp wurde gelöscht."
 
-#: views.py:749
-msgid "The data check has been started. Please note that it may take a while before you are able to fetch the data on this page."
-msgstr "Die Datenüberprüfung wurde gestartet. Bitte beachten Sie, dass es eine Weile dauern kann, bevor Sie auf dieser Seite Ergebnisse abrufen können."
+#: aleksis/core/views.py:855
+msgid "Progress: Run data checks"
+msgstr "Fortschritt: Datenprüfungen ausführen"
 
-#: views.py:754
-msgid "The data check has finished."
-msgstr "Die Datenüberprüfung wurde beendet."
+#: aleksis/core/views.py:856
+msgid "Run data checks …"
+msgstr "Datenprüfungen laufen …"
 
-#: views.py:769
+#: aleksis/core/views.py:857
+msgid "The data checks were run successfully."
+msgstr "Die Datenprüfungen wurden erfolgreich ausgeführt."
+
+#: aleksis/core/views.py:858
+msgid "There was a problem while running data checks."
+msgstr "Es gab ein Problem beim Ausführen der Datenprüfungen."
+
+#: aleksis/core/views.py:874
 #, python-brace-format
 msgid "The solve option '{solve_option_obj.verbose_name}' "
 msgstr "Die Lösungsoption \"{solve_option_obj.verbose_name}\" "
 
-#: views.py:811
+#: aleksis/core/views.py:916
 msgid "The dashboard widget has been saved."
 msgstr "Das Dashboard-Widget wurde gespeichert."
 
-#: views.py:841
+#: aleksis/core/views.py:946
 msgid "The dashboard widget has been created."
 msgstr "Das Dashboard-Widget wurde erstellt."
 
-#: views.py:851
+#: aleksis/core/views.py:956
 msgid "The dashboard widget has been deleted."
 msgstr "Das Dashboard-Widget wurde gelöscht."
 
-#: views.py:914
+#: aleksis/core/views.py:1023
 msgid "Your dashboard configuration has been saved successfully."
 msgstr "Ihre Dashboardkonfiguration wurde erfolgreich gespeichert."
 
-#: views.py:916
+#: aleksis/core/views.py:1025
 msgid "The configuration of the default dashboard has been saved successfully."
-msgstr ""
-"Die Konfiguration des Standard-Dashboardes wurde erfolgreich gespeichert."
+msgstr "Die Konfiguration des Standard-Dashboardes wurde erfolgreich gespeichert."
 
-#~ msgid "All"
-#~ msgstr "Alle"
+#: aleksis/core/views.py:1153
+msgid "The third-party account could not be disconnected because it is the only login method available."
+msgstr "Das Drittanbieter-Konto konnte nicht deaktiviert werden, weil es die einzige verfügbare Anmeldeoption ist."
+
+#: aleksis/core/views.py:1160
+msgid "The third-party account has been successfully disconnected."
+msgstr "Das Drittanbieter-Konto wurde erfolgreich getrennt."
+
+#~ msgid "Persons and accounts"
+#~ msgstr "Personen und Konten"
+
+#~ msgid "Can link persons to accounts"
+#~ msgstr "Kann Personen mit Benutzerkonten verknüpfen"
+
+#~ msgid "Enabled custom authentication backends"
+#~ msgstr "Benutzerdefinierte Authentifizierungsbackends aktivieren"
+
+#~ msgid "Link persons to accounts"
+#~ msgstr "Personen mit Benutzerkonten verknüpfen"
 
 #~ msgid ""
 #~ "\n"
-#~ "              Created by %(person)s\n"
-#~ "            "
+#~ "        You can use this form to assign user accounts to persons. Use the\n"
+#~ "        dropdowns to select existing accounts; use the text fields to create new\n"
+#~ "        accounts on-the-fly. The latter will create a new account with the\n"
+#~ "        entered username and copy all other details from the person.\n"
+#~ "      "
 #~ msgstr ""
 #~ "\n"
-#~ "              Erstellt von %(person)s\n"
-#~ "            "
+#~ "        Sie können dieses Formular nutzen, um Benutzerkonten Personen zuzuweisen. Nutzen Sie das\n"
+#~ "    Auswahlfeld um ein existierendes Benutzerkonto auszuwählen; nutzen Sie das Textfeld, um einen neuen Benutzer zu\n"
+#~ "    erstellen. Letzteres erstellt ein neues Benutzerkonto mit dem\n"
+#~ "    eingegebenen Benutzernamen und kopiert alle anderen Daten der Person.\n"
+#~ "      "
+
+#~ msgid "Update"
+#~ msgstr "Aktualisieren"
+
+#~ msgid "Existing account"
+#~ msgstr "Existierendes Konto"
+
+#~ msgid "New account"
+#~ msgstr "Neues Konto"
+
+#~ msgid "Impersonate user"
+#~ msgstr "Als Benutzer verkleiden"
+
+#~ msgid "Authorized tokens"
+#~ msgstr "Autorisierte Tokens"
+
+#~ msgid "No authorized tokens defined."
+#~ msgstr "Keine autorisierten Token definiert."
+
+#~ msgid "Social Network Login Failure"
+#~ msgstr "Anmeldung über soziales Netzwerk fehlgeschlagen"
+
+#~ msgid ""
+#~ "We are calling your phone right now, please enter the\n"
+#~ "              digits you hear."
+#~ msgstr ""
+#~ "Wir rufen Ihr Telefon jetzt an, bitte geben Sie die\n"
+#~ "                Zahlen ein, die Sie hören."
+
+#~ msgid ""
+#~ "We sent you a text message, please enter the tokens we\n"
+#~ "              sent."
+#~ msgstr ""
+#~ "Wir haben Ihnen eine Textnachricht geschickt. Bitte geben Sie die Tokens ein,\n"
+#~ "              die wir Ihnen geschickt haben."
+
+#~ msgid "French"
+#~ msgstr "Französisch"
+
+#~ msgid "Norwegian (bokmål)"
+#~ msgstr "Norwegisch (bokmål)"
+
+#~ msgid "The data check has been started. Please note that it may take a while before you are able to fetch the data on this page."
+#~ msgstr "Die Datenüberprüfung wurde gestartet. Bitte beachten Sie, dass es eine Weile dauern kann, bevor Sie auf dieser Seite Ergebnisse abrufen können."
+
+#~ msgid "The data check has finished."
+#~ msgstr "Die Datenüberprüfung wurde beendet."
+
+#~ msgid "All"
+#~ msgstr "Alle"
 
 #~ msgid ""
 #~ "\n"
-#~ "              Updated by %(person)s\n"
+#~ "              Created by %(person)s\n"
 #~ "            "
 #~ msgstr ""
 #~ "\n"
-#~ "              Aktualisiert von %(person)s\n"
+#~ "              Erstellt von %(person)s\n"
 #~ "            "
 
 #~ msgid ""
@@ -2455,9 +3337,6 @@ msgstr ""
 #~ msgid "Official name"
 #~ msgstr "Offizieller Name"
 
-#~ msgid "School"
-#~ msgstr "Schule"
-
 #~ msgid "Schools"
 #~ msgstr "Schulen"
 
@@ -2494,9 +3373,6 @@ msgstr ""
 #~ msgid "Forbidden"
 #~ msgstr "Verboten"
 
-#~ msgid "Not found"
-#~ msgstr "Nicht gefunden"
-
 #~ msgid "Internal Server Error"
 #~ msgstr "Interner Serverfehler"
 
diff --git a/aleksis/core/locale/de_DE/LC_MESSAGES/djangojs.po b/aleksis/core/locale/de_DE/LC_MESSAGES/djangojs.po
index ea24fc3ff92afc211898fb8b426dccb97fce56ed..6bbb636d795adbbe19acdf95ed873b27814d9284 100644
--- a/aleksis/core/locale/de_DE/LC_MESSAGES/djangojs.po
+++ b/aleksis/core/locale/de_DE/LC_MESSAGES/djangojs.po
@@ -3,32 +3,36 @@
 # This file is distributed under the same license as the PACKAGE package.
 # FIRST AUTHOR <EMAIL@ADDRESS>, YEAR.
 #
-#, fuzzy
 msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2021-01-11 21:30+0100\n"
-"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
-"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
-"Language-Team: LANGUAGE <LL@li.org>\n"
-"Language: \n"
+"POT-Creation-Date: 2021-08-28 17:53+0200\n"
+"PO-Revision-Date: 2021-10-28 14:37+0000\n"
+"Last-Translator: Jonathan Weth <teckids@jonathanweth.de>\n"
+"Language-Team: German <https://translate.edugit.org/projects/aleksis/"
+"aleksis-core-js/de/>\n"
+"Language: de_DE\n"
 "MIME-Version: 1.0\n"
 "Content-Type: text/plain; charset=UTF-8\n"
 "Content-Transfer-Encoding: 8bit\n"
+"Plural-Forms: nplurals=2; plural=n != 1;\n"
+"X-Generator: Weblate 4.8\n"
 
-#: static/js/main.js:15
+#: aleksis/core/static/js/main.js:15
 msgid "Today"
-msgstr ""
+msgstr "Heute"
 
-#: static/js/main.js:16
+#: aleksis/core/static/js/main.js:16
 msgid "Cancel"
-msgstr ""
+msgstr "Abbrechen"
 
-#: static/js/main.js:17
+#: aleksis/core/static/js/main.js:17
 msgid "OK"
-msgstr ""
+msgstr "OK"
 
-#: static/js/main.js:118
+#: aleksis/core/static/js/main.js:127
 msgid "This page may contain outdated information since there is no internet connection."
 msgstr ""
+"Diese Seite enthält vielleicht veraltete Informationen, da es keine "
+"Internetverbindung gibt."
diff --git a/aleksis/core/locale/fr/LC_MESSAGES/django.po b/aleksis/core/locale/fr/LC_MESSAGES/django.po
index 0938d0f9e68b50c905b6e0c232ade944ee0907a6..3c118dd9a39119c4615e68b5f729d60de3726bff 100644
--- a/aleksis/core/locale/fr/LC_MESSAGES/django.po
+++ b/aleksis/core/locale/fr/LC_MESSAGES/django.po
@@ -7,879 +7,1178 @@ msgid ""
 msgstr ""
 "Project-Id-Version: AlekSIS (School Information System) 0.1\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2021-01-11 21:30+0100\n"
-"PO-Revision-Date: 2020-04-27 13:03+0000\n"
-"Last-Translator: Marlene Grundey <grundema@katharineum.de>\n"
+"POT-Creation-Date: 2021-10-28 16:18+0200\n"
+"PO-Revision-Date: 2021-06-16 12:00+0000\n"
+"Last-Translator: Jonathan Weth <teckids@jonathanweth.de>\n"
 "Language-Team: French <https://translate.edugit.org/projects/aleksis/aleksis/fr/>\n"
 "Language: fr\n"
 "MIME-Version: 1.0\n"
 "Content-Type: text/plain; charset=UTF-8\n"
 "Content-Transfer-Encoding: 8bit\n"
 "Plural-Forms: nplurals=2; plural=n > 1;\n"
-"X-Generator: Weblate 4.0.1\n"
+"X-Generator: Weblate 4.4\n"
 
-#: data_checks.py:53
+#: aleksis/core/apps.py:150
+msgid "OpenID Connect scope"
+msgstr ""
+
+#: aleksis/core/apps.py:151
+msgid "Given name, family name, link to profile and picture if existing."
+msgstr ""
+
+#: aleksis/core/apps.py:152
+msgid "Full home postal address"
+msgstr ""
+
+#: aleksis/core/apps.py:153
+#, fuzzy
+#| msgid "Contact details"
+msgid "Email address"
+msgstr "Détails de contact"
+
+#: aleksis/core/apps.py:154
+msgid "Home and mobile phone"
+msgstr ""
+
+#: aleksis/core/data_checks.py:55
 msgid "Ignore problem"
 msgstr ""
 
-#: data_checks.py:174
+#: aleksis/core/data_checks.py:184
 #, python-brace-format
 msgid "Solve option '{solve_option_obj.verbose_name}' "
 msgstr ""
 
-#: filters.py:37 templates/core/base.html:77 templates/core/group/list.html:20
-#: templates/core/person/list.html:24 templates/search/search.html:7
-#: templates/search/search.html:22
+#: aleksis/core/data_checks.py:291
+msgid "Deactivate DashboardWidget"
+msgstr ""
+
+#: aleksis/core/data_checks.py:303
+msgid "Ensure that there are no broken DashboardWidgets."
+msgstr ""
+
+#: aleksis/core/data_checks.py:304
+msgid "The DashboardWidget was reported broken automatically."
+msgstr ""
+
+#: aleksis/core/filters.py:37 aleksis/core/templates/core/base.html:83
+#: aleksis/core/templates/core/base.html:84
+#: aleksis/core/templates/core/group/list.html:20
+#: aleksis/core/templates/core/person/list.html:24
+#: aleksis/core/templates/search/search.html:7
+#: aleksis/core/templates/search/search.html:22
 msgid "Search"
 msgstr ""
 
-#: filters.py:53
+#: aleksis/core/filters.py:53
 msgid "Search by name"
 msgstr ""
 
-#: filters.py:65
+#: aleksis/core/filters.py:65
 #, fuzzy
 #| msgid "Contact details"
 msgid "Search by contact details"
 msgstr "Détails de contact"
 
-#: forms.py:54
-msgid "You cannot set a new username when also selecting an existing user."
-msgstr ""
-
-#: forms.py:58
-msgid "This username is already in use."
-msgstr ""
-
-#: forms.py:82
+#: aleksis/core/forms.py:41 aleksis/core/forms.py:387
 msgid "Base data"
 msgstr ""
 
-#: forms.py:88
+#: aleksis/core/forms.py:47
 msgid "Address"
 msgstr ""
 
-#: forms.py:89
+#: aleksis/core/forms.py:48
 #, fuzzy
 #| msgid "Contact details"
 msgid "Contact data"
 msgstr "Détails de contact"
 
-#: forms.py:91
+#: aleksis/core/forms.py:50
 #, fuzzy
 #| msgid "Contact details"
 msgid "Advanced personal data"
 msgstr "Détails de contact"
 
-#: forms.py:134
+#: aleksis/core/forms.py:93
 msgid "New user"
 msgstr ""
 
-#: forms.py:134
+#: aleksis/core/forms.py:93
 msgid "Create a new account"
 msgstr ""
 
-#: forms.py:146 models.py:102
+#: aleksis/core/forms.py:124
+msgid "You cannot set a new username when also selecting an existing user."
+msgstr ""
+
+#: aleksis/core/forms.py:128
+msgid "This username is already in use."
+msgstr "Cet nom est deja en utilisation."
+
+#: aleksis/core/forms.py:145 aleksis/core/models.py:117
 msgid "School term"
 msgstr ""
 
-#: forms.py:147
+#: aleksis/core/forms.py:146
 #, fuzzy
 #| msgid "Contact details"
 msgid "Common data"
 msgstr "Détails de contact"
 
-#: forms.py:148 forms.py:197 menus.py:169 models.py:116
-#: templates/core/person/list.html:8 templates/core/person/list.html:9
+#: aleksis/core/forms.py:147 aleksis/core/forms.py:196
+#: aleksis/core/menus.py:238 aleksis/core/models.py:140
+#: aleksis/core/templates/core/person/list.html:8
+#: aleksis/core/templates/core/person/list.html:9
 #, fuzzy
 #| msgid "Person"
 msgid "Persons"
 msgstr "Personne"
 
-#: forms.py:149
+#: aleksis/core/forms.py:148
 #, fuzzy
 #| msgid "Contact details"
 msgid "Additional data"
 msgstr "Détails de contact"
 
-#: forms.py:189 forms.py:192 models.py:45
+#: aleksis/core/forms.py:188 aleksis/core/forms.py:191
+#: aleksis/core/models.py:60
 msgid "Date"
 msgstr "Date"
 
-#: forms.py:190 forms.py:193 models.py:53
+#: aleksis/core/forms.py:189 aleksis/core/forms.py:192
+#: aleksis/core/models.py:68
 msgid "Time"
 msgstr ""
 
-#: forms.py:210 menus.py:177 models.py:338 templates/core/group/list.html:8
-#: templates/core/group/list.html:9 templates/core/person/full.html:144
+#: aleksis/core/forms.py:209 aleksis/core/menus.py:249
+#: aleksis/core/models.py:398 aleksis/core/templates/core/group/list.html:8
+#: aleksis/core/templates/core/group/list.html:9
+#: aleksis/core/templates/core/person/full.html:144
 #, fuzzy
 #| msgid "Group"
 msgid "Groups"
 msgstr "Groupe"
 
-#: forms.py:220
+#: aleksis/core/forms.py:219
 msgid "From when until when should the announcement be displayed?"
 msgstr ""
 
-#: forms.py:223
+#: aleksis/core/forms.py:222
 msgid "Who should see the announcement?"
 msgstr ""
 
-#: forms.py:224
+#: aleksis/core/forms.py:223
 msgid "Write your announcement:"
 msgstr ""
 
-#: forms.py:263
+#: aleksis/core/forms.py:262
 msgid "You are not allowed to create announcements which are only valid in the past."
 msgstr ""
 
-#: forms.py:267
+#: aleksis/core/forms.py:266
 msgid "The from date and time must be earlier then the until date and time."
 msgstr ""
 
-#: forms.py:276
+#: aleksis/core/forms.py:275
 msgid "You need at least one recipient."
 msgstr ""
 
-#: health_checks.py:15
+#: aleksis/core/forms.py:389
+#, fuzzy
+#| msgid "Contact details"
+msgid "Account data"
+msgstr "Détails de contact"
+
+#: aleksis/core/forms.py:391
+msgid "Consents"
+msgstr ""
+
+#: aleksis/core/forms.py:396
+msgid "Password"
+msgstr ""
+
+#: aleksis/core/forms.py:402
+msgid "Password (again)"
+msgstr ""
+
+#: aleksis/core/forms.py:411
+#, python-brace-format
+msgid "I have read the <a href='{privacy_policy}'>Privacy policy</a> and agree with them."
+msgstr ""
+
+#: aleksis/core/forms.py:435
+msgid "You must type the same password each time."
+msgstr ""
+
+#: aleksis/core/forms.py:580
+msgid "No valid selection."
+msgstr ""
+
+#: aleksis/core/health_checks.py:21
 msgid "There are unresolved data problems."
 msgstr ""
 
-#: menus.py:7 templates/two_factor/core/login.html:6
-#: templates/two_factor/core/login.html:10
-#: templates/two_factor/core/login.html:86
+#: aleksis/core/health_checks.py:38
+msgid "The backup folder doesn't exist."
+msgstr ""
+
+#: aleksis/core/health_checks.py:47
+#, python-brace-format
+msgid "Last backup {time_gone_since_backup}!"
+msgstr ""
+
+#: aleksis/core/health_checks.py:49
+msgid "No backup found!"
+msgstr ""
+
+#: aleksis/core/health_checks.py:76
+msgid "No backup result found!"
+msgstr ""
+
+#: aleksis/core/health_checks.py:78
+#, python-brace-format
+msgid "{task.status} - {task.result}"
+msgstr ""
+
+#: aleksis/core/menus.py:9 aleksis/core/templates/two_factor/core/login.html:6
+#: aleksis/core/templates/two_factor/core/login.html:22
+#: aleksis/core/templates/two_factor/core/login.html:76
 msgid "Login"
 msgstr ""
 
-#: menus.py:13
+#: aleksis/core/menus.py:15 aleksis/core/templates/account/signup.html:20
+#: aleksis/core/templates/socialaccount/signup.html:23
+msgid "Sign up"
+msgstr ""
+
+#: aleksis/core/menus.py:24
 msgid "Dashboard"
 msgstr ""
 
-#: menus.py:19
+#: aleksis/core/menus.py:32 aleksis/core/models.py:605
+#: aleksis/core/preferences.py:26
+#: aleksis/core/templates/core/notifications.html:4
+#: aleksis/core/templates/core/notifications.html:5
+msgid "Notifications"
+msgstr ""
+
+#: aleksis/core/menus.py:41
 msgid "Account"
 msgstr ""
 
-#: menus.py:26
+#: aleksis/core/menus.py:48
 msgid "Stop impersonation"
 msgstr ""
 
-#: menus.py:35 templates/core/base.html:56
+#: aleksis/core/menus.py:57 aleksis/core/templates/core/base.html:62
 msgid "Logout"
 msgstr ""
 
-#: menus.py:41
+#: aleksis/core/menus.py:63
 msgid "2FA"
 msgstr ""
 
-#: menus.py:47
+#: aleksis/core/menus.py:69
+#: aleksis/core/templates/account/password_change.html:5
+#: aleksis/core/templates/account/password_change.html:6
+#: aleksis/core/templates/account/password_change.html:19
+#: aleksis/core/templates/account/password_reset_from_key.html:5
+#: aleksis/core/templates/account/password_reset_from_key.html:42
+#: aleksis/core/templates/account/password_reset_from_key.html:46
+#: aleksis/core/templates/account/password_reset_from_key_done.html:5
+#: aleksis/core/templates/account/password_reset_from_key_done.html:6
+msgid "Change password"
+msgstr ""
+
+#: aleksis/core/menus.py:81
 msgid "Me"
 msgstr ""
 
-#: menus.py:56 templates/dynamic_preferences/form.html:5
+#: aleksis/core/menus.py:90
+#: aleksis/core/templates/dynamic_preferences/form.html:5
 msgid "Preferences"
 msgstr ""
 
-#: menus.py:67
+#: aleksis/core/menus.py:99
+msgid "Third-party accounts"
+msgstr ""
+
+#: aleksis/core/menus.py:108
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:5
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:6
+msgid "Authorized applications"
+msgstr ""
+
+#: aleksis/core/menus.py:119
 msgid "Admin"
 msgstr ""
 
-#: menus.py:75 models.py:602 templates/core/announcement/list.html:7
-#: templates/core/announcement/list.html:8
+#: aleksis/core/menus.py:127 aleksis/core/models.py:704
+#: aleksis/core/templates/core/announcement/list.html:7
+#: aleksis/core/templates/core/announcement/list.html:8
 msgid "Announcements"
 msgstr ""
 
-#: menus.py:86 models.py:103 templates/core/school_term/list.html:8
-#: templates/core/school_term/list.html:9
+#: aleksis/core/menus.py:138 aleksis/core/models.py:118
+#: aleksis/core/templates/core/school_term/list.html:8
+#: aleksis/core/templates/core/school_term/list.html:9
 msgid "School terms"
 msgstr ""
 
-#: menus.py:97 templates/core/dashboard_widget/list.html:8
-#: templates/core/dashboard_widget/list.html:9
+#: aleksis/core/menus.py:149
+#: aleksis/core/templates/core/dashboard_widget/list.html:8
+#: aleksis/core/templates/core/dashboard_widget/list.html:9
 msgid "Dashboard widgets"
 msgstr ""
 
-#: menus.py:108 templates/core/management/data_management.html:6
-#: templates/core/management/data_management.html:7
+#: aleksis/core/menus.py:160
+#: aleksis/core/templates/core/management/data_management.html:6
+#: aleksis/core/templates/core/management/data_management.html:7
 msgid "Data management"
 msgstr ""
 
-#: menus.py:116 templates/core/pages/system_status.html:5
-#: templates/core/pages/system_status.html:7
+#: aleksis/core/menus.py:171
+#: aleksis/core/templates/core/pages/system_status.html:5
+#: aleksis/core/templates/core/pages/system_status.html:7
 msgid "System status"
 msgstr ""
 
-#: menus.py:127
+#: aleksis/core/menus.py:182
 msgid "Impersonation"
 msgstr ""
 
-#: menus.py:135
+#: aleksis/core/menus.py:193
 msgid "Configuration"
 msgstr ""
 
-#: menus.py:146 templates/core/data_check/list.html:9
-#: templates/core/data_check/list.html:10
+#: aleksis/core/menus.py:204 aleksis/core/templates/core/data_check/list.html:9
+#: aleksis/core/templates/core/data_check/list.html:10
 msgid "Data checks"
 msgstr ""
 
-#: menus.py:152
+#: aleksis/core/menus.py:210
 msgid "Backend Admin"
 msgstr ""
 
-#: menus.py:160
+#: aleksis/core/menus.py:216
+#: aleksis/core/templates/oauth2_provider/application_detail.html:5
+#: aleksis/core/templates/oauth2_provider/application_list.html:5
+msgid "OAuth2 Applications"
+msgstr ""
+
+#: aleksis/core/menus.py:229
 msgid "People"
 msgstr ""
 
-#: menus.py:185 models.py:812 templates/core/group_type/list.html:8
-#: templates/core/group_type/list.html:9
+#: aleksis/core/menus.py:260 aleksis/core/models.py:958
+#: aleksis/core/templates/core/group_type/list.html:8
+#: aleksis/core/templates/core/group_type/list.html:9
 #, fuzzy
 #| msgid "Group"
 msgid "Group types"
 msgstr "Groupe"
 
-#: menus.py:196
-msgid "Persons and accounts"
-msgstr ""
-
-#: menus.py:207
+#: aleksis/core/menus.py:271
 msgid "Groups and child groups"
 msgstr ""
 
-#: menus.py:218 models.py:385 templates/core/additional_field/list.html:8
-#: templates/core/additional_field/list.html:9
+#: aleksis/core/menus.py:282 aleksis/core/models.py:446
+#: aleksis/core/templates/core/additional_field/list.html:8
+#: aleksis/core/templates/core/additional_field/list.html:9
 msgid "Additional fields"
 msgstr ""
 
-#: menus.py:233 templates/core/group/child_groups.html:7
-#: templates/core/group/child_groups.html:9
+#: aleksis/core/menus.py:297
+#: aleksis/core/templates/core/group/child_groups.html:7
+#: aleksis/core/templates/core/group/child_groups.html:9
 msgid "Assign child groups to groups"
 msgstr ""
 
-#: mixins.py:384
+#: aleksis/core/mixins.py:498
 msgid "Linked school term"
 msgstr ""
 
-#: models.py:43
+#: aleksis/core/models.py:58
 msgid "Boolean (Yes/No)"
 msgstr ""
 
-#: models.py:44
+#: aleksis/core/models.py:59
 msgid "Text (one line)"
 msgstr ""
 
-#: models.py:46
+#: aleksis/core/models.py:61
 msgid "Date and time"
 msgstr ""
 
-#: models.py:47
+#: aleksis/core/models.py:62
 msgid "Decimal number"
 msgstr ""
 
-#: models.py:48 models.py:157
+#: aleksis/core/models.py:63 aleksis/core/models.py:186
 msgid "E-mail address"
 msgstr ""
 
-#: models.py:49
+#: aleksis/core/models.py:64
 msgid "Integer"
 msgstr ""
 
-#: models.py:50
+#: aleksis/core/models.py:65
 msgid "IP address"
 msgstr ""
 
-#: models.py:51
+#: aleksis/core/models.py:66
 msgid "Boolean or empty (Yes/No/Neither)"
 msgstr ""
 
-#: models.py:52
+#: aleksis/core/models.py:67
 msgid "Text (multi-line)"
 msgstr ""
 
-#: models.py:54
+#: aleksis/core/models.py:69
 msgid "URL / Link"
 msgstr ""
 
-#: models.py:66 models.py:785
+#: aleksis/core/models.py:81 aleksis/core/models.py:927
 msgid "Name"
 msgstr ""
 
-#: models.py:68
+#: aleksis/core/models.py:83
 #, fuzzy
 #| msgid "Contact details"
 msgid "Start date"
 msgstr "Détails de contact"
 
-#: models.py:69
+#: aleksis/core/models.py:84
 msgid "End date"
 msgstr ""
 
-#: models.py:88
+#: aleksis/core/models.py:103
 msgid "The start date must be earlier than the end date."
 msgstr ""
 
-#: models.py:95
+#: aleksis/core/models.py:110
 msgid "There is already a school term for this time or a part of this time."
 msgstr ""
 
-#: models.py:115 models.py:744 templates/core/person/accounts.html:41
+#: aleksis/core/models.py:139 aleksis/core/models.py:876
 msgid "Person"
 msgstr "Personne"
 
-#: models.py:118
+#: aleksis/core/models.py:142
 #, fuzzy
 #| msgid "Contact details"
 msgid "Can view address"
 msgstr "Détails de contact"
 
-#: models.py:119
+#: aleksis/core/models.py:143
 #, fuzzy
 #| msgid "Contact details"
 msgid "Can view contact details"
 msgstr "Détails de contact"
 
-#: models.py:120
+#: aleksis/core/models.py:144
 #, fuzzy
 #| msgid "Contact details"
 msgid "Can view photo"
 msgstr "Détails de contact"
 
-#: models.py:121
+#: aleksis/core/models.py:145
 #, fuzzy
 #| msgid "Contact details"
 msgid "Can view persons groups"
 msgstr "Détails de contact"
 
-#: models.py:122
+#: aleksis/core/models.py:146
 #, fuzzy
 #| msgid "Contact details"
 msgid "Can view personal details"
 msgstr "Détails de contact"
 
-#: models.py:127
+#: aleksis/core/models.py:156
 msgid "female"
 msgstr ""
 
-#: models.py:127
+#: aleksis/core/models.py:156
 msgid "male"
 msgstr ""
 
-#: models.py:135
+#: aleksis/core/models.py:164
 msgid "Linked user"
 msgstr ""
 
-#: models.py:137
+#: aleksis/core/models.py:166
 msgid "Is person active?"
 msgstr ""
 
-#: models.py:139
+#: aleksis/core/models.py:168
 msgid "First name"
 msgstr "Prénom"
 
-#: models.py:140
+#: aleksis/core/models.py:169
 msgid "Last name"
 msgstr "Nom de famille"
 
-#: models.py:142
+#: aleksis/core/models.py:171
 msgid "Additional name(s)"
 msgstr ""
 
-#: models.py:146 models.py:354
+#: aleksis/core/models.py:175 aleksis/core/models.py:415
 #, fuzzy
 #| msgid "First name"
 msgid "Short name"
 msgstr "Prénom"
 
-#: models.py:149
+#: aleksis/core/models.py:178
 msgid "Street"
 msgstr ""
 
-#: models.py:150
+#: aleksis/core/models.py:179
 msgid "Street number"
 msgstr ""
 
-#: models.py:151
+#: aleksis/core/models.py:180
 msgid "Postal code"
 msgstr ""
 
-#: models.py:152
+#: aleksis/core/models.py:181
 msgid "Place"
 msgstr ""
 
-#: models.py:154
+#: aleksis/core/models.py:183
 msgid "Home phone"
 msgstr ""
 
-#: models.py:155
+#: aleksis/core/models.py:184
 msgid "Mobile phone"
 msgstr ""
 
-#: models.py:159
+#: aleksis/core/models.py:188
 msgid "Date of birth"
 msgstr "Date d'anniversaire"
 
-#: models.py:160
+#: aleksis/core/models.py:189
 msgid "Sex"
 msgstr "Sexe"
 
-#: models.py:162
+#: aleksis/core/models.py:191
 msgid "Photo"
 msgstr ""
 
-#: models.py:166 templates/core/person/full.html:137
+#: aleksis/core/models.py:195 aleksis/core/templates/core/person/full.html:137
 msgid "Guardians / Parents"
 msgstr ""
 
-#: models.py:173
+#: aleksis/core/models.py:202
 msgid "Primary group"
 msgstr ""
 
-#: models.py:176 models.py:461 models.py:485 models.py:570 models.py:805
-#: templates/core/person/full.html:120
+#: aleksis/core/models.py:205 aleksis/core/models.py:563
+#: aleksis/core/models.py:587 aleksis/core/models.py:672
+#: aleksis/core/models.py:951 aleksis/core/templates/core/person/full.html:120
 msgid "Description"
 msgstr "Description"
 
-#: models.py:313
+#: aleksis/core/models.py:370
 msgid "Title of field"
 msgstr ""
 
-#: models.py:315
+#: aleksis/core/models.py:372
 msgid "Type of field"
 msgstr ""
 
-#: models.py:322
+#: aleksis/core/models.py:379
 msgid "Addtitional field for groups"
 msgstr ""
 
-#: models.py:323
+#: aleksis/core/models.py:380
 msgid "Addtitional fields for groups"
 msgstr ""
 
-#: models.py:337
+#: aleksis/core/models.py:397
 msgid "Group"
-msgstr "Groupe"
+msgstr "groupe"
 
-#: models.py:340
+#: aleksis/core/models.py:400
 msgid "Can assign child groups to groups"
 msgstr ""
 
-#: models.py:341
+#: aleksis/core/models.py:401
 #, fuzzy
 #| msgid "Contact details"
 msgid "Can view statistics about group."
 msgstr "Détails de contact"
 
-#: models.py:352
+#: aleksis/core/models.py:413
 #, fuzzy
 #| msgid "Last name"
 msgid "Long name"
 msgstr "Nom de famille"
 
-#: models.py:362 templates/core/group/full.html:85
+#: aleksis/core/models.py:423 aleksis/core/templates/core/group/full.html:85
 msgid "Members"
 msgstr ""
 
-#: models.py:365 templates/core/group/full.html:82
+#: aleksis/core/models.py:426 aleksis/core/templates/core/group/full.html:82
 msgid "Owners"
 msgstr "Propriétaires"
 
-#: models.py:372 templates/core/group/full.html:55
+#: aleksis/core/models.py:433 aleksis/core/templates/core/group/full.html:55
 msgid "Parent groups"
 msgstr ""
 
-#: models.py:380
+#: aleksis/core/models.py:441
 msgid "Type of group"
 msgstr ""
 
-#: models.py:457
+#: aleksis/core/models.py:559
 msgid "User"
 msgstr ""
 
-#: models.py:460 models.py:484 models.py:569
-#: templates/core/announcement/list.html:18
+#: aleksis/core/models.py:562 aleksis/core/models.py:586
+#: aleksis/core/models.py:671
+#: aleksis/core/templates/core/announcement/list.html:18
 msgid "Title"
 msgstr ""
 
-#: models.py:463
+#: aleksis/core/models.py:565
 msgid "Application"
 msgstr ""
 
-#: models.py:469
+#: aleksis/core/models.py:571
 msgid "Activity"
 msgstr ""
 
-#: models.py:470
+#: aleksis/core/models.py:572
 msgid "Activities"
 msgstr ""
 
-#: models.py:476
+#: aleksis/core/models.py:578
 msgid "Sender"
 msgstr ""
 
-#: models.py:481
+#: aleksis/core/models.py:583
 msgid "Recipient"
 msgstr ""
 
-#: models.py:486 models.py:786
+#: aleksis/core/models.py:588 aleksis/core/models.py:928
 msgid "Link"
 msgstr ""
 
-#: models.py:488
+#: aleksis/core/models.py:590
 msgid "Read"
 msgstr ""
 
-#: models.py:489
+#: aleksis/core/models.py:591
 msgid "Sent"
 msgstr ""
 
-#: models.py:502
+#: aleksis/core/models.py:604
 msgid "Notification"
 msgstr ""
 
-#: models.py:503
-msgid "Notifications"
-msgstr ""
-
-#: models.py:571
+#: aleksis/core/models.py:673
 msgid "Link to detailed view"
 msgstr ""
 
-#: models.py:574
+#: aleksis/core/models.py:676
 msgid "Date and time from when to show"
 msgstr ""
 
-#: models.py:577
+#: aleksis/core/models.py:679
 msgid "Date and time until when to show"
 msgstr ""
 
-#: models.py:601
+#: aleksis/core/models.py:703
 msgid "Announcement"
 msgstr ""
 
-#: models.py:639
+#: aleksis/core/models.py:741
 msgid "Announcement recipient"
 msgstr ""
 
-#: models.py:640
+#: aleksis/core/models.py:742
 msgid "Announcement recipients"
 msgstr ""
 
-#: models.py:690
+#: aleksis/core/models.py:797
 msgid "Widget Title"
 msgstr ""
 
-#: models.py:691
+#: aleksis/core/models.py:798
 msgid "Activate Widget"
 msgstr ""
 
-#: models.py:694
+#: aleksis/core/models.py:799
+msgid "Widget is broken"
+msgstr ""
+
+#: aleksis/core/models.py:802
 msgid "Size on mobile devices"
 msgstr ""
 
-#: models.py:695
+#: aleksis/core/models.py:803
 msgid "<= 600 px, 12 columns"
 msgstr ""
 
-#: models.py:700
+#: aleksis/core/models.py:808
 msgid "Size on tablet devices"
 msgstr ""
 
-#: models.py:701
+#: aleksis/core/models.py:809
 msgid "> 600 px, 12 columns"
 msgstr ""
 
-#: models.py:706
+#: aleksis/core/models.py:814
 msgid "Size on desktop devices"
 msgstr ""
 
-#: models.py:707
+#: aleksis/core/models.py:815
 msgid "> 992 px, 12 columns"
 msgstr ""
 
-#: models.py:712
+#: aleksis/core/models.py:820
 msgid "Size on large desktop devices"
 msgstr ""
 
-#: models.py:713
+#: aleksis/core/models.py:821
 msgid "> 1200 px>, 12 columns"
 msgstr ""
 
-#: models.py:734
+#: aleksis/core/models.py:852
 msgid "Can edit default dashboard"
 msgstr ""
 
-#: models.py:735
+#: aleksis/core/models.py:853
 msgid "Dashboard Widget"
 msgstr ""
 
-#: models.py:736
+#: aleksis/core/models.py:854
 msgid "Dashboard Widgets"
 msgstr ""
 
-#: models.py:741
+#: aleksis/core/models.py:860
+msgid "URL"
+msgstr ""
+
+#: aleksis/core/models.py:861
+msgid "Icon URL"
+msgstr ""
+
+#: aleksis/core/models.py:867
+msgid "External link widget"
+msgstr ""
+
+#: aleksis/core/models.py:868
+msgid "External link widgets"
+msgstr ""
+
+#: aleksis/core/models.py:873
 msgid "Dashboard widget"
 msgstr ""
 
-#: models.py:746
+#: aleksis/core/models.py:878
 msgid "Order"
 msgstr ""
 
-#: models.py:747
+#: aleksis/core/models.py:879
 msgid "Part of the default dashboard"
 msgstr ""
 
-#: models.py:755
+#: aleksis/core/models.py:894
 msgid "Dashboard widget order"
 msgstr ""
 
-#: models.py:756
+#: aleksis/core/models.py:895
 msgid "Dashboard widget orders"
 msgstr ""
 
-#: models.py:762
+#: aleksis/core/models.py:901
 msgid "Menu ID"
 msgstr ""
 
-#: models.py:775
+#: aleksis/core/models.py:914
 msgid "Custom menu"
 msgstr ""
 
-#: models.py:776
+#: aleksis/core/models.py:915
 msgid "Custom menus"
 msgstr ""
 
-#: models.py:783
+#: aleksis/core/models.py:925
 msgid "Menu"
 msgstr ""
 
-#: models.py:787
+#: aleksis/core/models.py:929
 msgid "Icon"
 msgstr ""
 
-#: models.py:793
+#: aleksis/core/models.py:935
 msgid "Custom menu item"
 msgstr ""
 
-#: models.py:794
+#: aleksis/core/models.py:936
 msgid "Custom menu items"
 msgstr ""
 
-#: models.py:804
+#: aleksis/core/models.py:950
 msgid "Title of type"
 msgstr ""
 
-#: models.py:811 templates/core/group/full.html:47
+#: aleksis/core/models.py:957 aleksis/core/templates/core/group/full.html:47
 #, fuzzy
 #| msgid "Group"
 msgid "Group type"
 msgstr "Groupe"
 
-#: models.py:821
+#: aleksis/core/models.py:971
 #, fuzzy
 #| msgid "Contact details"
 msgid "Can view system status"
 msgstr "Détails de contact"
 
-#: models.py:822
-#, fuzzy
-#| msgid "Contact details"
-msgid "Can link persons to accounts"
-msgstr "Détails de contact"
-
-#: models.py:823
+#: aleksis/core/models.py:972
 msgid "Can manage data"
 msgstr ""
 
-#: models.py:824
+#: aleksis/core/models.py:973
 #, fuzzy
 #| msgid "Contact details"
 msgid "Can impersonate"
 msgstr "Détails de contact"
 
-#: models.py:825
+#: aleksis/core/models.py:974
 msgid "Can use search"
 msgstr ""
 
-#: models.py:826
+#: aleksis/core/models.py:975
 msgid "Can change site preferences"
 msgstr ""
 
-#: models.py:827
+#: aleksis/core/models.py:976
 msgid "Can change person preferences"
 msgstr ""
 
-#: models.py:828
+#: aleksis/core/models.py:977
 msgid "Can change group preferences"
 msgstr ""
 
-#: models.py:864
+#: aleksis/core/models.py:978
+msgid "Can add oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:979
+msgid "Can list oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:980
+#, fuzzy
+#| msgid "Contact details"
+msgid "Can view oauth applications"
+msgstr "Détails de contact"
+
+#: aleksis/core/models.py:981
+msgid "Can update oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:982
+msgid "Can delete oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:983
+msgid "Can test PDF generation"
+msgstr ""
+
+#: aleksis/core/models.py:1019
 msgid "Related data check task"
 msgstr ""
 
-#: models.py:872
+#: aleksis/core/models.py:1027
 msgid "Issue solved"
 msgstr ""
 
-#: models.py:873
+#: aleksis/core/models.py:1028
 msgid "Notification sent"
 msgstr ""
 
-#: models.py:886
+#: aleksis/core/models.py:1041
 msgid "Data check result"
 msgstr ""
 
-#: models.py:887
+#: aleksis/core/models.py:1042
 msgid "Data check results"
 msgstr ""
 
-#: models.py:889
+#: aleksis/core/models.py:1044
 msgid "Can run data checks"
 msgstr ""
 
-#: models.py:890
+#: aleksis/core/models.py:1045
 msgid "Can solve data check problems"
 msgstr ""
 
-#: preferences.py:27
+#: aleksis/core/models.py:1060
+#, fuzzy
+#| msgid "Owners"
+msgid "Owner"
+msgstr "Propriétaires"
+
+#: aleksis/core/models.py:1064
+msgid "File expires at"
+msgstr ""
+
+#: aleksis/core/models.py:1066
+msgid "Generated HTML file"
+msgstr ""
+
+#: aleksis/core/models.py:1068
+msgid "Generated PDF file"
+msgstr ""
+
+#: aleksis/core/models.py:1075
+msgid "PDF file"
+msgstr ""
+
+#: aleksis/core/models.py:1076
+msgid "PDF files"
+msgstr ""
+
+#: aleksis/core/models.py:1081
+msgid "Task result"
+msgstr ""
+
+#: aleksis/core/models.py:1084
+msgid "Task user"
+msgstr ""
+
+#: aleksis/core/models.py:1096
+msgid "Task user assignment"
+msgstr ""
+
+#: aleksis/core/models.py:1097
+msgid "Task user assignments"
+msgstr ""
+
+#: aleksis/core/preferences.py:22
+msgid "General"
+msgstr ""
+
+#: aleksis/core/preferences.py:23
+msgid "School"
+msgstr ""
+
+#: aleksis/core/preferences.py:24
+msgid "Theme"
+msgstr ""
+
+#: aleksis/core/preferences.py:25
+msgid "Mail"
+msgstr ""
+
+#: aleksis/core/preferences.py:27
+msgid "Footer"
+msgstr ""
+
+#: aleksis/core/preferences.py:28
+#, fuzzy
+#| msgid "Contact details"
+msgid "Accounts"
+msgstr "Détails de contact"
+
+#: aleksis/core/preferences.py:29
 msgid "Authentication"
 msgstr ""
 
-#: preferences.py:28
+#: aleksis/core/preferences.py:30
 msgid "Internationalisation"
 msgstr ""
 
-#: preferences.py:37
+#: aleksis/core/preferences.py:41
 msgid "Site title"
 msgstr ""
 
-#: preferences.py:46
+#: aleksis/core/preferences.py:52
 #, fuzzy
 #| msgid "Description"
 msgid "Site description"
 msgstr "Description"
 
-#: preferences.py:55
+#: aleksis/core/preferences.py:63
 msgid "Primary colour"
 msgstr ""
 
-#: preferences.py:64
+#: aleksis/core/preferences.py:75
 msgid "Secondary colour"
 msgstr ""
 
-#: preferences.py:72
+#: aleksis/core/preferences.py:86
 msgid "Logo"
 msgstr ""
 
-#: preferences.py:80
+#: aleksis/core/preferences.py:96
 msgid "Favicon"
 msgstr ""
 
-#: preferences.py:88
+#: aleksis/core/preferences.py:106
 msgid "PWA-Icon"
 msgstr ""
 
-#: preferences.py:97
+#: aleksis/core/preferences.py:117
 #, fuzzy
 #| msgid "Last name"
 msgid "Mail out name"
 msgstr "Nom de famille"
 
-#: preferences.py:106
+#: aleksis/core/preferences.py:128
 msgid "Mail out address"
 msgstr ""
 
-#: preferences.py:116
+#: aleksis/core/preferences.py:140
 msgid "Link to privacy policy"
 msgstr ""
 
-#: preferences.py:126
+#: aleksis/core/preferences.py:152
 msgid "Link to imprint"
 msgstr ""
 
-#: preferences.py:136
+#: aleksis/core/preferences.py:164
 msgid "Name format for addressing"
 msgstr ""
 
-#: preferences.py:150
+#: aleksis/core/preferences.py:180
 msgid "Channels to use for notifications"
 msgstr ""
 
-#: preferences.py:160
+#: aleksis/core/preferences.py:192
 msgid "Regular expression to match primary group, e.g. '^Class .*'"
 msgstr ""
 
-#: preferences.py:169
+#: aleksis/core/preferences.py:203
 msgid "Field on person to match primary group against"
 msgstr ""
 
-#: preferences.py:181
+#: aleksis/core/preferences.py:215
+msgid "Automatically create new persons for new users"
+msgstr ""
+
+#: aleksis/core/preferences.py:224
+msgid "Automatically link existing persons to new users by their e-mail address"
+msgstr ""
+
+#: aleksis/core/preferences.py:235
 msgid "Display name of the school"
 msgstr ""
 
-#: preferences.py:190
+#: aleksis/core/preferences.py:246
 msgid "Official name of the school, e.g. as given by supervisory authority"
 msgstr ""
 
-#: preferences.py:198
-msgid "Enabled custom authentication backends"
+#: aleksis/core/preferences.py:254
+msgid "Allow users to change their passwords"
+msgstr ""
+
+#: aleksis/core/preferences.py:262
+msgid "Enable signup"
 msgstr ""
 
-#: preferences.py:211
+#: aleksis/core/preferences.py:273
 msgid "Available languages"
 msgstr ""
 
-#: preferences.py:223
+#: aleksis/core/preferences.py:285
 msgid "Send emails if data checks detect problems"
 msgstr ""
 
-#: preferences.py:234
+#: aleksis/core/preferences.py:296
 msgid "Email recipients for data checks problem emails"
 msgstr ""
 
-#: preferences.py:245
+#: aleksis/core/preferences.py:307
 msgid "Email recipient groups for data checks problem emails"
 msgstr ""
 
-#: settings.py:322
-msgid "English"
+#: aleksis/core/preferences.py:316
+msgid "Show dashboard to users without login"
 msgstr ""
 
-#: settings.py:323
-msgid "German"
+#: aleksis/core/preferences.py:325
+msgid "Allow users to edit their dashboard"
+msgstr ""
+
+#: aleksis/core/preferences.py:336
+msgid "Fields on person model which are editable by themselves."
+msgstr ""
+
+#: aleksis/core/preferences.py:350
+msgid "Editable fields on person model which should trigger a notification on change"
+msgstr ""
+
+#: aleksis/core/preferences.py:363
+msgid "Contact for notification if a person changes their data"
+msgstr ""
+
+#: aleksis/core/preferences.py:373
+msgid "PDF file expiration duration"
+msgstr ""
+
+#: aleksis/core/preferences.py:374
+msgid "in minutes"
+msgstr ""
+
+#: aleksis/core/preferences.py:384
+msgid "Automatically update the dashboard and its widgets"
 msgstr ""
 
-#: settings.py:324
-msgid "French"
+#: aleksis/core/preferences.py:394
+msgid "Automatically update the dashboard and its widgets sitewide"
 msgstr ""
 
-#: settings.py:325
-msgid "Norwegian (bokmål)"
+#: aleksis/core/settings.py:452
+msgid "English"
+msgstr ""
+
+#: aleksis/core/settings.py:453
+msgid "German"
 msgstr ""
 
-#: tables.py:19 templates/core/announcement/list.html:36
-#: templates/core/group/full.html:24 templates/core/person/full.html:23
+#: aleksis/core/tables.py:19
+#: aleksis/core/templates/core/announcement/list.html:36
+#: aleksis/core/templates/core/group/full.html:24
+#: aleksis/core/templates/core/person/full.html:23
+#: aleksis/core/templates/oauth2_provider/application_detail.html:17
 msgid "Edit"
 msgstr ""
 
-#: tables.py:21 tables.py:89 templates/core/announcement/list.html:22
+#: aleksis/core/tables.py:21 aleksis/core/tables.py:89
+#: aleksis/core/templates/core/announcement/list.html:22
 msgid "Actions"
 msgstr ""
 
-#: tables.py:56 tables.py:57 tables.py:71 tables.py:87
-#: templates/core/announcement/list.html:42 templates/core/group/full.html:31
-#: templates/core/pages/delete.html:22 templates/core/person/full.html:30
+#: aleksis/core/tables.py:56 aleksis/core/tables.py:57
+#: aleksis/core/tables.py:71 aleksis/core/tables.py:87
+#: aleksis/core/templates/core/announcement/list.html:42
+#: aleksis/core/templates/core/group/full.html:31
+#: aleksis/core/templates/core/pages/delete.html:22
+#: aleksis/core/templates/core/person/full.html:30
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:20
+#: aleksis/core/templates/oauth2_provider/application_detail.html:21
 msgid "Delete"
 msgstr ""
 
-#: templates/403.html:14 templates/404.html:10 templates/500.html:10
+#: aleksis/core/templates/403.html:14 aleksis/core/templates/404.html:10
+#: aleksis/core/templates/500.html:10
+#: aleksis/core/templates/oauth2_provider/authorize.html:47
+#: aleksis/core/templates/oauth2_provider/authorized-oob.html:24
 msgid "Error"
 msgstr ""
 
-#: templates/403.html:14
+#: aleksis/core/templates/403.html:14
 msgid ""
 "You are not allowed to access the requested page or\n"
 "          object."
 msgstr ""
 
-#: templates/403.html:19 templates/404.html:17
+#: aleksis/core/templates/403.html:19 aleksis/core/templates/404.html:17
 msgid ""
 "\n"
 "            If you think this is an error in AlekSIS, please contact your site\n"
@@ -887,13 +1186,13 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/404.html:10
+#: aleksis/core/templates/404.html:10
 msgid ""
 "The requested page or object was not\n"
 "          found."
 msgstr ""
 
-#: templates/404.html:13
+#: aleksis/core/templates/404.html:13
 msgid ""
 "\n"
 "            If you were redirected by a link on an external page,\n"
@@ -901,13 +1200,13 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/500.html:10
+#: aleksis/core/templates/500.html:10
 msgid ""
 "An unexpected error has\n"
 "          occured."
 msgstr ""
 
-#: templates/500.html:13
+#: aleksis/core/templates/500.html:13
 msgid ""
 "\n"
 "            Your site administrators will automatically be notified about this\n"
@@ -915,171 +1214,400 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/503.html:10
+#: aleksis/core/templates/503.html:10
 msgid ""
 "The maintenance mode is currently enabled. Please try again\n"
 "          later."
 msgstr ""
 
-#: templates/503.html:13
+#: aleksis/core/templates/503.html:13
 msgid ""
 "\n"
 "            This page is currently unavailable. If this error persists, contact your site administrators:\n"
 "          "
 msgstr ""
 
-#: templates/core/additional_field/edit.html:6
-#: templates/core/additional_field/edit.html:7
+#: aleksis/core/templates/account/account_inactive.html:5
+#: aleksis/core/templates/account/account_inactive.html:6
+msgid "Account inactive"
+msgstr ""
+
+#: aleksis/core/templates/account/account_inactive.html:13
+msgid "Account inactive."
+msgstr ""
+
+#: aleksis/core/templates/account/account_inactive.html:15
+msgid ""
+"\n"
+"            This account is currently inactive. If you think this is an\n"
+"            error, please contact one of your site administrators.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/email/base_message.txt:5
+msgid "Hello!"
+msgstr ""
+
+#: aleksis/core/templates/account/email/base_message.txt:9
+#: aleksis/core/templates/templated_email/notification.email:22
+#: aleksis/core/templates/templated_email/notification.email:46
+msgid "Your AlekSIS team"
+msgstr ""
+
+#: aleksis/core/templates/account/email_confirm.html:5
+#: aleksis/core/templates/account/email_confirm.html:6
+#: aleksis/core/templates/account/email_confirm.html:17
+msgid "Confirm"
+msgstr ""
+
+#: aleksis/core/templates/account/email_confirm.html:12
+#, python-format
+msgid "Please confirm that <a href=\"mailto:%(email)s\">%(email)s</a> is an e-mail address for user %(user_display)s."
+msgstr ""
+
+#: aleksis/core/templates/account/email_confirm.html:25
+#, python-format
+msgid "This e-mail confirmation link expired or is invalid. Please <a href=\"%(email_url)s\">issue a new e-mail confirmation request</a>."
+msgstr ""
+
+#: aleksis/core/templates/account/password_change.html:12
+msgid "Forgot your current password? Click here to reset it:"
+msgstr ""
+
+#: aleksis/core/templates/account/password_change.html:12
+msgid "Forgot Password?"
+msgstr ""
+
+#: aleksis/core/templates/account/password_change_disabled.html:5
+#: aleksis/core/templates/account/password_change_disabled.html:6
+msgid "Changing of password disabled"
+msgstr ""
+
+#: aleksis/core/templates/account/password_change_disabled.html:13
+msgid "Changing of password disabled."
+msgstr ""
+
+#: aleksis/core/templates/account/password_change_disabled.html:15
+msgid ""
+"\n"
+"            Users are not allowed to edit their own passwords. If you think\n"
+"            this is an error please contact one of your site administrators.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset.html:5
+#: aleksis/core/templates/account/password_reset.html:15
+#: aleksis/core/templates/account/password_reset.html:23
+#: aleksis/core/templates/account/password_reset_done.html:5
+#: aleksis/core/templates/account/verification_email_required.html:5
+#: aleksis/core/templates/account/verification_email_required.html:6
+#: aleksis/core/templates/two_factor/core/login.html:81
+msgid "Reset password"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset.html:17
+msgid "Forgotten your password? Enter your e-mail address below, and we'll send you an e-mail allowing you to reset it."
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset.html:30
+msgid ""
+"Please contact one of your site administrators, if you\n"
+"        have any trouble resetting your password:"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_done.html:15
+msgid "Password reset mail sent"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_done.html:18
+#: aleksis/core/templates/account/verification_email_required.html:16
+msgid ""
+"\n"
+"            We have sent you an e-mail. Please contact one of your site\n"
+"            administrators if you do not receive it within a few minutes.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:15
+msgid "Bad token"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:19
+#, python-format
+msgid ""
+"\n"
+"              The password reset link was invalid, possibly because it has already been used. Please request a <a href=\"%(passwd_reset_url)s\"\n"
+"              class=\"blue-text text-lighten-2\">new password reset</a>.\n"
+"            "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:25
+msgid ""
+"\n"
+"              If this issue persists, please contact one of your site\n"
+"              administrators\n"
+"            "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:56
+#: aleksis/core/templates/account/password_reset_from_key_done.html:15
+msgid ""
+"\n"
+"            Your password is now changed!\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:61
+msgid "Back to login"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key_done.html:13
+msgid "Password changed!"
+msgstr ""
+
+#: aleksis/core/templates/account/password_set.html:5
+#: aleksis/core/templates/account/password_set.html:6
+#: aleksis/core/templates/account/password_set.html:12
+msgid "Set password"
+msgstr ""
+
+#: aleksis/core/templates/account/signup.html:5
+#: aleksis/core/templates/account/signup.html:6
+#: aleksis/core/templates/socialaccount/signup.html:5
+#: aleksis/core/templates/socialaccount/signup.html:6
+msgid "Signup"
+msgstr ""
+
+#: aleksis/core/templates/account/signup.html:12
+#, python-format
+msgid "Already have an account? Then please <a href=\"%(login_url)s\">sign in</a>."
+msgstr ""
+
+#: aleksis/core/templates/account/signup_closed.html:5
+#: aleksis/core/templates/account/signup_closed.html:6
+msgid "Signup closed"
+msgstr ""
+
+#: aleksis/core/templates/account/signup_closed.html:13
+msgid "Signup closed."
+msgstr ""
+
+#: aleksis/core/templates/account/signup_closed.html:15
+msgid ""
+"\n"
+"            This sign up is currently closed. If you think this is an\n"
+"            error, please contact one of your site administrators.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/verification_email_required.html:14
+msgid "Password reset mail sent!"
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:5
+#: aleksis/core/templates/account/verification_sent.html:6
+msgid "Verify your email address"
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:14
+msgid "Verify your email!"
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:16
+msgid ""
+"\n"
+"            This part of the site requires us to verify that you are who you claim to be.\n"
+"            For this purpose, we require that you verify ownership of your e-mail address.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:22
+msgid ""
+"\n"
+"            We have sent an e-mail to you for verification.\n"
+"            Please click on the link inside this e-mail. Please\n"
+"            contact us if you do not receive it within a few minutes.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:30
+#, python-format
+msgid "<strong>Note:</strong> you can still <a href=\"%(email_url)s\">change your e-mail address</a>"
+msgstr ""
+
+#: aleksis/core/templates/core/additional_field/edit.html:6
+#: aleksis/core/templates/core/additional_field/edit.html:7
 msgid "Edit additional field"
 msgstr ""
 
-#: templates/core/additional_field/list.html:14
+#: aleksis/core/templates/core/additional_field/list.html:14
 msgid "Create additional field"
 msgstr ""
 
-#: templates/core/announcement/form.html:14
-#: templates/core/announcement/form.html:21
+#: aleksis/core/templates/core/announcement/form.html:14
+#: aleksis/core/templates/core/announcement/form.html:21
 msgid "Edit announcement"
 msgstr ""
 
-#: templates/core/announcement/form.html:16
+#: aleksis/core/templates/core/announcement/form.html:16
 msgid "Publish announcement"
 msgstr ""
 
-#: templates/core/announcement/form.html:23
-#: templates/core/announcement/list.html:13
+#: aleksis/core/templates/core/announcement/form.html:23
+#: aleksis/core/templates/core/announcement/list.html:13
 msgid "Publish new announcement"
 msgstr ""
 
-#: templates/core/announcement/form.html:34
+#: aleksis/core/templates/core/announcement/form.html:34
 msgid "Save und publish announcement"
 msgstr ""
 
-#: templates/core/announcement/list.html:19
+#: aleksis/core/templates/core/announcement/list.html:19
 msgid "Valid from"
 msgstr ""
 
-#: templates/core/announcement/list.html:20
+#: aleksis/core/templates/core/announcement/list.html:20
 msgid "Valid until"
 msgstr ""
 
-#: templates/core/announcement/list.html:21
+#: aleksis/core/templates/core/announcement/list.html:21
 msgid "Recipients"
 msgstr ""
 
-#: templates/core/announcement/list.html:50
+#: aleksis/core/templates/core/announcement/list.html:50
 msgid "There are no announcements."
 msgstr ""
 
-#: templates/core/base.html:54
+#: aleksis/core/templates/core/base.html:60
 msgid "Logged in as"
 msgstr ""
 
-#: templates/core/base.html:147
+#: aleksis/core/templates/core/base.html:154
 msgid "About AlekSIS — The Free School Information System"
 msgstr ""
 
-#: templates/core/base.html:155
+#: aleksis/core/templates/core/base.html:162
 msgid "Impress"
 msgstr ""
 
-#: templates/core/base.html:163
+#: aleksis/core/templates/core/base.html:170
 msgid "Privacy Policy"
 msgstr ""
 
-#: templates/core/base_print.html:64
+#: aleksis/core/templates/core/base_print.html:72
 msgid "Powered by AlekSIS"
 msgstr ""
 
-#: templates/core/dashboard_widget/create.html:8
-#: templates/core/dashboard_widget/create.html:12
+#: aleksis/core/templates/core/dashboard_widget/create.html:8
+#: aleksis/core/templates/core/dashboard_widget/create.html:12
 #, fuzzy, python-format
 #| msgid "Contact details"
 msgid "Create %(widget)s"
 msgstr "Détails de contact"
 
-#: templates/core/dashboard_widget/edit.html:8
-#: templates/core/dashboard_widget/edit.html:12
+#: aleksis/core/templates/core/dashboard_widget/dashboardwidget_broken.html:10
+msgid "This widget is currently not available."
+msgstr ""
+
+#: aleksis/core/templates/core/dashboard_widget/dashboardwidget_broken.html:14
+#, python-format
+msgid ""
+"\n"
+"            There is a problem getting the widget \"%(title)s\".\n"
+"            There is no need for you to take any action.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/core/dashboard_widget/edit.html:8
+#: aleksis/core/templates/core/dashboard_widget/edit.html:12
 #, python-format
 msgid "Edit %(widget)s"
 msgstr ""
 
-#: templates/core/dashboard_widget/list.html:17
+#: aleksis/core/templates/core/dashboard_widget/list.html:15
+#, fuzzy
+#| msgid "Contact details"
+msgid "Create dashboard widget"
+msgstr "Détails de contact"
+
+#: aleksis/core/templates/core/dashboard_widget/list.html:22
 #, fuzzy, python-format
 #| msgid "Contact details"
 msgid "Create %(name)s"
 msgstr "Détails de contact"
 
-#: templates/core/dashboard_widget/list.html:25
-#: templates/core/edit_dashboard.html:8 templates/core/edit_dashboard.html:15
+#: aleksis/core/templates/core/dashboard_widget/list.html:32
+#: aleksis/core/templates/core/edit_dashboard.html:8
+#: aleksis/core/templates/core/edit_dashboard.html:15
 msgid "Edit default dashboard"
 msgstr ""
 
-#: templates/core/data_check/list.html:15
+#: aleksis/core/templates/core/data_check/list.html:15
 msgid "Check data again"
 msgstr ""
 
-#: templates/core/data_check/list.html:22
+#: aleksis/core/templates/core/data_check/list.html:22
 msgid "The system detected some problems with your data."
 msgstr ""
 
-#: templates/core/data_check/list.html:23
+#: aleksis/core/templates/core/data_check/list.html:23
 msgid ""
 "Please go through all data and check whether some extra action is\n"
 "          needed."
 msgstr ""
 
-#: templates/core/data_check/list.html:31
+#: aleksis/core/templates/core/data_check/list.html:31
 msgid "Everything is fine."
 msgstr ""
 
-#: templates/core/data_check/list.html:32
+#: aleksis/core/templates/core/data_check/list.html:32
 msgid "The system hasn't detected any problems with your data."
 msgstr ""
 
-#: templates/core/data_check/list.html:40
+#: aleksis/core/templates/core/data_check/list.html:40
 msgid "Detected problems"
 msgstr ""
 
-#: templates/core/data_check/list.html:45
+#: aleksis/core/templates/core/data_check/list.html:45
 msgid "Affected object"
 msgstr ""
 
-#: templates/core/data_check/list.html:46
+#: aleksis/core/templates/core/data_check/list.html:46
 msgid "Detected problem"
 msgstr ""
 
-#: templates/core/data_check/list.html:47
+#: aleksis/core/templates/core/data_check/list.html:47
 #, fuzzy
 #| msgid "Contact details"
 msgid "Show details"
 msgstr "Détails de contact"
 
-#: templates/core/data_check/list.html:48
+#: aleksis/core/templates/core/data_check/list.html:48
 msgid "Options to solve the problem"
 msgstr ""
 
-#: templates/core/data_check/list.html:62
+#: aleksis/core/templates/core/data_check/list.html:62
 msgid "Show object"
 msgstr ""
 
-#: templates/core/data_check/list.html:84
+#: aleksis/core/templates/core/data_check/list.html:84
 msgid "Registered checks"
 msgstr ""
 
-#: templates/core/data_check/list.html:88
+#: aleksis/core/templates/core/data_check/list.html:88
 msgid ""
 "\n"
 "            The system will check for the following problems:\n"
 "          "
 msgstr ""
 
-#: templates/core/edit_dashboard.html:6 templates/core/edit_dashboard.html:13
-#: templates/core/index.html:14
+#: aleksis/core/templates/core/edit_dashboard.html:6
+#: aleksis/core/templates/core/edit_dashboard.html:13
+#: aleksis/core/templates/core/index.html:17
 msgid "Edit dashboard"
 msgstr ""
 
-#: templates/core/edit_dashboard.html:24
+#: aleksis/core/templates/core/edit_dashboard.html:24
 msgid ""
 "\n"
 "          On this page you can arrange your personal dashboard. You can drag any items from \"Available widgets\" to \"Your\n"
@@ -1088,7 +1616,7 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/core/edit_dashboard.html:30
+#: aleksis/core/templates/core/edit_dashboard.html:30
 msgid ""
 "\n"
 "          On this page you can arrange the default dashboard which is shown when a user doesn't arrange his own\n"
@@ -1097,19 +1625,19 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/core/edit_dashboard.html:48
+#: aleksis/core/templates/core/edit_dashboard.html:48
 msgid "Available widgets"
 msgstr ""
 
-#: templates/core/edit_dashboard.html:57
+#: aleksis/core/templates/core/edit_dashboard.html:57
 msgid "Your dashboard"
 msgstr ""
 
-#: templates/core/edit_dashboard.html:59
+#: aleksis/core/templates/core/edit_dashboard.html:59
 msgid "Default dashboard"
 msgstr ""
 
-#: templates/core/group/child_groups.html:18
+#: aleksis/core/templates/core/group/child_groups.html:18
 msgid ""
 "\n"
 "          You can use this to assign child groups to groups. Please use the filters below to select groups you want to\n"
@@ -1117,38 +1645,38 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/core/group/child_groups.html:31
+#: aleksis/core/templates/core/group/child_groups.html:31
 msgid "Update selection"
 msgstr ""
 
-#: templates/core/group/child_groups.html:35
+#: aleksis/core/templates/core/group/child_groups.html:35
 msgid "Clear all filters"
 msgstr ""
 
-#: templates/core/group/child_groups.html:39
+#: aleksis/core/templates/core/group/child_groups.html:39
 msgid "Currently selected groups"
 msgstr ""
 
-#: templates/core/group/child_groups.html:52
+#: aleksis/core/templates/core/group/child_groups.html:52
 msgid "Start assigning child groups for this groups"
 msgstr ""
 
-#: templates/core/group/child_groups.html:61
+#: aleksis/core/templates/core/group/child_groups.html:61
 msgid ""
 "\n"
 "            Please select some groups in order to go on with assigning.\n"
 "          "
 msgstr ""
 
-#: templates/core/group/child_groups.html:72
+#: aleksis/core/templates/core/group/child_groups.html:72
 msgid "Current group:"
 msgstr ""
 
-#: templates/core/group/child_groups.html:78
+#: aleksis/core/templates/core/group/child_groups.html:78
 msgid "Please be careful!"
 msgstr ""
 
-#: templates/core/group/child_groups.html:79
+#: aleksis/core/templates/core/group/child_groups.html:79
 msgid ""
 "\n"
 "            If you click \"Back\" or \"Next\" the current group assignments are not saved.\n"
@@ -1157,131 +1685,139 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/core/group/child_groups.html:93
-#: templates/core/group/child_groups.html:128
-#: templates/two_factor/_wizard_actions.html:15
-#: templates/two_factor/_wizard_actions.html:20
+#: aleksis/core/templates/core/group/child_groups.html:93
+#: aleksis/core/templates/core/group/child_groups.html:128
+#: aleksis/core/templates/oauth2_provider/application_detail.html:9
+#: aleksis/core/templates/two_factor/_wizard_actions.html:15
+#: aleksis/core/templates/two_factor/_wizard_actions.html:20
 msgid "Back"
 msgstr ""
 
-#: templates/core/group/child_groups.html:99
-#: templates/core/group/child_groups.html:134
-#: templates/two_factor/_wizard_actions.html:26
+#: aleksis/core/templates/core/group/child_groups.html:99
+#: aleksis/core/templates/core/group/child_groups.html:134
+#: aleksis/core/templates/two_factor/_wizard_actions.html:26
 msgid "Next"
 msgstr ""
 
-#: templates/core/group/child_groups.html:106
-#: templates/core/group/child_groups.html:141
-#: templates/core/partials/save_button.html:3
+#: aleksis/core/templates/core/group/child_groups.html:106
+#: aleksis/core/templates/core/group/child_groups.html:141
+#: aleksis/core/templates/core/partials/save_button.html:3
 msgid "Save"
 msgstr ""
 
-#: templates/core/group/child_groups.html:112
-#: templates/core/group/child_groups.html:147
+#: aleksis/core/templates/core/group/child_groups.html:112
+#: aleksis/core/templates/core/group/child_groups.html:147
 msgid "Save and next"
 msgstr ""
 
-#: templates/core/group/edit.html:11 templates/core/group/edit.html:12
+#: aleksis/core/templates/core/group/edit.html:11
+#: aleksis/core/templates/core/group/edit.html:12
 msgid "Edit group"
 msgstr ""
 
-#: templates/core/group/full.html:38 templates/core/person/full.html:37
+#: aleksis/core/templates/core/group/full.html:38
+#: aleksis/core/templates/core/person/full.html:37
 msgid "Change preferences"
 msgstr ""
 
-#: templates/core/group/full.html:64
+#: aleksis/core/templates/core/group/full.html:64
 msgid "Statistics"
 msgstr ""
 
-#: templates/core/group/full.html:67
+#: aleksis/core/templates/core/group/full.html:67
 msgid "Count of members"
 msgstr ""
 
-#: templates/core/group/full.html:71
+#: aleksis/core/templates/core/group/full.html:71
 msgid "Average age"
 msgstr ""
 
-#: templates/core/group/full.html:76
+#: aleksis/core/templates/core/group/full.html:76
 msgid "Age range"
 msgstr ""
 
-#: templates/core/group/full.html:76
+#: aleksis/core/templates/core/group/full.html:76
 msgid "years to"
 msgstr ""
 
-#: templates/core/group/full.html:76
+#: aleksis/core/templates/core/group/full.html:76
 msgid "years "
 msgstr ""
 
-#: templates/core/group/list.html:14
+#: aleksis/core/templates/core/group/list.html:14
 msgid "Create group"
 msgstr ""
 
-#: templates/core/group/list.html:17
+#: aleksis/core/templates/core/group/list.html:17
 msgid "Filter groups"
 msgstr ""
 
-#: templates/core/group/list.html:24 templates/core/person/list.html:28
+#: aleksis/core/templates/core/group/list.html:24
+#: aleksis/core/templates/core/person/list.html:28
 msgid "Clear"
 msgstr ""
 
-#: templates/core/group/list.html:28
+#: aleksis/core/templates/core/group/list.html:28
 msgid "Selected groups"
 msgstr ""
 
-#: templates/core/group_type/edit.html:6 templates/core/group_type/edit.html:7
+#: aleksis/core/templates/core/group_type/edit.html:6
+#: aleksis/core/templates/core/group_type/edit.html:7
 #, fuzzy
 #| msgid "Group"
 msgid "Edit group type"
 msgstr "Groupe"
 
-#: templates/core/group_type/list.html:14
+#: aleksis/core/templates/core/group_type/list.html:14
 #, fuzzy
 #| msgid "Group"
 msgid "Create group type"
 msgstr "Groupe"
 
-#: templates/core/index.html:4
+#: aleksis/core/templates/core/index.html:4
 msgid "Home"
 msgstr ""
 
-#: templates/core/index.html:50
+#: aleksis/core/templates/core/index.html:49
 msgid ""
 "\n"
-"          You didn't customise your dashboard so that you see the system default. Please click on \"Edit dashboard\" to\n"
-"          customise your personal dashboard.\n"
-"        "
+"        You didn't customise your dashboard so that you see the system default. Please click on \"Edit dashboard\" to\n"
+"        customise your personal dashboard.\n"
+"      "
 msgstr ""
 
-#: templates/core/index.html:59
+#: aleksis/core/templates/core/index.html:59
 msgid "Last activities"
 msgstr ""
 
-#: templates/core/index.html:77
+#: aleksis/core/templates/core/index.html:77
 msgid "No activities available yet."
 msgstr ""
 
-#: templates/core/index.html:82
+#: aleksis/core/templates/core/index.html:82
 msgid "Recent notifications"
 msgstr ""
 
-#: templates/core/index.html:98
+#: aleksis/core/templates/core/index.html:98
+#: aleksis/core/templates/core/notifications.html:23
 msgid "More information →"
 msgstr ""
 
-#: templates/core/index.html:105
+#: aleksis/core/templates/core/index.html:105
+#: aleksis/core/templates/core/notifications.html:30
 msgid "No notifications available yet."
 msgstr ""
 
-#: templates/core/pages/about.html:6 templates/core/pages/about.html:15
+#: aleksis/core/templates/core/pages/about.html:6
+#: aleksis/core/templates/core/pages/about.html:15
 msgid "About AlekSIS"
 msgstr ""
 
-#: templates/core/pages/about.html:7
+#: aleksis/core/templates/core/pages/about.html:7
 msgid "AlekSIS – The Free School Information System"
 msgstr ""
 
-#: templates/core/pages/about.html:17
+#: aleksis/core/templates/core/pages/about.html:17
 msgid ""
 "\n"
 "              This platform is powered by AlekSIS, a web-based school information system (SIS) which can be used\n"
@@ -1290,19 +1826,19 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/pages/about.html:25
+#: aleksis/core/templates/core/pages/about.html:25
 msgid "Website of AlekSIS"
 msgstr ""
 
-#: templates/core/pages/about.html:26
+#: aleksis/core/templates/core/pages/about.html:26
 msgid "Source code"
 msgstr ""
 
-#: templates/core/pages/about.html:35
+#: aleksis/core/templates/core/pages/about.html:35
 msgid "Licence information"
 msgstr ""
 
-#: templates/core/pages/about.html:37
+#: aleksis/core/templates/core/pages/about.html:37
 msgid ""
 "\n"
 "              The core and the official apps of AlekSIS are licenced under the EUPL, version 1.2 or later. For licence\n"
@@ -1311,23 +1847,23 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/pages/about.html:45
+#: aleksis/core/templates/core/pages/about.html:45
 msgid "Free/Open Source Licence"
 msgstr ""
 
-#: templates/core/pages/about.html:46
+#: aleksis/core/templates/core/pages/about.html:46
 msgid "Other Licence"
 msgstr ""
 
-#: templates/core/pages/about.html:50
+#: aleksis/core/templates/core/pages/about.html:50
 msgid "Full licence text"
 msgstr ""
 
-#: templates/core/pages/about.html:51
+#: aleksis/core/templates/core/pages/about.html:51
 msgid "More information about the EUPL"
 msgstr ""
 
-#: templates/core/pages/about.html:90
+#: aleksis/core/templates/core/pages/about.html:90
 #, python-format
 msgid ""
 "\n"
@@ -1335,12 +1871,12 @@ msgid ""
 "                  "
 msgstr ""
 
-#: templates/core/pages/delete.html:6
+#: aleksis/core/templates/core/pages/delete.html:6
 #, python-format
 msgid "Delete %(object_name)s"
 msgstr ""
 
-#: templates/core/pages/delete.html:13
+#: aleksis/core/templates/core/pages/delete.html:13
 #, python-format
 msgid ""
 "\n"
@@ -1348,104 +1884,116 @@ msgid ""
 "    "
 msgstr ""
 
-#: templates/core/pages/progress.html:27
+#: aleksis/core/templates/core/pages/progress.html:27
 msgid ""
 "\n"
 "              Without activated JavaScript the progress status can't be updated.\n"
 "            "
 msgstr ""
 
-#: templates/core/pages/progress.html:47
-#: templates/two_factor/core/otp_required.html:19
+#: aleksis/core/templates/core/pages/progress.html:47
+#: aleksis/core/templates/two_factor/core/otp_required.html:19
 msgid "Go back"
 msgstr ""
 
-#: templates/core/pages/system_status.html:12
+#: aleksis/core/templates/core/pages/system_status.html:12
 msgid "System checks"
 msgstr ""
 
-#: templates/core/pages/system_status.html:21
+#: aleksis/core/templates/core/pages/system_status.html:21
 msgid "Maintenance mode enabled"
 msgstr ""
 
-#: templates/core/pages/system_status.html:23
+#: aleksis/core/templates/core/pages/system_status.html:23
 msgid ""
 "\n"
 "                Only admin and visitors from internal IPs can access thesite.\n"
 "              "
 msgstr ""
 
-#: templates/core/pages/system_status.html:34
+#: aleksis/core/templates/core/pages/system_status.html:34
 msgid "Maintenance mode disabled"
 msgstr ""
 
-#: templates/core/pages/system_status.html:35
+#: aleksis/core/templates/core/pages/system_status.html:35
 msgid "Everyone can access the site."
 msgstr ""
 
-#: templates/core/pages/system_status.html:45
+#: aleksis/core/templates/core/pages/system_status.html:45
 msgid "Debug mode enabled"
 msgstr ""
 
-#: templates/core/pages/system_status.html:47
+#: aleksis/core/templates/core/pages/system_status.html:47
 msgid ""
 "\n"
 "                The web server throws back debug information on errors. Do not use in production!\n"
 "              "
 msgstr ""
 
-#: templates/core/pages/system_status.html:54
+#: aleksis/core/templates/core/pages/system_status.html:54
 msgid "Debug mode disabled"
 msgstr ""
 
-#: templates/core/pages/system_status.html:56
+#: aleksis/core/templates/core/pages/system_status.html:56
 msgid ""
 "\n"
 "                Debug mode is disabled. Default error pages are displayed on errors.\n"
 "              "
 msgstr ""
 
-#: templates/core/pages/system_status.html:69
+#: aleksis/core/templates/core/pages/system_status.html:69
 msgid "System health checks"
 msgstr ""
 
-#: templates/core/pages/system_status.html:75
+#: aleksis/core/templates/core/pages/system_status.html:75
 msgid "Service"
 msgstr ""
 
-#: templates/core/pages/system_status.html:76
-#: templates/core/pages/system_status.html:115
+#: aleksis/core/templates/core/pages/system_status.html:76
+#: aleksis/core/templates/core/pages/system_status.html:115
 msgid "Status"
 msgstr ""
 
-#: templates/core/pages/system_status.html:77
+#: aleksis/core/templates/core/pages/system_status.html:77
 msgid "Time taken"
 msgstr ""
 
-#: templates/core/pages/system_status.html:96
+#: aleksis/core/templates/core/pages/system_status.html:96
 msgid "seconds"
 msgstr ""
 
-#: templates/core/pages/system_status.html:107
+#: aleksis/core/templates/core/pages/system_status.html:107
 msgid "Celery task results"
 msgstr ""
 
-#: templates/core/pages/system_status.html:112
+#: aleksis/core/templates/core/pages/system_status.html:112
 msgid "Task"
 msgstr ""
 
-#: templates/core/pages/system_status.html:113
+#: aleksis/core/templates/core/pages/system_status.html:113
 msgid "ID"
 msgstr ""
 
-#: templates/core/pages/system_status.html:114
+#: aleksis/core/templates/core/pages/system_status.html:114
 #, fuzzy
 #| msgid "Date"
 msgid "Date done"
 msgstr "Date"
 
-#: templates/core/partials/announcements.html:9
-#: templates/core/partials/announcements.html:36
+#: aleksis/core/templates/core/pages/test_pdf.html:7
+#: aleksis/core/templates/core/pages/test_pdf.html:8
+msgid "Test PDF generation"
+msgstr ""
+
+#: aleksis/core/templates/core/pages/test_pdf.html:14
+msgid ""
+"\n"
+"        This simple view can be used to ensure the correct function of the built-in PDF generation system.\n"
+"      "
+msgstr ""
+
+#: aleksis/core/templates/core/partials/announcements.html:8
+#: aleksis/core/templates/core/partials/announcements.html:35
 #, python-format
 msgid ""
 "\n"
@@ -1453,7 +2001,7 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/partials/announcements.html:13
+#: aleksis/core/templates/core/partials/announcements.html:12
 #, python-format
 msgid ""
 "\n"
@@ -1461,7 +2009,7 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/partials/announcements.html:40
+#: aleksis/core/templates/core/partials/announcements.html:39
 #, python-format
 msgid ""
 "\n"
@@ -1469,23 +2017,23 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/partials/crud_events.html:15
+#: aleksis/core/templates/core/partials/crud_events.html:15
 msgid "Changed by"
 msgstr ""
 
-#: templates/core/partials/crud_events.html:15
+#: aleksis/core/templates/core/partials/crud_events.html:15
 msgid "Unknown"
 msgstr ""
 
-#: templates/core/partials/language_form.html:15
+#: aleksis/core/templates/core/partials/language_form.html:15
 msgid "Language"
 msgstr ""
 
-#: templates/core/partials/language_form.html:27
+#: aleksis/core/templates/core/partials/language_form.html:27
 msgid "Select language"
 msgstr ""
 
-#: templates/core/partials/no_person.html:12
+#: aleksis/core/templates/core/partials/no_person.html:12
 msgid ""
 "\n"
 "            Your administrator account is not linked to any person. Therefore,\n"
@@ -1493,7 +2041,7 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/core/partials/no_person.html:19
+#: aleksis/core/templates/core/partials/no_person.html:19
 msgid ""
 "\n"
 "            Your user account is not linked to a person. This means you\n"
@@ -1502,109 +2050,184 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/core/person/accounts.html:12
-#: templates/core/person/accounts.html:14
-msgid "Link persons to accounts"
-msgstr ""
-
-#: templates/core/person/accounts.html:21
-msgid ""
-"\n"
-"        You can use this form to assign user accounts to persons. Use the\n"
-"        dropdowns to select existing accounts; use the text fields to create new\n"
-"        accounts on-the-fly. The latter will create a new account with the\n"
-"        entered username and copy all other details from the person.\n"
-"      "
-msgstr ""
-
-#: templates/core/person/accounts.html:36
-#: templates/core/person/accounts.html:60
-msgid "Update"
-msgstr ""
-
-#: templates/core/person/accounts.html:42
-msgid "Existing account"
-msgstr ""
-
-#: templates/core/person/accounts.html:43
-msgid "New account"
-msgstr ""
+#: aleksis/core/templates/core/person/create.html:12
+#: aleksis/core/templates/core/person/create.html:13
+#: aleksis/core/templates/core/person/list.html:17
+#, fuzzy
+#| msgid "Contact details"
+msgid "Create person"
+msgstr "Détails de contact"
 
-#: templates/core/person/edit.html:12 templates/core/person/edit.html:13
+#: aleksis/core/templates/core/person/edit.html:12
+#: aleksis/core/templates/core/person/edit.html:13
 msgid "Edit person"
 msgstr ""
 
-#: templates/core/person/full.html:44
+#: aleksis/core/templates/core/person/full.html:44
+#: aleksis/core/templates/impersonate/list_users.html:7
+#: aleksis/core/templates/impersonate/list_users.html:8
 #, fuzzy
 #| msgid "Contact details"
 msgid "Impersonate"
 msgstr "Détails de contact"
 
-#: templates/core/person/full.html:50
+#: aleksis/core/templates/core/person/full.html:50
 msgid "Contact details"
 msgstr "Détails de contact"
 
-#: templates/core/person/full.html:130
+#: aleksis/core/templates/core/person/full.html:130
 msgid "Children"
 msgstr ""
 
-#: templates/core/person/list.html:17
-#, fuzzy
-#| msgid "Contact details"
-msgid "Create person"
-msgstr "Détails de contact"
-
-#: templates/core/person/list.html:21
+#: aleksis/core/templates/core/person/list.html:21
 msgid "Filter persons"
 msgstr ""
 
-#: templates/core/person/list.html:32
+#: aleksis/core/templates/core/person/list.html:32
 msgid "Selected persons"
 msgstr ""
 
-#: templates/core/school_term/create.html:6
-#: templates/core/school_term/create.html:7
-#: templates/core/school_term/list.html:14
+#: aleksis/core/templates/core/school_term/create.html:6
+#: aleksis/core/templates/core/school_term/create.html:7
+#: aleksis/core/templates/core/school_term/list.html:14
 msgid "Create school term"
 msgstr ""
 
-#: templates/core/school_term/edit.html:6
-#: templates/core/school_term/edit.html:7
+#: aleksis/core/templates/core/school_term/edit.html:6
+#: aleksis/core/templates/core/school_term/edit.html:7
 msgid "Edit school term"
 msgstr ""
 
-#: templates/dynamic_preferences/form.html:9
+#: aleksis/core/templates/dynamic_preferences/form.html:9
 msgid "Site preferences"
 msgstr ""
 
-#: templates/dynamic_preferences/form.html:11
+#: aleksis/core/templates/dynamic_preferences/form.html:11
 msgid "My preferences"
 msgstr ""
 
-#: templates/dynamic_preferences/form.html:13
+#: aleksis/core/templates/dynamic_preferences/form.html:13
 #, python-format
 msgid "Preferences for %(instance)s"
 msgstr ""
 
-#: templates/dynamic_preferences/form.html:25
+#: aleksis/core/templates/dynamic_preferences/form.html:25
 msgid "Save preferences"
 msgstr ""
 
-#: templates/impersonate/list_users.html:8
-msgid "Impersonate user"
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:5
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:6
+msgid "Delete application"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:12
+#, python-format
+msgid "Are you sure to delete the application %(application_name)s?"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:24
+#: aleksis/core/templates/oauth2_provider/application_form.html:18
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:24
+#: aleksis/core/templates/two_factor/_wizard_actions.html:6
+msgid "Cancel"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:27
+msgid "Client id"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:35
+msgid "Client secret"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:43
+msgid "Client type"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:51
+msgid "Authorization Grant Type"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:59
+msgid "Redirect URIs"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_form.html:5
+msgid "Create OAuth2 Application"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_form.html:10
+msgid "Edit application"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_list.html:8
+msgid "OAuth2 applications"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_list.html:12
+msgid "Register new applications"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_list.html:23
+msgid "No applications defined."
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorize.html:5
+#: aleksis/core/templates/oauth2_provider/authorize.html:16
+msgid "Authorize"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorize.html:18
+msgid "The application requests access to the following scopes:"
 msgstr ""
 
-#: templates/offline.html:5
+#: aleksis/core/templates/oauth2_provider/authorize.html:33
+msgid "Allow"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorize.html:36
+msgid "Disallow"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-oob.html:12
+msgid "Success!"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-oob.html:14
+msgid "Please return to your application and enter this code:"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:5
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:6
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:23
+msgid "Revoke access"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:12
+msgid "Are you sure to revoke the access for this application?"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:20
+msgid "Revoke"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:33
+#, fuzzy
+#| msgid "Contact details"
+msgid "No authorized applications."
+msgstr "Détails de contact"
+
+#: aleksis/core/templates/offline.html:5
 msgid "Network error"
 msgstr ""
 
-#: templates/offline.html:8
+#: aleksis/core/templates/offline.html:8
 msgid ""
 "No internet\n"
 "    connection."
 msgstr ""
 
-#: templates/offline.html:12
+#: aleksis/core/templates/offline.html:12
 msgid ""
 "\n"
 "      There was an error accessing this page. You probably don't have an internet connection. Check to see if your WiFi\n"
@@ -1613,36 +2236,116 @@ msgid ""
 "    "
 msgstr ""
 
-#: templates/search/search.html:8
+#: aleksis/core/templates/search/search.html:8
 msgid "Global Search"
 msgstr ""
 
-#: templates/search/search.html:15
+#: aleksis/core/templates/search/search.html:15
 msgid "Search Term"
 msgstr ""
 
-#: templates/search/search.html:26
+#: aleksis/core/templates/search/search.html:26
 msgid "Results"
 msgstr ""
 
-#: templates/search/search.html:38
+#: aleksis/core/templates/search/search.html:38
 msgid "No search results could be found to your search."
 msgstr ""
 
-#: templates/search/search.html:87
+#: aleksis/core/templates/search/search.html:87
 msgid "Please enter a search term above."
 msgstr ""
 
-#: templates/templated_email/data_checks.email:4
+#: aleksis/core/templates/socialaccount/authentication_error.html:5
+#: aleksis/core/templates/socialaccount/authentication_error.html:6
+msgid "Third-party Account Login Failure"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/authentication_error.html:13
+msgid "Third-party Account Login Failure."
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/authentication_error.html:15
+msgid ""
+"\n"
+"            An error occurred while attempting to login via your third-party account.\n"
+"            Please contact one of your site administrators.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/connections.html:5
+#: aleksis/core/templates/socialaccount/connections.html:6
+msgid "Connections"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/connections.html:24
+msgid "Remove"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/connections.html:34
+msgid "You currently have no third-party accounts connected to this account."
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/connections.html:37
+msgid "Add a Third-party Account"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/login_cancelled.html:5
+#: aleksis/core/templates/socialaccount/login_cancelled.html:6
+#: aleksis/core/templates/socialaccount/login_cancelled.html:13
+msgid "Login cancelled"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/login_cancelled.html:15
+#, python-format
+msgid ""
+"\n"
+"            You decided to cancel logging in to our site using one of your existing accounts. If this was a mistake, please proceed to <a href=\"%(login_url)s\">sign in</a>.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/signup.html:12
+#, python-format
+msgid ""
+"You are about to use your %(provider_name)s account to login to\n"
+"        %(site_name)s. As a final step, please complete the following form:"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/snippets/provider_list.html:12
+#, python-format
+msgid ""
+"\n"
+"                Login with %(name)s\n"
+"              "
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/snippets/provider_list.html:21
+#, python-format
+msgid ""
+"\n"
+"            Login with %(name)s\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/snippets/provider_list.html:30
+msgid ""
+"\n"
+"          No third-party account providers available.\n"
+"        "
+msgstr ""
+
+#: aleksis/core/templates/templated_email/data_checks.email:4
 msgid "The system detected some new problems with your data."
 msgstr ""
 
-#: templates/templated_email/data_checks.email:8
-#: templates/templated_email/data_checks.email:24
+#: aleksis/core/templates/templated_email/data_checks.email:8
+#: aleksis/core/templates/templated_email/data_checks.email:24
+#: aleksis/core/templates/templated_email/person_changed.email:8
+#: aleksis/core/templates/templated_email/person_changed.email:20
 msgid "Hello,"
 msgstr ""
 
-#: templates/templated_email/data_checks.email:10
+#: aleksis/core/templates/templated_email/data_checks.email:10
 msgid ""
 "\n"
 "  the system detected some new problems with your data.\n"
@@ -1650,7 +2353,7 @@ msgid ""
 " "
 msgstr ""
 
-#: templates/templated_email/data_checks.email:26
+#: aleksis/core/templates/templated_email/data_checks.email:26
 msgid ""
 "\n"
 "   the system detected some new problems with your data.\n"
@@ -1658,37 +2361,37 @@ msgid ""
 "  "
 msgstr ""
 
-#: templates/templated_email/data_checks.email:34
+#: aleksis/core/templates/templated_email/data_checks.email:34
 #, fuzzy
 #| msgid "Description"
 msgid "Problem description"
 msgstr "Description"
 
-#: templates/templated_email/data_checks.email:35
+#: aleksis/core/templates/templated_email/data_checks.email:35
 msgid "Count of objects with new problems"
 msgstr ""
 
-#: templates/templated_email/notification.email:3
+#: aleksis/core/templates/templated_email/notification.email:3
 msgid "New notification for"
 msgstr ""
 
-#: templates/templated_email/notification.email:6
-#: templates/templated_email/notification.email:27
+#: aleksis/core/templates/templated_email/notification.email:6
+#: aleksis/core/templates/templated_email/notification.email:27
 #, python-format
 msgid "Dear %(notification_user)s,"
 msgstr ""
 
-#: templates/templated_email/notification.email:8
-#: templates/templated_email/notification.email:29
+#: aleksis/core/templates/templated_email/notification.email:8
+#: aleksis/core/templates/templated_email/notification.email:29
 msgid "we got a new notification for you:"
 msgstr ""
 
-#: templates/templated_email/notification.email:15
-#: templates/templated_email/notification.email:35
+#: aleksis/core/templates/templated_email/notification.email:15
+#: aleksis/core/templates/templated_email/notification.email:35
 msgid "More information"
 msgstr ""
 
-#: templates/templated_email/notification.email:18
+#: aleksis/core/templates/templated_email/notification.email:18
 #, python-format
 msgid ""
 "\n"
@@ -1696,12 +2399,7 @@ msgid ""
 "    "
 msgstr ""
 
-#: templates/templated_email/notification.email:22
-#: templates/templated_email/notification.email:46
-msgid "Your AlekSIS team"
-msgstr ""
-
-#: templates/templated_email/notification.email:40
+#: aleksis/core/templates/templated_email/notification.email:40
 #, python-format
 msgid ""
 "\n"
@@ -1709,24 +2407,41 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/two_factor/_base_focus.html:6
-#: templates/two_factor/core/otp_required.html:22
-#: templates/two_factor/core/setup.html:5
-#: templates/two_factor/profile/profile.html:87
-msgid "Enable Two-Factor Authentication"
+#: aleksis/core/templates/templated_email/person_changed.email:4
+#, python-format
+msgid "%(person)s changed their data!"
 msgstr ""
 
-#: templates/two_factor/_wizard_actions.html:6
-msgid "Cancel"
+#: aleksis/core/templates/templated_email/person_changed.email:10
+#, python-format
+msgid ""
+"\n"
+"   the person %(person)s recently changed the following fields:\n"
+" "
+msgstr ""
+
+#: aleksis/core/templates/templated_email/person_changed.email:22
+#, python-format
+msgid ""
+"\n"
+"    the person %(person)s recently changed the following fields:\n"
+"  "
+msgstr ""
+
+#: aleksis/core/templates/two_factor/_base_focus.html:6
+#: aleksis/core/templates/two_factor/core/otp_required.html:22
+#: aleksis/core/templates/two_factor/core/setup.html:5
+#: aleksis/core/templates/two_factor/profile/profile.html:87
+msgid "Enable Two-Factor Authentication"
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:5
-#: templates/two_factor/core/backup_tokens.html:9
-#: templates/two_factor/profile/profile.html:46
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:5
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:9
+#: aleksis/core/templates/two_factor/profile/profile.html:46
 msgid "Backup Tokens"
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:14
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:14
 msgid ""
 "\n"
 "        Backup tokens can be used when your primary and backup\n"
@@ -1737,115 +2452,129 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:33
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:33
 msgid ""
 "\n"
 "          Print these tokens and keep them somewhere safe.\n"
 "        "
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:39
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:39
 msgid "You don't have any backup codes yet."
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:45
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:45
 msgid "Back to Account Security"
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:49
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:49
 msgid "Generate Tokens"
 msgstr ""
 
-#: templates/two_factor/core/login.html:16
-msgid "You have no permission to view this page. Please login with an other account."
+#: aleksis/core/templates/two_factor/core/login.html:20
+msgid "Login with username and password"
 msgstr ""
 
-#: templates/two_factor/core/login.html:24
-msgid "Please login to see this page."
+#: aleksis/core/templates/two_factor/core/login.html:28
+msgid ""
+"You have no permission to view this page. Please login with an other\n"
+"                    account."
 msgstr ""
 
-#: templates/two_factor/core/login.html:30
-msgid "Login with username and password"
+#: aleksis/core/templates/two_factor/core/login.html:36
+msgid "Please login to see this page."
 msgstr ""
 
-#: templates/two_factor/core/login.html:40
+#: aleksis/core/templates/two_factor/core/login.html:46
 msgid ""
-"We are calling your phone right now, please enter the\n"
-"              digits you hear."
+"\n"
+"                        We are calling your phone right now, please enter the\n"
+"                        digits you hear.\n"
+"                      "
 msgstr ""
 
-#: templates/two_factor/core/login.html:43
+#: aleksis/core/templates/two_factor/core/login.html:51
 msgid ""
-"We sent you a text message, please enter the tokens we\n"
-"              sent."
+"\n"
+"                        We sent you a text message, please enter the tokens we\n"
+"                        sent.\n"
+"                      "
 msgstr ""
 
-#: templates/two_factor/core/login.html:46
+#: aleksis/core/templates/two_factor/core/login.html:56
 msgid ""
-"Please enter the tokens generated by your token\n"
-"              generator."
+"\n"
+"                        Please enter the tokens generated by your token\n"
+"                        generator.\n"
+"                      "
 msgstr ""
 
-#: templates/two_factor/core/login.html:50
+#: aleksis/core/templates/two_factor/core/login.html:62
 msgid ""
-"Use this form for entering backup tokens for logging in.\n"
-"            These tokens have been generated for you to print and keep safe. Please\n"
-"            enter one of these backup tokens to login to your account."
+"\n"
+"                      Use this form for entering backup tokens for logging in.\n"
+"                      These tokens have been generated for you to print and keep safe. Please\n"
+"                      enter one of these backup tokens to login to your account.\n"
+"                    "
+msgstr ""
+
+#: aleksis/core/templates/two_factor/core/login.html:90
+msgid "Device currently not available?"
 msgstr ""
 
-#: templates/two_factor/core/login.html:68
+#: aleksis/core/templates/two_factor/core/login.html:92
 msgid "Or, alternatively, use one of your backup phones:"
 msgstr ""
 
-#: templates/two_factor/core/login.html:78
+#: aleksis/core/templates/two_factor/core/login.html:102
 msgid "As a last resort, you can use a backup token:"
 msgstr ""
 
-#: templates/two_factor/core/login.html:81
+#: aleksis/core/templates/two_factor/core/login.html:105
 msgid "Use Backup Token"
 msgstr ""
 
-#: templates/two_factor/core/login.html:93
+#: aleksis/core/templates/two_factor/core/login.html:116
 msgid "Use alternative login options"
 msgstr ""
 
-#: templates/two_factor/core/otp_required.html:9
+#: aleksis/core/templates/two_factor/core/otp_required.html:9
 msgid "Permission Denied"
 msgstr ""
 
-#: templates/two_factor/core/otp_required.html:10
+#: aleksis/core/templates/two_factor/core/otp_required.html:10
 msgid ""
 "The page you requested, enforces users to verify using\n"
 "          two-factor authentication for security reasons. You need to enable these\n"
 "          security features in order to access this page."
 msgstr ""
 
-#: templates/two_factor/core/otp_required.html:14
+#: aleksis/core/templates/two_factor/core/otp_required.html:14
 msgid ""
 "Two-factor authentication is not enabled for your\n"
 "          account. Enable two-factor authentication for enhanced account\n"
 "          security."
 msgstr ""
 
-#: templates/two_factor/core/phone_register.html:5
-#: templates/two_factor/core/phone_register.html:9
+#: aleksis/core/templates/two_factor/core/phone_register.html:5
+#: aleksis/core/templates/two_factor/core/phone_register.html:9
 msgid "Add Backup Phone"
 msgstr ""
 
-#: templates/two_factor/core/phone_register.html:12
+#: aleksis/core/templates/two_factor/core/phone_register.html:12
 msgid ""
 "You'll be adding a backup phone number to your\n"
 "      account. This number will be used if your primary method of\n"
 "      registration is not available."
 msgstr ""
 
-#: templates/two_factor/core/phone_register.html:16
+#: aleksis/core/templates/two_factor/core/phone_register.html:16
 msgid ""
 "We've sent a token to your phone number. Please\n"
 "      enter the token you've received."
 msgstr ""
 
-#: templates/two_factor/core/setup.html:9
+#: aleksis/core/templates/two_factor/core/setup.html:9
 msgid ""
 "\n"
 "        You are about to take your account security to the\n"
@@ -1854,14 +2583,14 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:17
+#: aleksis/core/templates/two_factor/core/setup.html:17
 msgid ""
 "\n"
 "        Please select which authentication method you would like to use:\n"
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:23
+#: aleksis/core/templates/two_factor/core/setup.html:23
 msgid ""
 "\n"
 "        To start using a token generator, please use your\n"
@@ -1870,7 +2599,7 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:34
+#: aleksis/core/templates/two_factor/core/setup.html:34
 msgid ""
 "\n"
 "        Please enter the phone number you wish to receive the\n"
@@ -1878,7 +2607,7 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:41
+#: aleksis/core/templates/two_factor/core/setup.html:41
 msgid ""
 "\n"
 "        Please enter the phone number you wish to be called on.\n"
@@ -1886,21 +2615,21 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:50
+#: aleksis/core/templates/two_factor/core/setup.html:50
 msgid ""
 "\n"
 "            We are calling your phone right now, please enter the digits you hear.\n"
 "          "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:56
+#: aleksis/core/templates/two_factor/core/setup.html:56
 msgid ""
 "\n"
 "            We sent you a text message, please enter the tokens we sent.\n"
 "          "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:63
+#: aleksis/core/templates/two_factor/core/setup.html:63
 msgid ""
 "\n"
 "          We've encountered an issue with the selected authentication method. Please\n"
@@ -1910,7 +2639,7 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:73
+#: aleksis/core/templates/two_factor/core/setup.html:73
 msgid ""
 "\n"
 "        To identify and verify your YubiKey, please insert a\n"
@@ -1919,29 +2648,29 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:5
-#: templates/two_factor/core/setup_complete.html:9
+#: aleksis/core/templates/two_factor/core/setup_complete.html:5
+#: aleksis/core/templates/two_factor/core/setup_complete.html:9
 msgid "Two-Factor Authentication successfully enabled"
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:14
+#: aleksis/core/templates/two_factor/core/setup_complete.html:14
 msgid ""
 "\n"
 "        Congratulations, you've successfully enabled two-factor authentication.\n"
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:24
-#: templates/two_factor/core/setup_complete.html:44
+#: aleksis/core/templates/two_factor/core/setup_complete.html:24
+#: aleksis/core/templates/two_factor/core/setup_complete.html:44
 msgid "Back to Profile"
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:28
-#: templates/two_factor/core/setup_complete.html:48
+#: aleksis/core/templates/two_factor/core/setup_complete.html:28
+#: aleksis/core/templates/two_factor/core/setup_complete.html:48
 msgid "Generate backup codes"
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:34
+#: aleksis/core/templates/two_factor/core/setup_complete.html:34
 msgid ""
 "\n"
 "          However, it might happen that you don't have access to\n"
@@ -1950,65 +2679,65 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:52
-#: templates/two_factor/profile/profile.html:41
+#: aleksis/core/templates/two_factor/core/setup_complete.html:52
+#: aleksis/core/templates/two_factor/profile/profile.html:41
 msgid "Add Phone Number"
 msgstr ""
 
-#: templates/two_factor/profile/disable.html:5
-#: templates/two_factor/profile/disable.html:9
-#: templates/two_factor/profile/profile.html:63
-#: templates/two_factor/profile/profile.html:73
+#: aleksis/core/templates/two_factor/profile/disable.html:5
+#: aleksis/core/templates/two_factor/profile/disable.html:9
+#: aleksis/core/templates/two_factor/profile/profile.html:63
+#: aleksis/core/templates/two_factor/profile/profile.html:73
 msgid "Disable Two-Factor Authentication"
 msgstr ""
 
-#: templates/two_factor/profile/disable.html:12
+#: aleksis/core/templates/two_factor/profile/disable.html:12
 msgid "You are about to disable two-factor authentication. This weakens your account security, are you sure?"
 msgstr ""
 
-#: templates/two_factor/profile/disable.html:26
+#: aleksis/core/templates/two_factor/profile/disable.html:26
 msgid "Disable"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:5
-#: templates/two_factor/profile/profile.html:10
+#: aleksis/core/templates/two_factor/profile/profile.html:5
+#: aleksis/core/templates/two_factor/profile/profile.html:10
 msgid "Account Security"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:15
+#: aleksis/core/templates/two_factor/profile/profile.html:15
 msgid "Tokens will be generated by your token generator."
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:17
+#: aleksis/core/templates/two_factor/profile/profile.html:17
 #, python-format
 msgid "Primary method: %(primary)s"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:19
+#: aleksis/core/templates/two_factor/profile/profile.html:19
 msgid "Tokens will be generated by your YubiKey."
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:23
+#: aleksis/core/templates/two_factor/profile/profile.html:23
 msgid "Backup Phone Numbers"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:24
+#: aleksis/core/templates/two_factor/profile/profile.html:24
 msgid ""
 "If your primary method is not available, we are able to\n"
 "        send backup tokens to the phone numbers listed below."
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:33
+#: aleksis/core/templates/two_factor/profile/profile.html:33
 msgid "Unregister"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:48
+#: aleksis/core/templates/two_factor/profile/profile.html:48
 msgid ""
 "If you don't have any device with you, you can access\n"
 "        your account using backup tokens."
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:50
+#: aleksis/core/templates/two_factor/profile/profile.html:50
 #, python-format
 msgid ""
 "\n"
@@ -2021,11 +2750,11 @@ msgid_plural ""
 msgstr[0] ""
 msgstr[1] ""
 
-#: templates/two_factor/profile/profile.html:59
+#: aleksis/core/templates/two_factor/profile/profile.html:59
 msgid "Show Codes"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:65
+#: aleksis/core/templates/two_factor/profile/profile.html:65
 msgid ""
 "\n"
 "        However we strongly discourage you to do so, you can\n"
@@ -2033,7 +2762,7 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:78
+#: aleksis/core/templates/two_factor/profile/profile.html:78
 msgid ""
 "\n"
 "        Two-factor authentication is not enabled for your\n"
@@ -2042,99 +2771,140 @@ msgid ""
 "      "
 msgstr ""
 
-#: util/notifications.py:65
+#: aleksis/core/util/notifications.py:65
 msgid "E-Mail"
 msgstr ""
 
-#: util/notifications.py:66
+#: aleksis/core/util/notifications.py:66
 msgid "SMS"
 msgstr ""
 
-#: views.py:141
+#: aleksis/core/util/pdf.py:105
+msgid "Progress: Generate PDF file"
+msgstr ""
+
+#: aleksis/core/util/pdf.py:106
+msgid "Generating PDF file …"
+msgstr ""
+
+#: aleksis/core/util/pdf.py:107
+msgid "The PDF file has been generated successfully."
+msgstr ""
+
+#: aleksis/core/util/pdf.py:108
+msgid "There was a problem while generating the PDF file."
+msgstr ""
+
+#: aleksis/core/util/pdf.py:111
+msgid "Download PDF"
+msgstr ""
+
+#: aleksis/core/views.py:251
 msgid "The school term has been created."
 msgstr ""
 
-#: views.py:153
+#: aleksis/core/views.py:263
 msgid "The school term has been saved."
 msgstr ""
 
-#: views.py:298
+#: aleksis/core/views.py:387
 msgid "The child groups were successfully saved."
 msgstr ""
 
-#: views.py:336
+#: aleksis/core/views.py:406 aleksis/core/views.py:416
 msgid "The person has been saved."
 msgstr ""
 
-#: views.py:375
+#: aleksis/core/views.py:466
 msgid "The group has been saved."
 msgstr ""
 
-#: views.py:467
+#: aleksis/core/views.py:563
 msgid "The announcement has been saved."
 msgstr ""
 
-#: views.py:483
+#: aleksis/core/views.py:579
 msgid "The announcement has been deleted."
 msgstr ""
 
-#: views.py:562
+#: aleksis/core/views.py:663
 msgid "The preferences have been saved successfully."
 msgstr ""
 
-#: views.py:586
+#: aleksis/core/views.py:687
 msgid "The person has been deleted."
 msgstr ""
 
-#: views.py:600
+#: aleksis/core/views.py:701
 msgid "The group has been deleted."
 msgstr ""
 
-#: views.py:632
+#: aleksis/core/views.py:733
 msgid "The additional_field has been saved."
 msgstr ""
 
-#: views.py:666
+#: aleksis/core/views.py:767
 msgid "The additional field has been deleted."
 msgstr ""
 
-#: views.py:691
+#: aleksis/core/views.py:792
 msgid "The group type has been saved."
 msgstr ""
 
-#: views.py:721
+#: aleksis/core/views.py:822
 msgid "The group type has been deleted."
 msgstr ""
 
-#: views.py:749
-msgid "The data check has been started. Please note that it may take a while before you are able to fetch the data on this page."
+#: aleksis/core/views.py:855
+msgid "Progress: Run data checks"
+msgstr ""
+
+#: aleksis/core/views.py:856
+msgid "Run data checks …"
+msgstr ""
+
+#: aleksis/core/views.py:857
+msgid "The data checks were run successfully."
 msgstr ""
 
-#: views.py:754
-msgid "The data check has finished."
+#: aleksis/core/views.py:858
+msgid "There was a problem while running data checks."
 msgstr ""
 
-#: views.py:769
+#: aleksis/core/views.py:874
 #, python-brace-format
 msgid "The solve option '{solve_option_obj.verbose_name}' "
 msgstr ""
 
-#: views.py:811
+#: aleksis/core/views.py:916
 msgid "The dashboard widget has been saved."
 msgstr ""
 
-#: views.py:841
+#: aleksis/core/views.py:946
 msgid "The dashboard widget has been created."
 msgstr ""
 
-#: views.py:851
+#: aleksis/core/views.py:956
 msgid "The dashboard widget has been deleted."
 msgstr ""
 
-#: views.py:914
+#: aleksis/core/views.py:1023
 msgid "Your dashboard configuration has been saved successfully."
 msgstr ""
 
-#: views.py:916
+#: aleksis/core/views.py:1025
 msgid "The configuration of the default dashboard has been saved successfully."
 msgstr ""
+
+#: aleksis/core/views.py:1153
+msgid "The third-party account could not be disconnected because it is the only login method available."
+msgstr ""
+
+#: aleksis/core/views.py:1160
+msgid "The third-party account has been successfully disconnected."
+msgstr ""
+
+#, fuzzy
+#~| msgid "Contact details"
+#~ msgid "Can link persons to accounts"
+#~ msgstr "Détails de contact"
diff --git a/aleksis/core/locale/fr/LC_MESSAGES/djangojs.po b/aleksis/core/locale/fr/LC_MESSAGES/djangojs.po
index 83a1c140c8889eef6a1633f00a741cd110c52e05..bbd51395a6ee28d076da3de7bd4d65b198574194 100644
--- a/aleksis/core/locale/fr/LC_MESSAGES/djangojs.po
+++ b/aleksis/core/locale/fr/LC_MESSAGES/djangojs.po
@@ -8,7 +8,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2021-01-11 21:30+0100\n"
+"POT-Creation-Date: 2021-10-28 16:18+0200\n"
 "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
@@ -18,18 +18,18 @@ msgstr ""
 "Content-Transfer-Encoding: 8bit\n"
 "Plural-Forms: nplurals=2; plural=(n > 1);\n"
 
-#: static/js/main.js:15
+#: aleksis/core/static/js/main.js:15
 msgid "Today"
 msgstr ""
 
-#: static/js/main.js:16
+#: aleksis/core/static/js/main.js:16
 msgid "Cancel"
 msgstr ""
 
-#: static/js/main.js:17
+#: aleksis/core/static/js/main.js:17
 msgid "OK"
 msgstr ""
 
-#: static/js/main.js:118
+#: aleksis/core/static/js/main.js:127
 msgid "This page may contain outdated information since there is no internet connection."
 msgstr ""
diff --git a/aleksis/core/locale/la/LC_MESSAGES/django.po b/aleksis/core/locale/la/LC_MESSAGES/django.po
index 5fb260aac6828ea32031b9b03bd64424984cd8ab..c34a5cc0811a09a7a1b1bf4430463e56cff5be3f 100644
--- a/aleksis/core/locale/la/LC_MESSAGES/django.po
+++ b/aleksis/core/locale/la/LC_MESSAGES/django.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2021-01-11 21:30+0100\n"
+"POT-Creation-Date: 2021-10-28 16:18+0200\n"
 "PO-Revision-Date: 2020-12-19 12:57+0000\n"
 "Last-Translator: Julian <leuckerj@gmail.com>\n"
 "Language-Team: Latin <https://translate.edugit.org/projects/aleksis/aleksis/la/>\n"
@@ -18,916 +18,1229 @@ msgstr ""
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
 "X-Generator: Weblate 4.3.2\n"
 
-#: data_checks.py:53
+#: aleksis/core/apps.py:150
+msgid "OpenID Connect scope"
+msgstr ""
+
+#: aleksis/core/apps.py:151
+msgid "Given name, family name, link to profile and picture if existing."
+msgstr ""
+
+#: aleksis/core/apps.py:152
+#, fuzzy
+#| msgid "E-mail address"
+msgid "Full home postal address"
+msgstr "Inscriptio electronica"
+
+#: aleksis/core/apps.py:153
+#, fuzzy
+#| msgid "E-mail address"
+msgid "Email address"
+msgstr "Inscriptio electronica"
+
+#: aleksis/core/apps.py:154
+#, fuzzy
+#| msgid "Mobile phone"
+msgid "Home and mobile phone"
+msgstr "Numerus telephoni mobilis"
+
+#: aleksis/core/data_checks.py:55
 msgid "Ignore problem"
 msgstr ""
 
-#: data_checks.py:174
+#: aleksis/core/data_checks.py:184
 #, python-brace-format
 msgid "Solve option '{solve_option_obj.verbose_name}' "
 msgstr ""
 
-#: filters.py:37 templates/core/base.html:77 templates/core/group/list.html:20
-#: templates/core/person/list.html:24 templates/search/search.html:7
-#: templates/search/search.html:22
+#: aleksis/core/data_checks.py:291
+#, fuzzy
+#| msgid "Dashboard"
+msgid "Deactivate DashboardWidget"
+msgstr "Forum"
+
+#: aleksis/core/data_checks.py:303
+msgid "Ensure that there are no broken DashboardWidgets."
+msgstr ""
+
+#: aleksis/core/data_checks.py:304
+msgid "The DashboardWidget was reported broken automatically."
+msgstr ""
+
+#: aleksis/core/filters.py:37 aleksis/core/templates/core/base.html:83
+#: aleksis/core/templates/core/base.html:84
+#: aleksis/core/templates/core/group/list.html:20
+#: aleksis/core/templates/core/person/list.html:24
+#: aleksis/core/templates/search/search.html:7
+#: aleksis/core/templates/search/search.html:22
 msgid "Search"
 msgstr "Quaerere"
 
-#: filters.py:53
+#: aleksis/core/filters.py:53
 msgid "Search by name"
 msgstr "Quaerere cum breve nomine"
 
-#: filters.py:65
+#: aleksis/core/filters.py:65
 #, fuzzy
 #| msgid "E-mail address"
 msgid "Search by contact details"
 msgstr "Inscriptio electronica"
 
-#: forms.py:54
-msgid "You cannot set a new username when also selecting an existing user."
-msgstr ""
-
-#: forms.py:58
-msgid "This username is already in use."
-msgstr ""
-
-#: forms.py:82
+#: aleksis/core/forms.py:41 aleksis/core/forms.py:387
 msgid "Base data"
 msgstr ""
 
-#: forms.py:88
+#: aleksis/core/forms.py:47
 #, fuzzy
 #| msgid "E-mail address"
 msgid "Address"
 msgstr "Inscriptio electronica"
 
-#: forms.py:89
+#: aleksis/core/forms.py:48
 msgid "Contact data"
 msgstr ""
 
-#: forms.py:91
+#: aleksis/core/forms.py:50
 msgid "Advanced personal data"
 msgstr ""
 
-#: forms.py:134
+#: aleksis/core/forms.py:93
 msgid "New user"
 msgstr ""
 
-#: forms.py:134
+#: aleksis/core/forms.py:93
 #, fuzzy
 #| msgid "Persons and accounts"
 msgid "Create a new account"
 msgstr "Personae et computi"
 
-#: forms.py:146 models.py:102
+#: aleksis/core/forms.py:124
+msgid "You cannot set a new username when also selecting an existing user."
+msgstr ""
+
+#: aleksis/core/forms.py:128
+msgid "This username is already in use."
+msgstr ""
+
+#: aleksis/core/forms.py:145 aleksis/core/models.py:117
 msgid "School term"
 msgstr "Anus scolae"
 
-#: forms.py:147
+#: aleksis/core/forms.py:146
 #, fuzzy
 #| msgid "Data management"
 msgid "Common data"
 msgstr "Adminstratio datarum"
 
-#: forms.py:148 forms.py:197 menus.py:169 models.py:116
-#: templates/core/person/list.html:8 templates/core/person/list.html:9
+#: aleksis/core/forms.py:147 aleksis/core/forms.py:196
+#: aleksis/core/menus.py:238 aleksis/core/models.py:140
+#: aleksis/core/templates/core/person/list.html:8
+#: aleksis/core/templates/core/person/list.html:9
 msgid "Persons"
 msgstr "personae"
 
-#: forms.py:149
+#: aleksis/core/forms.py:148
 #, fuzzy
 #| msgid "Additional name(s)"
 msgid "Additional data"
 msgstr "addita nomines"
 
-#: forms.py:189 forms.py:192 models.py:45
+#: aleksis/core/forms.py:188 aleksis/core/forms.py:191
+#: aleksis/core/models.py:60
 msgid "Date"
 msgstr "dies"
 
-#: forms.py:190 forms.py:193 models.py:53
+#: aleksis/core/forms.py:189 aleksis/core/forms.py:192
+#: aleksis/core/models.py:68
 msgid "Time"
 msgstr "tempus"
 
-#: forms.py:210 menus.py:177 models.py:338 templates/core/group/list.html:8
-#: templates/core/group/list.html:9 templates/core/person/full.html:144
+#: aleksis/core/forms.py:209 aleksis/core/menus.py:249
+#: aleksis/core/models.py:398 aleksis/core/templates/core/group/list.html:8
+#: aleksis/core/templates/core/group/list.html:9
+#: aleksis/core/templates/core/person/full.html:144
 msgid "Groups"
 msgstr "Greges"
 
-#: forms.py:220
+#: aleksis/core/forms.py:219
 msgid "From when until when should the announcement be displayed?"
 msgstr ""
 
-#: forms.py:223
+#: aleksis/core/forms.py:222
 msgid "Who should see the announcement?"
 msgstr "Quis nuntium videatne?"
 
-#: forms.py:224
+#: aleksis/core/forms.py:223
 msgid "Write your announcement:"
 msgstr "Scribe nuntium:"
 
-#: forms.py:263
+#: aleksis/core/forms.py:262
 msgid "You are not allowed to create announcements which are only valid in the past."
 msgstr ""
 
-#: forms.py:267
+#: aleksis/core/forms.py:266
 msgid "The from date and time must be earlier then the until date and time."
 msgstr ""
 
-#: forms.py:276
+#: aleksis/core/forms.py:275
 msgid "You need at least one recipient."
 msgstr ""
 
-#: health_checks.py:15
+#: aleksis/core/forms.py:389
+#, fuzzy
+#| msgid "Data management"
+msgid "Account data"
+msgstr "Adminstratio datarum"
+
+#: aleksis/core/forms.py:391
+msgid "Consents"
+msgstr ""
+
+#: aleksis/core/forms.py:396
+msgid "Password"
+msgstr ""
+
+#: aleksis/core/forms.py:402
+msgid "Password (again)"
+msgstr ""
+
+#: aleksis/core/forms.py:411
+#, python-brace-format
+msgid "I have read the <a href='{privacy_policy}'>Privacy policy</a> and agree with them."
+msgstr ""
+
+#: aleksis/core/forms.py:435
+msgid "You must type the same password each time."
+msgstr ""
+
+#: aleksis/core/forms.py:580
+msgid "No valid selection."
+msgstr ""
+
+#: aleksis/core/health_checks.py:21
 #, fuzzy
 #| msgid "Write your announcement:"
 msgid "There are unresolved data problems."
 msgstr "Scribe nuntium:"
 
-#: menus.py:7 templates/two_factor/core/login.html:6
-#: templates/two_factor/core/login.html:10
-#: templates/two_factor/core/login.html:86
+#: aleksis/core/health_checks.py:38
+msgid "The backup folder doesn't exist."
+msgstr ""
+
+#: aleksis/core/health_checks.py:47
+#, python-brace-format
+msgid "Last backup {time_gone_since_backup}!"
+msgstr ""
+
+#: aleksis/core/health_checks.py:49
+msgid "No backup found!"
+msgstr ""
+
+#: aleksis/core/health_checks.py:76
+msgid "No backup result found!"
+msgstr ""
+
+#: aleksis/core/health_checks.py:78
+#, python-brace-format
+msgid "{task.status} - {task.result}"
+msgstr ""
+
+#: aleksis/core/menus.py:9 aleksis/core/templates/two_factor/core/login.html:6
+#: aleksis/core/templates/two_factor/core/login.html:22
+#: aleksis/core/templates/two_factor/core/login.html:76
 msgid "Login"
 msgstr "nomen profiteri"
 
-#: menus.py:13
+#: aleksis/core/menus.py:15 aleksis/core/templates/account/signup.html:20
+#: aleksis/core/templates/socialaccount/signup.html:23
+msgid "Sign up"
+msgstr ""
+
+#: aleksis/core/menus.py:24
 msgid "Dashboard"
 msgstr "Forum"
 
-#: menus.py:19
+#: aleksis/core/menus.py:32 aleksis/core/models.py:605
+#: aleksis/core/preferences.py:26
+#: aleksis/core/templates/core/notifications.html:4
+#: aleksis/core/templates/core/notifications.html:5
+msgid "Notifications"
+msgstr "Nuntii"
+
+#: aleksis/core/menus.py:41
 msgid "Account"
 msgstr ""
 
-#: menus.py:26
+#: aleksis/core/menus.py:48
 msgid "Stop impersonation"
 msgstr "Simulandum aliquem finire"
 
-#: menus.py:35 templates/core/base.html:56
+#: aleksis/core/menus.py:57 aleksis/core/templates/core/base.html:62
 msgid "Logout"
 msgstr "nomen retractare"
 
-#: menus.py:41
+#: aleksis/core/menus.py:63
 msgid "2FA"
 msgstr ""
 
-#: menus.py:47
+#: aleksis/core/menus.py:69
+#: aleksis/core/templates/account/password_change.html:5
+#: aleksis/core/templates/account/password_change.html:6
+#: aleksis/core/templates/account/password_change.html:19
+#: aleksis/core/templates/account/password_reset_from_key.html:5
+#: aleksis/core/templates/account/password_reset_from_key.html:42
+#: aleksis/core/templates/account/password_reset_from_key.html:46
+#: aleksis/core/templates/account/password_reset_from_key_done.html:5
+#: aleksis/core/templates/account/password_reset_from_key_done.html:6
+msgid "Change password"
+msgstr ""
+
+#: aleksis/core/menus.py:81
 msgid "Me"
 msgstr ""
 
-#: menus.py:56 templates/dynamic_preferences/form.html:5
+#: aleksis/core/menus.py:90
+#: aleksis/core/templates/dynamic_preferences/form.html:5
 msgid "Preferences"
 msgstr ""
 
-#: menus.py:67
+#: aleksis/core/menus.py:99
+#, fuzzy
+#| msgid "Persons and accounts"
+msgid "Third-party accounts"
+msgstr "Personae et computi"
+
+#: aleksis/core/menus.py:108
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:5
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:6
+#, fuzzy
+#| msgid "Notifications"
+msgid "Authorized applications"
+msgstr "Nuntii"
+
+#: aleksis/core/menus.py:119
 msgid "Admin"
 msgstr "Administratio"
 
-#: menus.py:75 models.py:602 templates/core/announcement/list.html:7
-#: templates/core/announcement/list.html:8
+#: aleksis/core/menus.py:127 aleksis/core/models.py:704
+#: aleksis/core/templates/core/announcement/list.html:7
+#: aleksis/core/templates/core/announcement/list.html:8
 msgid "Announcements"
 msgstr "Nuntii"
 
-#: menus.py:86 models.py:103 templates/core/school_term/list.html:8
-#: templates/core/school_term/list.html:9
+#: aleksis/core/menus.py:138 aleksis/core/models.py:118
+#: aleksis/core/templates/core/school_term/list.html:8
+#: aleksis/core/templates/core/school_term/list.html:9
 msgid "School terms"
 msgstr "ani scolae"
 
-#: menus.py:97 templates/core/dashboard_widget/list.html:8
-#: templates/core/dashboard_widget/list.html:9
+#: aleksis/core/menus.py:149
+#: aleksis/core/templates/core/dashboard_widget/list.html:8
+#: aleksis/core/templates/core/dashboard_widget/list.html:9
 #, fuzzy
 #| msgid "Dashboard"
 msgid "Dashboard widgets"
 msgstr "Forum"
 
-#: menus.py:108 templates/core/management/data_management.html:6
-#: templates/core/management/data_management.html:7
+#: aleksis/core/menus.py:160
+#: aleksis/core/templates/core/management/data_management.html:6
+#: aleksis/core/templates/core/management/data_management.html:7
 msgid "Data management"
 msgstr "Adminstratio datarum"
 
-#: menus.py:116 templates/core/pages/system_status.html:5
-#: templates/core/pages/system_status.html:7
+#: aleksis/core/menus.py:171
+#: aleksis/core/templates/core/pages/system_status.html:5
+#: aleksis/core/templates/core/pages/system_status.html:7
 msgid "System status"
 msgstr "Status systemae"
 
-#: menus.py:127
+#: aleksis/core/menus.py:182
 msgid "Impersonation"
 msgstr "Simulare aliquem"
 
-#: menus.py:135
+#: aleksis/core/menus.py:193
 #, fuzzy
 #| msgid "Notification"
 msgid "Configuration"
 msgstr "Nuntius"
 
-#: menus.py:146 templates/core/data_check/list.html:9
-#: templates/core/data_check/list.html:10
+#: aleksis/core/menus.py:204 aleksis/core/templates/core/data_check/list.html:9
+#: aleksis/core/templates/core/data_check/list.html:10
 #, fuzzy
 #| msgid "System status"
 msgid "Data checks"
 msgstr "Status systemae"
 
-#: menus.py:152
+#: aleksis/core/menus.py:210
 msgid "Backend Admin"
 msgstr ""
 
-#: menus.py:160
+#: aleksis/core/menus.py:216
+#: aleksis/core/templates/oauth2_provider/application_detail.html:5
+#: aleksis/core/templates/oauth2_provider/application_list.html:5
+#, fuzzy
+#| msgid "Notifications"
+msgid "OAuth2 Applications"
+msgstr "Nuntii"
+
+#: aleksis/core/menus.py:229
 msgid "People"
 msgstr "Personae"
 
-#: menus.py:185 models.py:812 templates/core/group_type/list.html:8
-#: templates/core/group_type/list.html:9
+#: aleksis/core/menus.py:260 aleksis/core/models.py:958
+#: aleksis/core/templates/core/group_type/list.html:8
+#: aleksis/core/templates/core/group_type/list.html:9
 #, fuzzy
 #| msgid "Groups"
 msgid "Group types"
 msgstr "Greges"
 
-#: menus.py:196
-msgid "Persons and accounts"
-msgstr "Personae et computi"
-
-#: menus.py:207
+#: aleksis/core/menus.py:271
 msgid "Groups and child groups"
 msgstr ""
 
-#: menus.py:218 models.py:385 templates/core/additional_field/list.html:8
-#: templates/core/additional_field/list.html:9
+#: aleksis/core/menus.py:282 aleksis/core/models.py:446
+#: aleksis/core/templates/core/additional_field/list.html:8
+#: aleksis/core/templates/core/additional_field/list.html:9
 #, fuzzy
 #| msgid "Additional name(s)"
 msgid "Additional fields"
 msgstr "addita nomines"
 
-#: menus.py:233 templates/core/group/child_groups.html:7
-#: templates/core/group/child_groups.html:9
+#: aleksis/core/menus.py:297
+#: aleksis/core/templates/core/group/child_groups.html:7
+#: aleksis/core/templates/core/group/child_groups.html:9
 msgid "Assign child groups to groups"
 msgstr ""
 
-#: mixins.py:384
+#: aleksis/core/mixins.py:498
 #, fuzzy
 #| msgid "Edit school term"
 msgid "Linked school term"
 msgstr "Muta anum scolae"
 
-#: models.py:43
+#: aleksis/core/models.py:58
 msgid "Boolean (Yes/No)"
 msgstr ""
 
-#: models.py:44
+#: aleksis/core/models.py:59
 msgid "Text (one line)"
 msgstr ""
 
-#: models.py:46
+#: aleksis/core/models.py:61
 msgid "Date and time"
 msgstr "Dies et hora"
 
-#: models.py:47
+#: aleksis/core/models.py:62
 msgid "Decimal number"
 msgstr ""
 
-#: models.py:48 models.py:157
+#: aleksis/core/models.py:63 aleksis/core/models.py:186
 msgid "E-mail address"
 msgstr "Inscriptio electronica"
 
-#: models.py:49
+#: aleksis/core/models.py:64
 msgid "Integer"
 msgstr ""
 
-#: models.py:50
+#: aleksis/core/models.py:65
 #, fuzzy
 #| msgid "E-mail address"
 msgid "IP address"
 msgstr "Inscriptio electronica"
 
-#: models.py:51
+#: aleksis/core/models.py:66
 msgid "Boolean or empty (Yes/No/Neither)"
 msgstr ""
 
-#: models.py:52
+#: aleksis/core/models.py:67
 msgid "Text (multi-line)"
 msgstr ""
 
-#: models.py:54
+#: aleksis/core/models.py:69
 msgid "URL / Link"
 msgstr ""
 
-#: models.py:66 models.py:785
+#: aleksis/core/models.py:81 aleksis/core/models.py:927
 msgid "Name"
 msgstr "Nomen"
 
-#: models.py:68
+#: aleksis/core/models.py:83
 msgid "Start date"
 msgstr ""
 
-#: models.py:69
+#: aleksis/core/models.py:84
 msgid "End date"
 msgstr ""
 
-#: models.py:88
+#: aleksis/core/models.py:103
 msgid "The start date must be earlier than the end date."
 msgstr ""
 
-#: models.py:95
+#: aleksis/core/models.py:110
 msgid "There is already a school term for this time or a part of this time."
 msgstr ""
 
-#: models.py:115 models.py:744 templates/core/person/accounts.html:41
+#: aleksis/core/models.py:139 aleksis/core/models.py:876
 msgid "Person"
 msgstr "Persona"
 
-#: models.py:118
+#: aleksis/core/models.py:142
 #, fuzzy
 #| msgid "E-mail address"
 msgid "Can view address"
 msgstr "Inscriptio electronica"
 
-#: models.py:119
+#: aleksis/core/models.py:143
 #, fuzzy
 #| msgid "E-mail address"
 msgid "Can view contact details"
 msgstr "Inscriptio electronica"
 
-#: models.py:120
+#: aleksis/core/models.py:144
 #, fuzzy
 #| msgid "E-mail address"
 msgid "Can view photo"
 msgstr "Inscriptio electronica"
 
-#: models.py:121
+#: aleksis/core/models.py:145
 #, fuzzy
 #| msgid "Persons and accounts"
 msgid "Can view persons groups"
 msgstr "Personae et computi"
 
-#: models.py:122
+#: aleksis/core/models.py:146
 #, fuzzy
 #| msgid "Stop impersonation"
 msgid "Can view personal details"
 msgstr "Simulandum aliquem finire"
 
-#: models.py:127
+#: aleksis/core/models.py:156
 msgid "female"
 msgstr "femininum"
 
-#: models.py:127
+#: aleksis/core/models.py:156
 msgid "male"
 msgstr "maskulinum"
 
-#: models.py:135
+#: aleksis/core/models.py:164
 msgid "Linked user"
 msgstr ""
 
-#: models.py:137
+#: aleksis/core/models.py:166
 #, fuzzy
 #| msgid "Impersonation"
 msgid "Is person active?"
 msgstr "Simulare aliquem"
 
-#: models.py:139
+#: aleksis/core/models.py:168
 msgid "First name"
 msgstr "Primus nomen"
 
-#: models.py:140
+#: aleksis/core/models.py:169
 msgid "Last name"
 msgstr "Secondus nomen"
 
-#: models.py:142
+#: aleksis/core/models.py:171
 msgid "Additional name(s)"
 msgstr "addita nomines"
 
-#: models.py:146 models.py:354
+#: aleksis/core/models.py:175 aleksis/core/models.py:415
 msgid "Short name"
 msgstr "Breve nomen"
 
-#: models.py:149
+#: aleksis/core/models.py:178
 msgid "Street"
 msgstr "Via"
 
-#: models.py:150
+#: aleksis/core/models.py:179
 msgid "Street number"
 msgstr "Numerus domini"
 
-#: models.py:151
+#: aleksis/core/models.py:180
 msgid "Postal code"
 msgstr "Numerus directorius"
 
-#: models.py:152
+#: aleksis/core/models.py:181
 msgid "Place"
 msgstr "Urbs"
 
-#: models.py:154
+#: aleksis/core/models.py:183
 msgid "Home phone"
 msgstr "Numerus telephoni domi"
 
-#: models.py:155
+#: aleksis/core/models.py:184
 msgid "Mobile phone"
 msgstr "Numerus telephoni mobilis"
 
-#: models.py:159
+#: aleksis/core/models.py:188
 msgid "Date of birth"
 msgstr "Dies natalis"
 
-#: models.py:160
+#: aleksis/core/models.py:189
 msgid "Sex"
 msgstr "Genus"
 
-#: models.py:162
+#: aleksis/core/models.py:191
 msgid "Photo"
 msgstr "Photographia"
 
-#: models.py:166 templates/core/person/full.html:137
+#: aleksis/core/models.py:195 aleksis/core/templates/core/person/full.html:137
 msgid "Guardians / Parents"
 msgstr "Parentes"
 
-#: models.py:173
+#: aleksis/core/models.py:202
 msgid "Primary group"
 msgstr ""
 
-#: models.py:176 models.py:461 models.py:485 models.py:570 models.py:805
-#: templates/core/person/full.html:120
+#: aleksis/core/models.py:205 aleksis/core/models.py:563
+#: aleksis/core/models.py:587 aleksis/core/models.py:672
+#: aleksis/core/models.py:951 aleksis/core/templates/core/person/full.html:120
 msgid "Description"
 msgstr "Descriptio"
 
-#: models.py:313
+#: aleksis/core/models.py:370
 msgid "Title of field"
 msgstr ""
 
-#: models.py:315
+#: aleksis/core/models.py:372
 msgid "Type of field"
 msgstr ""
 
-#: models.py:322
+#: aleksis/core/models.py:379
 #, fuzzy
 #| msgid "Additional name(s)"
 msgid "Addtitional field for groups"
 msgstr "addita nomines"
 
-#: models.py:323
+#: aleksis/core/models.py:380
 #, fuzzy
 #| msgid "Additional name(s)"
 msgid "Addtitional fields for groups"
 msgstr "addita nomines"
 
-#: models.py:337
+#: aleksis/core/models.py:397
 msgid "Group"
 msgstr "Grex"
 
-#: models.py:340
+#: aleksis/core/models.py:400
 msgid "Can assign child groups to groups"
 msgstr ""
 
-#: models.py:341
+#: aleksis/core/models.py:401
 #, fuzzy
 #| msgid "Persons and accounts"
 msgid "Can view statistics about group."
 msgstr "Personae et computi"
 
-#: models.py:352
+#: aleksis/core/models.py:413
 #, fuzzy
 #| msgid "Last name"
 msgid "Long name"
 msgstr "Secondus nomen"
 
-#: models.py:362 templates/core/group/full.html:85
+#: aleksis/core/models.py:423 aleksis/core/templates/core/group/full.html:85
 msgid "Members"
 msgstr ""
 
-#: models.py:365 templates/core/group/full.html:82
+#: aleksis/core/models.py:426 aleksis/core/templates/core/group/full.html:82
 msgid "Owners"
 msgstr ""
 
-#: models.py:372 templates/core/group/full.html:55
+#: aleksis/core/models.py:433 aleksis/core/templates/core/group/full.html:55
 msgid "Parent groups"
 msgstr ""
 
-#: models.py:380
+#: aleksis/core/models.py:441
 msgid "Type of group"
 msgstr ""
 
-#: models.py:457
+#: aleksis/core/models.py:559
 msgid "User"
 msgstr ""
 
-#: models.py:460 models.py:484 models.py:569
-#: templates/core/announcement/list.html:18
+#: aleksis/core/models.py:562 aleksis/core/models.py:586
+#: aleksis/core/models.py:671
+#: aleksis/core/templates/core/announcement/list.html:18
 msgid "Title"
 msgstr "Titulus"
 
-#: models.py:463
+#: aleksis/core/models.py:565
 msgid "Application"
 msgstr ""
 
-#: models.py:469
+#: aleksis/core/models.py:571
 msgid "Activity"
 msgstr ""
 
-#: models.py:470
+#: aleksis/core/models.py:572
 msgid "Activities"
 msgstr ""
 
-#: models.py:476
+#: aleksis/core/models.py:578
 msgid "Sender"
 msgstr "Mittens"
 
-#: models.py:481
+#: aleksis/core/models.py:583
 msgid "Recipient"
 msgstr ""
 
-#: models.py:486 models.py:786
+#: aleksis/core/models.py:588 aleksis/core/models.py:928
 msgid "Link"
 msgstr ""
 
-#: models.py:488
+#: aleksis/core/models.py:590
 msgid "Read"
 msgstr ""
 
-#: models.py:489
+#: aleksis/core/models.py:591
 msgid "Sent"
 msgstr ""
 
-#: models.py:502
+#: aleksis/core/models.py:604
 #, fuzzy
 #| msgid "Notifications"
 msgid "Notification"
 msgstr "Nuntii"
 
-#: models.py:503
-msgid "Notifications"
-msgstr "Nuntii"
-
-#: models.py:571
+#: aleksis/core/models.py:673
 msgid "Link to detailed view"
 msgstr ""
 
-#: models.py:574
+#: aleksis/core/models.py:676
 msgid "Date and time from when to show"
 msgstr ""
 
-#: models.py:577
+#: aleksis/core/models.py:679
 msgid "Date and time until when to show"
 msgstr ""
 
-#: models.py:601
+#: aleksis/core/models.py:703
 #, fuzzy
 #| msgid "Announcements"
 msgid "Announcement"
 msgstr "Nuntii"
 
-#: models.py:639
+#: aleksis/core/models.py:741
 #, fuzzy
 #| msgid "Announcements"
 msgid "Announcement recipient"
 msgstr "Nuntii"
 
-#: models.py:640
+#: aleksis/core/models.py:742
 #, fuzzy
 #| msgid "Announcements"
 msgid "Announcement recipients"
 msgstr "Nuntii"
 
-#: models.py:690
+#: aleksis/core/models.py:797
 #, fuzzy
 #| msgid "Site title"
 msgid "Widget Title"
 msgstr "Titulus paginae"
 
-#: models.py:691
+#: aleksis/core/models.py:798
 msgid "Activate Widget"
 msgstr ""
 
-#: models.py:694
+#: aleksis/core/models.py:799
+#, fuzzy
+#| msgid "Site title"
+msgid "Widget is broken"
+msgstr "Titulus paginae"
+
+#: aleksis/core/models.py:802
 msgid "Size on mobile devices"
 msgstr ""
 
-#: models.py:695
+#: aleksis/core/models.py:803
 msgid "<= 600 px, 12 columns"
 msgstr ""
 
-#: models.py:700
+#: aleksis/core/models.py:808
 msgid "Size on tablet devices"
 msgstr ""
 
-#: models.py:701
+#: aleksis/core/models.py:809
 msgid "> 600 px, 12 columns"
 msgstr ""
 
-#: models.py:706
+#: aleksis/core/models.py:814
 msgid "Size on desktop devices"
 msgstr ""
 
-#: models.py:707
+#: aleksis/core/models.py:815
 msgid "> 992 px, 12 columns"
 msgstr ""
 
-#: models.py:712
+#: aleksis/core/models.py:820
 msgid "Size on large desktop devices"
 msgstr ""
 
-#: models.py:713
+#: aleksis/core/models.py:821
 msgid "> 1200 px>, 12 columns"
 msgstr ""
 
-#: models.py:734
+#: aleksis/core/models.py:852
 #, fuzzy
 #| msgid "Dashboard"
 msgid "Can edit default dashboard"
 msgstr "Forum"
 
-#: models.py:735
+#: aleksis/core/models.py:853
 #, fuzzy
 #| msgid "Dashboard"
 msgid "Dashboard Widget"
 msgstr "Forum"
 
-#: models.py:736
+#: aleksis/core/models.py:854
 #, fuzzy
 #| msgid "Dashboard"
 msgid "Dashboard Widgets"
 msgstr "Forum"
 
-#: models.py:741
+#: aleksis/core/models.py:860
+msgid "URL"
+msgstr ""
+
+#: aleksis/core/models.py:861
+#, fuzzy
+#| msgid "Icon"
+msgid "Icon URL"
+msgstr "Nota"
+
+#: aleksis/core/models.py:867
+msgid "External link widget"
+msgstr ""
+
+#: aleksis/core/models.py:868
+msgid "External link widgets"
+msgstr ""
+
+#: aleksis/core/models.py:873
 #, fuzzy
 #| msgid "Dashboard"
 msgid "Dashboard widget"
 msgstr "Forum"
 
-#: models.py:746
+#: aleksis/core/models.py:878
 msgid "Order"
 msgstr ""
 
-#: models.py:747
+#: aleksis/core/models.py:879
 msgid "Part of the default dashboard"
 msgstr ""
 
-#: models.py:755
+#: aleksis/core/models.py:894
 #, fuzzy
 #| msgid "Dashboard"
 msgid "Dashboard widget order"
 msgstr "Forum"
 
-#: models.py:756
+#: aleksis/core/models.py:895
 #, fuzzy
 #| msgid "Dashboard"
 msgid "Dashboard widget orders"
 msgstr "Forum"
 
-#: models.py:762
+#: aleksis/core/models.py:901
 msgid "Menu ID"
 msgstr ""
 
-#: models.py:775
+#: aleksis/core/models.py:914
 msgid "Custom menu"
 msgstr ""
 
-#: models.py:776
+#: aleksis/core/models.py:915
 msgid "Custom menus"
 msgstr ""
 
-#: models.py:783
+#: aleksis/core/models.py:925
 msgid "Menu"
 msgstr ""
 
-#: models.py:787
+#: aleksis/core/models.py:929
 msgid "Icon"
 msgstr "Nota"
 
-#: models.py:793
+#: aleksis/core/models.py:935
 msgid "Custom menu item"
 msgstr ""
 
-#: models.py:794
+#: aleksis/core/models.py:936
 msgid "Custom menu items"
 msgstr ""
 
-#: models.py:804
+#: aleksis/core/models.py:950
 msgid "Title of type"
 msgstr ""
 
-#: models.py:811 templates/core/group/full.html:47
+#: aleksis/core/models.py:957 aleksis/core/templates/core/group/full.html:47
 #, fuzzy
 #| msgid "Group"
 msgid "Group type"
 msgstr "Grex"
 
-#: models.py:821
+#: aleksis/core/models.py:971
 #, fuzzy
 #| msgid "System status"
 msgid "Can view system status"
 msgstr "Status systemae"
 
-#: models.py:822
-#, fuzzy
-#| msgid "Persons and accounts"
-msgid "Can link persons to accounts"
-msgstr "Personae et computi"
-
-#: models.py:823
+#: aleksis/core/models.py:972
 #, fuzzy
 #| msgid "Data management"
 msgid "Can manage data"
 msgstr "Adminstratio datarum"
 
-#: models.py:824
+#: aleksis/core/models.py:973
 #, fuzzy
 #| msgid "Stop impersonation"
 msgid "Can impersonate"
 msgstr "Simulandum aliquem finire"
 
-#: models.py:825
+#: aleksis/core/models.py:974
 msgid "Can use search"
 msgstr ""
 
-#: models.py:826
+#: aleksis/core/models.py:975
 msgid "Can change site preferences"
 msgstr ""
 
-#: models.py:827
+#: aleksis/core/models.py:976
 msgid "Can change person preferences"
 msgstr ""
 
-#: models.py:828
+#: aleksis/core/models.py:977
 msgid "Can change group preferences"
 msgstr ""
 
-#: models.py:864
+#: aleksis/core/models.py:978
+msgid "Can add oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:979
+msgid "Can list oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:980
+#, fuzzy
+#| msgid "E-mail address"
+msgid "Can view oauth applications"
+msgstr "Inscriptio electronica"
+
+#: aleksis/core/models.py:981
+msgid "Can update oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:982
+msgid "Can delete oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:983
+msgid "Can test PDF generation"
+msgstr ""
+
+#: aleksis/core/models.py:1019
 msgid "Related data check task"
 msgstr ""
 
-#: models.py:872
+#: aleksis/core/models.py:1027
 msgid "Issue solved"
 msgstr ""
 
-#: models.py:873
+#: aleksis/core/models.py:1028
 #, fuzzy
 #| msgid "Notifications"
 msgid "Notification sent"
 msgstr "Nuntii"
 
-#: models.py:886
+#: aleksis/core/models.py:1041
 msgid "Data check result"
 msgstr ""
 
-#: models.py:887
+#: aleksis/core/models.py:1042
 msgid "Data check results"
 msgstr ""
 
-#: models.py:889
+#: aleksis/core/models.py:1044
 msgid "Can run data checks"
 msgstr ""
 
-#: models.py:890
+#: aleksis/core/models.py:1045
 msgid "Can solve data check problems"
 msgstr ""
 
-#: preferences.py:27
+#: aleksis/core/models.py:1060
+msgid "Owner"
+msgstr ""
+
+#: aleksis/core/models.py:1064
+msgid "File expires at"
+msgstr ""
+
+#: aleksis/core/models.py:1066
+msgid "Generated HTML file"
+msgstr ""
+
+#: aleksis/core/models.py:1068
+msgid "Generated PDF file"
+msgstr ""
+
+#: aleksis/core/models.py:1075
+msgid "PDF file"
+msgstr ""
+
+#: aleksis/core/models.py:1076
+msgid "PDF files"
+msgstr ""
+
+#: aleksis/core/models.py:1081
+msgid "Task result"
+msgstr ""
+
+#: aleksis/core/models.py:1084
+msgid "Task user"
+msgstr ""
+
+#: aleksis/core/models.py:1096
+msgid "Task user assignment"
+msgstr ""
+
+#: aleksis/core/models.py:1097
+msgid "Task user assignments"
+msgstr ""
+
+#: aleksis/core/preferences.py:22
+msgid "General"
+msgstr ""
+
+#: aleksis/core/preferences.py:23
+msgid "School"
+msgstr "Scola"
+
+#: aleksis/core/preferences.py:24
+msgid "Theme"
+msgstr ""
+
+#: aleksis/core/preferences.py:25
+msgid "Mail"
+msgstr ""
+
+#: aleksis/core/preferences.py:27
+msgid "Footer"
+msgstr ""
+
+#: aleksis/core/preferences.py:28
+#, fuzzy
+#| msgid "Data management"
+msgid "Accounts"
+msgstr "Adminstratio datarum"
+
+#: aleksis/core/preferences.py:29
 #, fuzzy
 #| msgid "Notifications"
 msgid "Authentication"
 msgstr "Nuntii"
 
-#: preferences.py:28
+#: aleksis/core/preferences.py:30
 #, fuzzy
 #| msgid "Impersonation"
 msgid "Internationalisation"
 msgstr "Simulare aliquem"
 
-#: preferences.py:37
+#: aleksis/core/preferences.py:41
 msgid "Site title"
 msgstr "Titulus paginae"
 
-#: preferences.py:46
+#: aleksis/core/preferences.py:52
 msgid "Site description"
 msgstr "Descriptio paginae"
 
-#: preferences.py:55
+#: aleksis/core/preferences.py:63
 msgid "Primary colour"
 msgstr ""
 
-#: preferences.py:64
+#: aleksis/core/preferences.py:75
 msgid "Secondary colour"
 msgstr ""
 
-#: preferences.py:72
+#: aleksis/core/preferences.py:86
 #, fuzzy
 #| msgid "Logout"
 msgid "Logo"
 msgstr "nomen retractare"
 
-#: preferences.py:80
+#: aleksis/core/preferences.py:96
 msgid "Favicon"
 msgstr ""
 
-#: preferences.py:88
+#: aleksis/core/preferences.py:106
 #, fuzzy
 #| msgid "Icon"
 msgid "PWA-Icon"
 msgstr "Nota"
 
-#: preferences.py:97
+#: aleksis/core/preferences.py:117
 #, fuzzy
 #| msgid "Last name"
 msgid "Mail out name"
 msgstr "Secondus nomen"
 
-#: preferences.py:106
+#: aleksis/core/preferences.py:128
 #, fuzzy
 #| msgid "E-mail address"
 msgid "Mail out address"
 msgstr "Inscriptio electronica"
 
-#: preferences.py:116
+#: aleksis/core/preferences.py:140
 msgid "Link to privacy policy"
 msgstr ""
 
-#: preferences.py:126
+#: aleksis/core/preferences.py:152
 msgid "Link to imprint"
 msgstr ""
 
-#: preferences.py:136
+#: aleksis/core/preferences.py:164
 msgid "Name format for addressing"
 msgstr ""
 
-#: preferences.py:150
+#: aleksis/core/preferences.py:180
 msgid "Channels to use for notifications"
 msgstr ""
 
-#: preferences.py:160
+#: aleksis/core/preferences.py:192
 msgid "Regular expression to match primary group, e.g. '^Class .*'"
 msgstr ""
 
-#: preferences.py:169
+#: aleksis/core/preferences.py:203
 msgid "Field on person to match primary group against"
 msgstr ""
 
-#: preferences.py:181
+#: aleksis/core/preferences.py:215
+msgid "Automatically create new persons for new users"
+msgstr ""
+
+#: aleksis/core/preferences.py:224
+msgid "Automatically link existing persons to new users by their e-mail address"
+msgstr ""
+
+#: aleksis/core/preferences.py:235
 msgid "Display name of the school"
 msgstr ""
 
-#: preferences.py:190
+#: aleksis/core/preferences.py:246
 msgid "Official name of the school, e.g. as given by supervisory authority"
 msgstr "Officialis nomen scolae, e. g."
 
-#: preferences.py:198
-msgid "Enabled custom authentication backends"
+#: aleksis/core/preferences.py:254
+msgid "Allow users to change their passwords"
+msgstr ""
+
+#: aleksis/core/preferences.py:262
+msgid "Enable signup"
 msgstr ""
 
-#: preferences.py:211
+#: aleksis/core/preferences.py:273
 msgid "Available languages"
 msgstr ""
 
-#: preferences.py:223
+#: aleksis/core/preferences.py:285
 msgid "Send emails if data checks detect problems"
 msgstr ""
 
-#: preferences.py:234
+#: aleksis/core/preferences.py:296
 msgid "Email recipients for data checks problem emails"
 msgstr ""
 
-#: preferences.py:245
+#: aleksis/core/preferences.py:307
 msgid "Email recipient groups for data checks problem emails"
 msgstr ""
 
-#: settings.py:322
-msgid "English"
-msgstr "Britannicus"
+#: aleksis/core/preferences.py:316
+msgid "Show dashboard to users without login"
+msgstr ""
 
-#: settings.py:323
-msgid "German"
-msgstr "Germanus"
+#: aleksis/core/preferences.py:325
+msgid "Allow users to edit their dashboard"
+msgstr ""
+
+#: aleksis/core/preferences.py:336
+msgid "Fields on person model which are editable by themselves."
+msgstr ""
+
+#: aleksis/core/preferences.py:350
+msgid "Editable fields on person model which should trigger a notification on change"
+msgstr ""
+
+#: aleksis/core/preferences.py:363
+msgid "Contact for notification if a person changes their data"
+msgstr ""
 
-#: settings.py:324
-msgid "French"
+#: aleksis/core/preferences.py:373
+msgid "PDF file expiration duration"
 msgstr ""
 
-#: settings.py:325
-msgid "Norwegian (bokmål)"
+#: aleksis/core/preferences.py:374
+msgid "in minutes"
 msgstr ""
 
-#: tables.py:19 templates/core/announcement/list.html:36
-#: templates/core/group/full.html:24 templates/core/person/full.html:23
+#: aleksis/core/preferences.py:384
+msgid "Automatically update the dashboard and its widgets"
+msgstr ""
+
+#: aleksis/core/preferences.py:394
+msgid "Automatically update the dashboard and its widgets sitewide"
+msgstr ""
+
+#: aleksis/core/settings.py:452
+msgid "English"
+msgstr "Britannicus"
+
+#: aleksis/core/settings.py:453
+msgid "German"
+msgstr "Germanus"
+
+#: aleksis/core/tables.py:19
+#: aleksis/core/templates/core/announcement/list.html:36
+#: aleksis/core/templates/core/group/full.html:24
+#: aleksis/core/templates/core/person/full.html:23
+#: aleksis/core/templates/oauth2_provider/application_detail.html:17
 msgid "Edit"
 msgstr ""
 
-#: tables.py:21 tables.py:89 templates/core/announcement/list.html:22
+#: aleksis/core/tables.py:21 aleksis/core/tables.py:89
+#: aleksis/core/templates/core/announcement/list.html:22
 #, fuzzy
 #| msgid "Notifications"
 msgid "Actions"
 msgstr "Nuntii"
 
-#: tables.py:56 tables.py:57 tables.py:71 tables.py:87
-#: templates/core/announcement/list.html:42 templates/core/group/full.html:31
-#: templates/core/pages/delete.html:22 templates/core/person/full.html:30
+#: aleksis/core/tables.py:56 aleksis/core/tables.py:57
+#: aleksis/core/tables.py:71 aleksis/core/tables.py:87
+#: aleksis/core/templates/core/announcement/list.html:42
+#: aleksis/core/templates/core/group/full.html:31
+#: aleksis/core/templates/core/pages/delete.html:22
+#: aleksis/core/templates/core/person/full.html:30
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:20
+#: aleksis/core/templates/oauth2_provider/application_detail.html:21
 msgid "Delete"
 msgstr ""
 
-#: templates/403.html:14 templates/404.html:10 templates/500.html:10
+#: aleksis/core/templates/403.html:14 aleksis/core/templates/404.html:10
+#: aleksis/core/templates/500.html:10
+#: aleksis/core/templates/oauth2_provider/authorize.html:47
+#: aleksis/core/templates/oauth2_provider/authorized-oob.html:24
 msgid "Error"
 msgstr ""
 
-#: templates/403.html:14
+#: aleksis/core/templates/403.html:14
 msgid ""
 "You are not allowed to access the requested page or\n"
 "          object."
 msgstr ""
 
-#: templates/403.html:19 templates/404.html:17
+#: aleksis/core/templates/403.html:19 aleksis/core/templates/404.html:17
 msgid ""
 "\n"
 "            If you think this is an error in AlekSIS, please contact your site\n"
@@ -935,13 +1248,13 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/404.html:10
+#: aleksis/core/templates/404.html:10
 msgid ""
 "The requested page or object was not\n"
 "          found."
 msgstr ""
 
-#: templates/404.html:13
+#: aleksis/core/templates/404.html:13
 msgid ""
 "\n"
 "            If you were redirected by a link on an external page,\n"
@@ -949,13 +1262,13 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/500.html:10
+#: aleksis/core/templates/500.html:10
 msgid ""
 "An unexpected error has\n"
 "          occured."
 msgstr ""
 
-#: templates/500.html:13
+#: aleksis/core/templates/500.html:13
 msgid ""
 "\n"
 "            Your site administrators will automatically be notified about this\n"
@@ -963,189 +1276,422 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/503.html:10
+#: aleksis/core/templates/503.html:10
 msgid ""
 "The maintenance mode is currently enabled. Please try again\n"
 "          later."
 msgstr ""
 
-#: templates/503.html:13
+#: aleksis/core/templates/503.html:13
 msgid ""
 "\n"
 "            This page is currently unavailable. If this error persists, contact your site administrators:\n"
 "          "
 msgstr ""
 
-#: templates/core/additional_field/edit.html:6
-#: templates/core/additional_field/edit.html:7
+#: aleksis/core/templates/account/account_inactive.html:5
+#: aleksis/core/templates/account/account_inactive.html:6
+msgid "Account inactive"
+msgstr ""
+
+#: aleksis/core/templates/account/account_inactive.html:13
+msgid "Account inactive."
+msgstr ""
+
+#: aleksis/core/templates/account/account_inactive.html:15
+msgid ""
+"\n"
+"            This account is currently inactive. If you think this is an\n"
+"            error, please contact one of your site administrators.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/email/base_message.txt:5
+msgid "Hello!"
+msgstr ""
+
+#: aleksis/core/templates/account/email/base_message.txt:9
+#: aleksis/core/templates/templated_email/notification.email:22
+#: aleksis/core/templates/templated_email/notification.email:46
+msgid "Your AlekSIS team"
+msgstr ""
+
+#: aleksis/core/templates/account/email_confirm.html:5
+#: aleksis/core/templates/account/email_confirm.html:6
+#: aleksis/core/templates/account/email_confirm.html:17
+#, fuzzy
+#| msgid "Notification"
+msgid "Confirm"
+msgstr "Nuntius"
+
+#: aleksis/core/templates/account/email_confirm.html:12
+#, python-format
+msgid "Please confirm that <a href=\"mailto:%(email)s\">%(email)s</a> is an e-mail address for user %(user_display)s."
+msgstr ""
+
+#: aleksis/core/templates/account/email_confirm.html:25
+#, python-format
+msgid "This e-mail confirmation link expired or is invalid. Please <a href=\"%(email_url)s\">issue a new e-mail confirmation request</a>."
+msgstr ""
+
+#: aleksis/core/templates/account/password_change.html:12
+msgid "Forgot your current password? Click here to reset it:"
+msgstr ""
+
+#: aleksis/core/templates/account/password_change.html:12
+msgid "Forgot Password?"
+msgstr ""
+
+#: aleksis/core/templates/account/password_change_disabled.html:5
+#: aleksis/core/templates/account/password_change_disabled.html:6
+msgid "Changing of password disabled"
+msgstr ""
+
+#: aleksis/core/templates/account/password_change_disabled.html:13
+msgid "Changing of password disabled."
+msgstr ""
+
+#: aleksis/core/templates/account/password_change_disabled.html:15
+msgid ""
+"\n"
+"            Users are not allowed to edit their own passwords. If you think\n"
+"            this is an error please contact one of your site administrators.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset.html:5
+#: aleksis/core/templates/account/password_reset.html:15
+#: aleksis/core/templates/account/password_reset.html:23
+#: aleksis/core/templates/account/password_reset_done.html:5
+#: aleksis/core/templates/account/verification_email_required.html:5
+#: aleksis/core/templates/account/verification_email_required.html:6
+#: aleksis/core/templates/two_factor/core/login.html:81
+msgid "Reset password"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset.html:17
+msgid "Forgotten your password? Enter your e-mail address below, and we'll send you an e-mail allowing you to reset it."
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset.html:30
+msgid ""
+"Please contact one of your site administrators, if you\n"
+"        have any trouble resetting your password:"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_done.html:15
+msgid "Password reset mail sent"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_done.html:18
+#: aleksis/core/templates/account/verification_email_required.html:16
+msgid ""
+"\n"
+"            We have sent you an e-mail. Please contact one of your site\n"
+"            administrators if you do not receive it within a few minutes.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:15
+msgid "Bad token"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:19
+#, python-format
+msgid ""
+"\n"
+"              The password reset link was invalid, possibly because it has already been used. Please request a <a href=\"%(passwd_reset_url)s\"\n"
+"              class=\"blue-text text-lighten-2\">new password reset</a>.\n"
+"            "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:25
+msgid ""
+"\n"
+"              If this issue persists, please contact one of your site\n"
+"              administrators\n"
+"            "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:56
+#: aleksis/core/templates/account/password_reset_from_key_done.html:15
+msgid ""
+"\n"
+"            Your password is now changed!\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:61
+msgid "Back to login"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key_done.html:13
+msgid "Password changed!"
+msgstr ""
+
+#: aleksis/core/templates/account/password_set.html:5
+#: aleksis/core/templates/account/password_set.html:6
+#: aleksis/core/templates/account/password_set.html:12
+msgid "Set password"
+msgstr ""
+
+#: aleksis/core/templates/account/signup.html:5
+#: aleksis/core/templates/account/signup.html:6
+#: aleksis/core/templates/socialaccount/signup.html:5
+#: aleksis/core/templates/socialaccount/signup.html:6
+msgid "Signup"
+msgstr ""
+
+#: aleksis/core/templates/account/signup.html:12
+#, python-format
+msgid "Already have an account? Then please <a href=\"%(login_url)s\">sign in</a>."
+msgstr ""
+
+#: aleksis/core/templates/account/signup_closed.html:5
+#: aleksis/core/templates/account/signup_closed.html:6
+msgid "Signup closed"
+msgstr ""
+
+#: aleksis/core/templates/account/signup_closed.html:13
+msgid "Signup closed."
+msgstr ""
+
+#: aleksis/core/templates/account/signup_closed.html:15
+msgid ""
+"\n"
+"            This sign up is currently closed. If you think this is an\n"
+"            error, please contact one of your site administrators.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/verification_email_required.html:14
+msgid "Password reset mail sent!"
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:5
+#: aleksis/core/templates/account/verification_sent.html:6
+#, fuzzy
+#| msgid "E-mail address"
+msgid "Verify your email address"
+msgstr "Inscriptio electronica"
+
+#: aleksis/core/templates/account/verification_sent.html:14
+msgid "Verify your email!"
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:16
+msgid ""
+"\n"
+"            This part of the site requires us to verify that you are who you claim to be.\n"
+"            For this purpose, we require that you verify ownership of your e-mail address.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:22
+msgid ""
+"\n"
+"            We have sent an e-mail to you for verification.\n"
+"            Please click on the link inside this e-mail. Please\n"
+"            contact us if you do not receive it within a few minutes.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:30
+#, python-format
+msgid "<strong>Note:</strong> you can still <a href=\"%(email_url)s\">change your e-mail address</a>"
+msgstr ""
+
+#: aleksis/core/templates/core/additional_field/edit.html:6
+#: aleksis/core/templates/core/additional_field/edit.html:7
 #, fuzzy
 #| msgid "Additional name(s)"
 msgid "Edit additional field"
 msgstr "addita nomines"
 
-#: templates/core/additional_field/list.html:14
+#: aleksis/core/templates/core/additional_field/list.html:14
 #, fuzzy
 #| msgid "Additional name(s)"
 msgid "Create additional field"
 msgstr "addita nomines"
 
-#: templates/core/announcement/form.html:14
-#: templates/core/announcement/form.html:21
+#: aleksis/core/templates/core/announcement/form.html:14
+#: aleksis/core/templates/core/announcement/form.html:21
 #, fuzzy
 #| msgid "Announcements"
 msgid "Edit announcement"
 msgstr "Nuntii"
 
-#: templates/core/announcement/form.html:16
+#: aleksis/core/templates/core/announcement/form.html:16
 #, fuzzy
 #| msgid "Announcements"
 msgid "Publish announcement"
 msgstr "Nuntii"
 
-#: templates/core/announcement/form.html:23
-#: templates/core/announcement/list.html:13
+#: aleksis/core/templates/core/announcement/form.html:23
+#: aleksis/core/templates/core/announcement/list.html:13
 #, fuzzy
 #| msgid "Who should see the announcement?"
 msgid "Publish new announcement"
 msgstr "Quis nuntium videatne?"
 
-#: templates/core/announcement/form.html:34
+#: aleksis/core/templates/core/announcement/form.html:34
 #, fuzzy
 #| msgid "Who should see the announcement?"
 msgid "Save und publish announcement"
 msgstr "Quis nuntium videatne?"
 
-#: templates/core/announcement/list.html:19
+#: aleksis/core/templates/core/announcement/list.html:19
 msgid "Valid from"
 msgstr ""
 
-#: templates/core/announcement/list.html:20
+#: aleksis/core/templates/core/announcement/list.html:20
 msgid "Valid until"
 msgstr ""
 
-#: templates/core/announcement/list.html:21
+#: aleksis/core/templates/core/announcement/list.html:21
 msgid "Recipients"
 msgstr ""
 
-#: templates/core/announcement/list.html:50
+#: aleksis/core/templates/core/announcement/list.html:50
 #, fuzzy
 #| msgid "Write your announcement:"
 msgid "There are no announcements."
 msgstr "Scribe nuntium:"
 
-#: templates/core/base.html:54
+#: aleksis/core/templates/core/base.html:60
 msgid "Logged in as"
 msgstr ""
 
-#: templates/core/base.html:147
+#: aleksis/core/templates/core/base.html:154
 msgid "About AlekSIS — The Free School Information System"
 msgstr ""
 
-#: templates/core/base.html:155
+#: aleksis/core/templates/core/base.html:162
 msgid "Impress"
 msgstr ""
 
-#: templates/core/base.html:163
+#: aleksis/core/templates/core/base.html:170
 msgid "Privacy Policy"
 msgstr ""
 
-#: templates/core/base_print.html:64
+#: aleksis/core/templates/core/base_print.html:72
 msgid "Powered by AlekSIS"
 msgstr ""
 
-#: templates/core/dashboard_widget/create.html:8
-#: templates/core/dashboard_widget/create.html:12
+#: aleksis/core/templates/core/dashboard_widget/create.html:8
+#: aleksis/core/templates/core/dashboard_widget/create.html:12
 #, fuzzy, python-format
 #| msgid "Stop impersonation"
 msgid "Create %(widget)s"
 msgstr "Simulandum aliquem finire"
 
-#: templates/core/dashboard_widget/edit.html:8
-#: templates/core/dashboard_widget/edit.html:12
+#: aleksis/core/templates/core/dashboard_widget/dashboardwidget_broken.html:10
+msgid "This widget is currently not available."
+msgstr ""
+
+#: aleksis/core/templates/core/dashboard_widget/dashboardwidget_broken.html:14
+#, python-format
+msgid ""
+"\n"
+"            There is a problem getting the widget \"%(title)s\".\n"
+"            There is no need for you to take any action.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/core/dashboard_widget/edit.html:8
+#: aleksis/core/templates/core/dashboard_widget/edit.html:12
 #, python-format
 msgid "Edit %(widget)s"
 msgstr ""
 
-#: templates/core/dashboard_widget/list.html:17
+#: aleksis/core/templates/core/dashboard_widget/list.html:15
+#, fuzzy
+#| msgid "Dashboard"
+msgid "Create dashboard widget"
+msgstr "Forum"
+
+#: aleksis/core/templates/core/dashboard_widget/list.html:22
 #, fuzzy, python-format
 #| msgid "Stop impersonation"
 msgid "Create %(name)s"
 msgstr "Simulandum aliquem finire"
 
-#: templates/core/dashboard_widget/list.html:25
-#: templates/core/edit_dashboard.html:8 templates/core/edit_dashboard.html:15
+#: aleksis/core/templates/core/dashboard_widget/list.html:32
+#: aleksis/core/templates/core/edit_dashboard.html:8
+#: aleksis/core/templates/core/edit_dashboard.html:15
 #, fuzzy
 #| msgid "Dashboard"
 msgid "Edit default dashboard"
 msgstr "Forum"
 
-#: templates/core/data_check/list.html:15
+#: aleksis/core/templates/core/data_check/list.html:15
 msgid "Check data again"
 msgstr ""
 
-#: templates/core/data_check/list.html:22
+#: aleksis/core/templates/core/data_check/list.html:22
 msgid "The system detected some problems with your data."
 msgstr ""
 
-#: templates/core/data_check/list.html:23
+#: aleksis/core/templates/core/data_check/list.html:23
 msgid ""
 "Please go through all data and check whether some extra action is\n"
 "          needed."
 msgstr ""
 
-#: templates/core/data_check/list.html:31
+#: aleksis/core/templates/core/data_check/list.html:31
 msgid "Everything is fine."
 msgstr ""
 
-#: templates/core/data_check/list.html:32
+#: aleksis/core/templates/core/data_check/list.html:32
 msgid "The system hasn't detected any problems with your data."
 msgstr ""
 
-#: templates/core/data_check/list.html:40
+#: aleksis/core/templates/core/data_check/list.html:40
 msgid "Detected problems"
 msgstr ""
 
-#: templates/core/data_check/list.html:45
+#: aleksis/core/templates/core/data_check/list.html:45
 msgid "Affected object"
 msgstr ""
 
-#: templates/core/data_check/list.html:46
+#: aleksis/core/templates/core/data_check/list.html:46
 msgid "Detected problem"
 msgstr ""
 
-#: templates/core/data_check/list.html:47
+#: aleksis/core/templates/core/data_check/list.html:47
 msgid "Show details"
 msgstr ""
 
-#: templates/core/data_check/list.html:48
+#: aleksis/core/templates/core/data_check/list.html:48
 msgid "Options to solve the problem"
 msgstr ""
 
-#: templates/core/data_check/list.html:62
+#: aleksis/core/templates/core/data_check/list.html:62
 msgid "Show object"
 msgstr ""
 
-#: templates/core/data_check/list.html:84
+#: aleksis/core/templates/core/data_check/list.html:84
 #, fuzzy
 #| msgid "System status"
 msgid "Registered checks"
 msgstr "Status systemae"
 
-#: templates/core/data_check/list.html:88
+#: aleksis/core/templates/core/data_check/list.html:88
 msgid ""
 "\n"
 "            The system will check for the following problems:\n"
 "          "
 msgstr ""
 
-#: templates/core/edit_dashboard.html:6 templates/core/edit_dashboard.html:13
-#: templates/core/index.html:14
+#: aleksis/core/templates/core/edit_dashboard.html:6
+#: aleksis/core/templates/core/edit_dashboard.html:13
+#: aleksis/core/templates/core/index.html:17
 #, fuzzy
 #| msgid "Dashboard"
 msgid "Edit dashboard"
 msgstr "Forum"
 
-#: templates/core/edit_dashboard.html:24
+#: aleksis/core/templates/core/edit_dashboard.html:24
 msgid ""
 "\n"
 "          On this page you can arrange your personal dashboard. You can drag any items from \"Available widgets\" to \"Your\n"
@@ -1154,7 +1700,7 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/core/edit_dashboard.html:30
+#: aleksis/core/templates/core/edit_dashboard.html:30
 msgid ""
 "\n"
 "          On this page you can arrange the default dashboard which is shown when a user doesn't arrange his own\n"
@@ -1163,23 +1709,23 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/core/edit_dashboard.html:48
+#: aleksis/core/templates/core/edit_dashboard.html:48
 msgid "Available widgets"
 msgstr ""
 
-#: templates/core/edit_dashboard.html:57
+#: aleksis/core/templates/core/edit_dashboard.html:57
 #, fuzzy
 #| msgid "Dashboard"
 msgid "Your dashboard"
 msgstr "Forum"
 
-#: templates/core/edit_dashboard.html:59
+#: aleksis/core/templates/core/edit_dashboard.html:59
 #, fuzzy
 #| msgid "Dashboard"
 msgid "Default dashboard"
 msgstr "Forum"
 
-#: templates/core/group/child_groups.html:18
+#: aleksis/core/templates/core/group/child_groups.html:18
 msgid ""
 "\n"
 "          You can use this to assign child groups to groups. Please use the filters below to select groups you want to\n"
@@ -1187,38 +1733,38 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/core/group/child_groups.html:31
+#: aleksis/core/templates/core/group/child_groups.html:31
 msgid "Update selection"
 msgstr ""
 
-#: templates/core/group/child_groups.html:35
+#: aleksis/core/templates/core/group/child_groups.html:35
 msgid "Clear all filters"
 msgstr ""
 
-#: templates/core/group/child_groups.html:39
+#: aleksis/core/templates/core/group/child_groups.html:39
 msgid "Currently selected groups"
 msgstr ""
 
-#: templates/core/group/child_groups.html:52
+#: aleksis/core/templates/core/group/child_groups.html:52
 msgid "Start assigning child groups for this groups"
 msgstr ""
 
-#: templates/core/group/child_groups.html:61
+#: aleksis/core/templates/core/group/child_groups.html:61
 msgid ""
 "\n"
 "            Please select some groups in order to go on with assigning.\n"
 "          "
 msgstr ""
 
-#: templates/core/group/child_groups.html:72
+#: aleksis/core/templates/core/group/child_groups.html:72
 msgid "Current group:"
 msgstr ""
 
-#: templates/core/group/child_groups.html:78
+#: aleksis/core/templates/core/group/child_groups.html:78
 msgid "Please be careful!"
 msgstr ""
 
-#: templates/core/group/child_groups.html:79
+#: aleksis/core/templates/core/group/child_groups.html:79
 msgid ""
 "\n"
 "            If you click \"Back\" or \"Next\" the current group assignments are not saved.\n"
@@ -1227,137 +1773,145 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/core/group/child_groups.html:93
-#: templates/core/group/child_groups.html:128
-#: templates/two_factor/_wizard_actions.html:15
-#: templates/two_factor/_wizard_actions.html:20
+#: aleksis/core/templates/core/group/child_groups.html:93
+#: aleksis/core/templates/core/group/child_groups.html:128
+#: aleksis/core/templates/oauth2_provider/application_detail.html:9
+#: aleksis/core/templates/two_factor/_wizard_actions.html:15
+#: aleksis/core/templates/two_factor/_wizard_actions.html:20
 msgid "Back"
 msgstr ""
 
-#: templates/core/group/child_groups.html:99
-#: templates/core/group/child_groups.html:134
-#: templates/two_factor/_wizard_actions.html:26
+#: aleksis/core/templates/core/group/child_groups.html:99
+#: aleksis/core/templates/core/group/child_groups.html:134
+#: aleksis/core/templates/two_factor/_wizard_actions.html:26
 msgid "Next"
 msgstr ""
 
-#: templates/core/group/child_groups.html:106
-#: templates/core/group/child_groups.html:141
-#: templates/core/partials/save_button.html:3
+#: aleksis/core/templates/core/group/child_groups.html:106
+#: aleksis/core/templates/core/group/child_groups.html:141
+#: aleksis/core/templates/core/partials/save_button.html:3
 msgid "Save"
 msgstr ""
 
-#: templates/core/group/child_groups.html:112
-#: templates/core/group/child_groups.html:147
+#: aleksis/core/templates/core/group/child_groups.html:112
+#: aleksis/core/templates/core/group/child_groups.html:147
 msgid "Save and next"
 msgstr ""
 
-#: templates/core/group/edit.html:11 templates/core/group/edit.html:12
+#: aleksis/core/templates/core/group/edit.html:11
+#: aleksis/core/templates/core/group/edit.html:12
 msgid "Edit group"
 msgstr ""
 
-#: templates/core/group/full.html:38 templates/core/person/full.html:37
+#: aleksis/core/templates/core/group/full.html:38
+#: aleksis/core/templates/core/person/full.html:37
 msgid "Change preferences"
 msgstr ""
 
-#: templates/core/group/full.html:64
+#: aleksis/core/templates/core/group/full.html:64
 #, fuzzy
 #| msgid "System status"
 msgid "Statistics"
 msgstr "Status systemae"
 
-#: templates/core/group/full.html:67
+#: aleksis/core/templates/core/group/full.html:67
 msgid "Count of members"
 msgstr ""
 
-#: templates/core/group/full.html:71
+#: aleksis/core/templates/core/group/full.html:71
 msgid "Average age"
 msgstr ""
 
-#: templates/core/group/full.html:76
+#: aleksis/core/templates/core/group/full.html:76
 msgid "Age range"
 msgstr ""
 
-#: templates/core/group/full.html:76
+#: aleksis/core/templates/core/group/full.html:76
 msgid "years to"
 msgstr ""
 
-#: templates/core/group/full.html:76
+#: aleksis/core/templates/core/group/full.html:76
 msgid "years "
 msgstr ""
 
-#: templates/core/group/list.html:14
+#: aleksis/core/templates/core/group/list.html:14
 msgid "Create group"
 msgstr ""
 
-#: templates/core/group/list.html:17
+#: aleksis/core/templates/core/group/list.html:17
 msgid "Filter groups"
 msgstr ""
 
-#: templates/core/group/list.html:24 templates/core/person/list.html:28
+#: aleksis/core/templates/core/group/list.html:24
+#: aleksis/core/templates/core/person/list.html:28
 msgid "Clear"
 msgstr ""
 
-#: templates/core/group/list.html:28
+#: aleksis/core/templates/core/group/list.html:28
 msgid "Selected groups"
 msgstr ""
 
-#: templates/core/group_type/edit.html:6 templates/core/group_type/edit.html:7
+#: aleksis/core/templates/core/group_type/edit.html:6
+#: aleksis/core/templates/core/group_type/edit.html:7
 #, fuzzy
 #| msgid "Group"
 msgid "Edit group type"
 msgstr "Grex"
 
-#: templates/core/group_type/list.html:14
+#: aleksis/core/templates/core/group_type/list.html:14
 #, fuzzy
 #| msgid "Group"
 msgid "Create group type"
 msgstr "Grex"
 
-#: templates/core/index.html:4
+#: aleksis/core/templates/core/index.html:4
 msgid "Home"
 msgstr ""
 
-#: templates/core/index.html:50
+#: aleksis/core/templates/core/index.html:49
 msgid ""
 "\n"
-"          You didn't customise your dashboard so that you see the system default. Please click on \"Edit dashboard\" to\n"
-"          customise your personal dashboard.\n"
-"        "
+"        You didn't customise your dashboard so that you see the system default. Please click on \"Edit dashboard\" to\n"
+"        customise your personal dashboard.\n"
+"      "
 msgstr ""
 
-#: templates/core/index.html:59
+#: aleksis/core/templates/core/index.html:59
 msgid "Last activities"
 msgstr ""
 
-#: templates/core/index.html:77
+#: aleksis/core/templates/core/index.html:77
 msgid "No activities available yet."
 msgstr ""
 
-#: templates/core/index.html:82
+#: aleksis/core/templates/core/index.html:82
 #, fuzzy
 #| msgid "Notifications"
 msgid "Recent notifications"
 msgstr "Nuntii"
 
-#: templates/core/index.html:98
+#: aleksis/core/templates/core/index.html:98
+#: aleksis/core/templates/core/notifications.html:23
 #, fuzzy
 #| msgid "Edit school information"
 msgid "More information →"
 msgstr "Muta informationes scolae"
 
-#: templates/core/index.html:105
+#: aleksis/core/templates/core/index.html:105
+#: aleksis/core/templates/core/notifications.html:30
 msgid "No notifications available yet."
 msgstr ""
 
-#: templates/core/pages/about.html:6 templates/core/pages/about.html:15
+#: aleksis/core/templates/core/pages/about.html:6
+#: aleksis/core/templates/core/pages/about.html:15
 msgid "About AlekSIS"
 msgstr ""
 
-#: templates/core/pages/about.html:7
+#: aleksis/core/templates/core/pages/about.html:7
 msgid "AlekSIS – The Free School Information System"
 msgstr ""
 
-#: templates/core/pages/about.html:17
+#: aleksis/core/templates/core/pages/about.html:17
 msgid ""
 "\n"
 "              This platform is powered by AlekSIS, a web-based school information system (SIS) which can be used\n"
@@ -1366,21 +1920,21 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/pages/about.html:25
+#: aleksis/core/templates/core/pages/about.html:25
 msgid "Website of AlekSIS"
 msgstr ""
 
-#: templates/core/pages/about.html:26
+#: aleksis/core/templates/core/pages/about.html:26
 msgid "Source code"
 msgstr ""
 
-#: templates/core/pages/about.html:35
+#: aleksis/core/templates/core/pages/about.html:35
 #, fuzzy
 #| msgid "Edit school information"
 msgid "Licence information"
 msgstr "Muta informationes scolae"
 
-#: templates/core/pages/about.html:37
+#: aleksis/core/templates/core/pages/about.html:37
 msgid ""
 "\n"
 "              The core and the official apps of AlekSIS are licenced under the EUPL, version 1.2 or later. For licence\n"
@@ -1389,25 +1943,25 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/pages/about.html:45
+#: aleksis/core/templates/core/pages/about.html:45
 msgid "Free/Open Source Licence"
 msgstr ""
 
-#: templates/core/pages/about.html:46
+#: aleksis/core/templates/core/pages/about.html:46
 msgid "Other Licence"
 msgstr ""
 
-#: templates/core/pages/about.html:50
+#: aleksis/core/templates/core/pages/about.html:50
 msgid "Full licence text"
 msgstr ""
 
-#: templates/core/pages/about.html:51
+#: aleksis/core/templates/core/pages/about.html:51
 #, fuzzy
 #| msgid "Edit school information"
 msgid "More information about the EUPL"
 msgstr "Muta informationes scolae"
 
-#: templates/core/pages/about.html:90
+#: aleksis/core/templates/core/pages/about.html:90
 #, python-format
 msgid ""
 "\n"
@@ -1415,12 +1969,12 @@ msgid ""
 "                  "
 msgstr ""
 
-#: templates/core/pages/delete.html:6
+#: aleksis/core/templates/core/pages/delete.html:6
 #, python-format
 msgid "Delete %(object_name)s"
 msgstr ""
 
-#: templates/core/pages/delete.html:13
+#: aleksis/core/templates/core/pages/delete.html:13
 #, python-format
 msgid ""
 "\n"
@@ -1428,110 +1982,122 @@ msgid ""
 "    "
 msgstr ""
 
-#: templates/core/pages/progress.html:27
+#: aleksis/core/templates/core/pages/progress.html:27
 msgid ""
 "\n"
 "              Without activated JavaScript the progress status can't be updated.\n"
 "            "
 msgstr ""
 
-#: templates/core/pages/progress.html:47
-#: templates/two_factor/core/otp_required.html:19
+#: aleksis/core/templates/core/pages/progress.html:47
+#: aleksis/core/templates/two_factor/core/otp_required.html:19
 msgid "Go back"
 msgstr ""
 
-#: templates/core/pages/system_status.html:12
+#: aleksis/core/templates/core/pages/system_status.html:12
 #, fuzzy
 #| msgid "System status"
 msgid "System checks"
 msgstr "Status systemae"
 
-#: templates/core/pages/system_status.html:21
+#: aleksis/core/templates/core/pages/system_status.html:21
 msgid "Maintenance mode enabled"
 msgstr ""
 
-#: templates/core/pages/system_status.html:23
+#: aleksis/core/templates/core/pages/system_status.html:23
 msgid ""
 "\n"
 "                Only admin and visitors from internal IPs can access thesite.\n"
 "              "
 msgstr ""
 
-#: templates/core/pages/system_status.html:34
+#: aleksis/core/templates/core/pages/system_status.html:34
 msgid "Maintenance mode disabled"
 msgstr ""
 
-#: templates/core/pages/system_status.html:35
+#: aleksis/core/templates/core/pages/system_status.html:35
 msgid "Everyone can access the site."
 msgstr ""
 
-#: templates/core/pages/system_status.html:45
+#: aleksis/core/templates/core/pages/system_status.html:45
 msgid "Debug mode enabled"
 msgstr ""
 
-#: templates/core/pages/system_status.html:47
+#: aleksis/core/templates/core/pages/system_status.html:47
 msgid ""
 "\n"
 "                The web server throws back debug information on errors. Do not use in production!\n"
 "              "
 msgstr ""
 
-#: templates/core/pages/system_status.html:54
+#: aleksis/core/templates/core/pages/system_status.html:54
 msgid "Debug mode disabled"
 msgstr ""
 
-#: templates/core/pages/system_status.html:56
+#: aleksis/core/templates/core/pages/system_status.html:56
 msgid ""
 "\n"
 "                Debug mode is disabled. Default error pages are displayed on errors.\n"
 "              "
 msgstr ""
 
-#: templates/core/pages/system_status.html:69
+#: aleksis/core/templates/core/pages/system_status.html:69
 #, fuzzy
 #| msgid "System status"
 msgid "System health checks"
 msgstr "Status systemae"
 
-#: templates/core/pages/system_status.html:75
+#: aleksis/core/templates/core/pages/system_status.html:75
 msgid "Service"
 msgstr ""
 
-#: templates/core/pages/system_status.html:76
-#: templates/core/pages/system_status.html:115
+#: aleksis/core/templates/core/pages/system_status.html:76
+#: aleksis/core/templates/core/pages/system_status.html:115
 #, fuzzy
 #| msgid "System status"
 msgid "Status"
 msgstr "Status systemae"
 
-#: templates/core/pages/system_status.html:77
+#: aleksis/core/templates/core/pages/system_status.html:77
 msgid "Time taken"
 msgstr ""
 
-#: templates/core/pages/system_status.html:96
+#: aleksis/core/templates/core/pages/system_status.html:96
 msgid "seconds"
 msgstr ""
 
-#: templates/core/pages/system_status.html:107
+#: aleksis/core/templates/core/pages/system_status.html:107
 msgid "Celery task results"
 msgstr ""
 
-#: templates/core/pages/system_status.html:112
+#: aleksis/core/templates/core/pages/system_status.html:112
 msgid "Task"
 msgstr ""
 
-#: templates/core/pages/system_status.html:113
+#: aleksis/core/templates/core/pages/system_status.html:113
 msgid "ID"
 msgstr ""
 
-#: templates/core/pages/system_status.html:114
+#: aleksis/core/templates/core/pages/system_status.html:114
 #, fuzzy
 #| msgid "Date"
 msgid "Date done"
 msgstr "dies"
 
-#: templates/core/partials/announcements.html:9
-#: templates/core/partials/announcements.html:36
+#: aleksis/core/templates/core/pages/test_pdf.html:7
+#: aleksis/core/templates/core/pages/test_pdf.html:8
+msgid "Test PDF generation"
+msgstr ""
+
+#: aleksis/core/templates/core/pages/test_pdf.html:14
+msgid ""
+"\n"
+"        This simple view can be used to ensure the correct function of the built-in PDF generation system.\n"
+"      "
+msgstr ""
+
+#: aleksis/core/templates/core/partials/announcements.html:8
+#: aleksis/core/templates/core/partials/announcements.html:35
 #, python-format
 msgid ""
 "\n"
@@ -1539,7 +2105,7 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/partials/announcements.html:13
+#: aleksis/core/templates/core/partials/announcements.html:12
 #, python-format
 msgid ""
 "\n"
@@ -1547,7 +2113,7 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/partials/announcements.html:40
+#: aleksis/core/templates/core/partials/announcements.html:39
 #, python-format
 msgid ""
 "\n"
@@ -1555,23 +2121,23 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/partials/crud_events.html:15
+#: aleksis/core/templates/core/partials/crud_events.html:15
 msgid "Changed by"
 msgstr ""
 
-#: templates/core/partials/crud_events.html:15
+#: aleksis/core/templates/core/partials/crud_events.html:15
 msgid "Unknown"
 msgstr ""
 
-#: templates/core/partials/language_form.html:15
+#: aleksis/core/templates/core/partials/language_form.html:15
 msgid "Language"
 msgstr ""
 
-#: templates/core/partials/language_form.html:27
+#: aleksis/core/templates/core/partials/language_form.html:27
 msgid "Select language"
 msgstr ""
 
-#: templates/core/partials/no_person.html:12
+#: aleksis/core/templates/core/partials/no_person.html:12
 msgid ""
 "\n"
 "            Your administrator account is not linked to any person. Therefore,\n"
@@ -1579,7 +2145,7 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/core/partials/no_person.html:19
+#: aleksis/core/templates/core/partials/no_person.html:19
 msgid ""
 "\n"
 "            Your user account is not linked to a person. This means you\n"
@@ -1588,115 +2154,190 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/core/person/accounts.html:12
-#: templates/core/person/accounts.html:14
+#: aleksis/core/templates/core/person/create.html:12
+#: aleksis/core/templates/core/person/create.html:13
+#: aleksis/core/templates/core/person/list.html:17
 #, fuzzy
-#| msgid "Persons and accounts"
-msgid "Link persons to accounts"
-msgstr "Personae et computi"
-
-#: templates/core/person/accounts.html:21
-msgid ""
-"\n"
-"        You can use this form to assign user accounts to persons. Use the\n"
-"        dropdowns to select existing accounts; use the text fields to create new\n"
-"        accounts on-the-fly. The latter will create a new account with the\n"
-"        entered username and copy all other details from the person.\n"
-"      "
-msgstr ""
-
-#: templates/core/person/accounts.html:36
-#: templates/core/person/accounts.html:60
-msgid "Update"
-msgstr ""
-
-#: templates/core/person/accounts.html:42
-msgid "Existing account"
-msgstr ""
-
-#: templates/core/person/accounts.html:43
-msgid "New account"
-msgstr ""
+#| msgid "Stop impersonation"
+msgid "Create person"
+msgstr "Simulandum aliquem finire"
 
-#: templates/core/person/edit.html:12 templates/core/person/edit.html:13
+#: aleksis/core/templates/core/person/edit.html:12
+#: aleksis/core/templates/core/person/edit.html:13
 msgid "Edit person"
 msgstr ""
 
-#: templates/core/person/full.html:44
+#: aleksis/core/templates/core/person/full.html:44
+#: aleksis/core/templates/impersonate/list_users.html:7
+#: aleksis/core/templates/impersonate/list_users.html:8
 #, fuzzy
 #| msgid "Impersonation"
 msgid "Impersonate"
 msgstr "Simulare aliquem"
 
-#: templates/core/person/full.html:50
+#: aleksis/core/templates/core/person/full.html:50
 msgid "Contact details"
 msgstr ""
 
-#: templates/core/person/full.html:130
+#: aleksis/core/templates/core/person/full.html:130
 msgid "Children"
 msgstr ""
 
-#: templates/core/person/list.html:17
-#, fuzzy
-#| msgid "Stop impersonation"
-msgid "Create person"
-msgstr "Simulandum aliquem finire"
-
-#: templates/core/person/list.html:21
+#: aleksis/core/templates/core/person/list.html:21
 msgid "Filter persons"
 msgstr ""
 
-#: templates/core/person/list.html:32
+#: aleksis/core/templates/core/person/list.html:32
 msgid "Selected persons"
 msgstr ""
 
-#: templates/core/school_term/create.html:6
-#: templates/core/school_term/create.html:7
-#: templates/core/school_term/list.html:14
+#: aleksis/core/templates/core/school_term/create.html:6
+#: aleksis/core/templates/core/school_term/create.html:7
+#: aleksis/core/templates/core/school_term/list.html:14
 #, fuzzy
 #| msgid "Edit school term"
 msgid "Create school term"
 msgstr "Muta anum scolae"
 
-#: templates/core/school_term/edit.html:6
-#: templates/core/school_term/edit.html:7
+#: aleksis/core/templates/core/school_term/edit.html:6
+#: aleksis/core/templates/core/school_term/edit.html:7
 msgid "Edit school term"
 msgstr "Muta anum scolae"
 
-#: templates/dynamic_preferences/form.html:9
+#: aleksis/core/templates/dynamic_preferences/form.html:9
 msgid "Site preferences"
 msgstr ""
 
-#: templates/dynamic_preferences/form.html:11
+#: aleksis/core/templates/dynamic_preferences/form.html:11
 msgid "My preferences"
 msgstr ""
 
-#: templates/dynamic_preferences/form.html:13
+#: aleksis/core/templates/dynamic_preferences/form.html:13
 #, python-format
 msgid "Preferences for %(instance)s"
 msgstr ""
 
-#: templates/dynamic_preferences/form.html:25
+#: aleksis/core/templates/dynamic_preferences/form.html:25
 msgid "Save preferences"
 msgstr ""
 
-#: templates/impersonate/list_users.html:8
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:5
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:6
+msgid "Delete application"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:12
+#, python-format
+msgid "Are you sure to delete the application %(application_name)s?"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:24
+#: aleksis/core/templates/oauth2_provider/application_form.html:18
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:24
+#: aleksis/core/templates/two_factor/_wizard_actions.html:6
+msgid "Cancel"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:27
+msgid "Client id"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:35
+msgid "Client secret"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:43
+msgid "Client type"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:51
+msgid "Authorization Grant Type"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:59
+msgid "Redirect URIs"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_form.html:5
+msgid "Create OAuth2 Application"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_form.html:10
 #, fuzzy
-#| msgid "Impersonation"
-msgid "Impersonate user"
-msgstr "Simulare aliquem"
+#| msgid "Edit school information"
+msgid "Edit application"
+msgstr "Muta informationes scolae"
+
+#: aleksis/core/templates/oauth2_provider/application_list.html:8
+#, fuzzy
+#| msgid "Notifications"
+msgid "OAuth2 applications"
+msgstr "Nuntii"
+
+#: aleksis/core/templates/oauth2_provider/application_list.html:12
+msgid "Register new applications"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_list.html:23
+msgid "No applications defined."
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorize.html:5
+#: aleksis/core/templates/oauth2_provider/authorize.html:16
+msgid "Authorize"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorize.html:18
+msgid "The application requests access to the following scopes:"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorize.html:33
+msgid "Allow"
+msgstr ""
 
-#: templates/offline.html:5
+#: aleksis/core/templates/oauth2_provider/authorize.html:36
+msgid "Disallow"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-oob.html:12
+msgid "Success!"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-oob.html:14
+msgid "Please return to your application and enter this code:"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:5
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:6
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:23
+msgid "Revoke access"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:12
+msgid "Are you sure to revoke the access for this application?"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:20
+msgid "Revoke"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:33
+#, fuzzy
+#| msgid "Notifications"
+msgid "No authorized applications."
+msgstr "Nuntii"
+
+#: aleksis/core/templates/offline.html:5
 msgid "Network error"
 msgstr ""
 
-#: templates/offline.html:8
+#: aleksis/core/templates/offline.html:8
 msgid ""
 "No internet\n"
 "    connection."
 msgstr ""
 
-#: templates/offline.html:12
+#: aleksis/core/templates/offline.html:12
 msgid ""
 "\n"
 "      There was an error accessing this page. You probably don't have an internet connection. Check to see if your WiFi\n"
@@ -1705,36 +2346,118 @@ msgid ""
 "    "
 msgstr ""
 
-#: templates/search/search.html:8
+#: aleksis/core/templates/search/search.html:8
 msgid "Global Search"
 msgstr ""
 
-#: templates/search/search.html:15
+#: aleksis/core/templates/search/search.html:15
 msgid "Search Term"
 msgstr ""
 
-#: templates/search/search.html:26
+#: aleksis/core/templates/search/search.html:26
 msgid "Results"
 msgstr ""
 
-#: templates/search/search.html:38
+#: aleksis/core/templates/search/search.html:38
 msgid "No search results could be found to your search."
 msgstr ""
 
-#: templates/search/search.html:87
+#: aleksis/core/templates/search/search.html:87
 msgid "Please enter a search term above."
 msgstr ""
 
-#: templates/templated_email/data_checks.email:4
+#: aleksis/core/templates/socialaccount/authentication_error.html:5
+#: aleksis/core/templates/socialaccount/authentication_error.html:6
+msgid "Third-party Account Login Failure"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/authentication_error.html:13
+msgid "Third-party Account Login Failure."
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/authentication_error.html:15
+msgid ""
+"\n"
+"            An error occurred while attempting to login via your third-party account.\n"
+"            Please contact one of your site administrators.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/connections.html:5
+#: aleksis/core/templates/socialaccount/connections.html:6
+#, fuzzy
+#| msgid "Notifications"
+msgid "Connections"
+msgstr "Nuntii"
+
+#: aleksis/core/templates/socialaccount/connections.html:24
+msgid "Remove"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/connections.html:34
+msgid "You currently have no third-party accounts connected to this account."
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/connections.html:37
+msgid "Add a Third-party Account"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/login_cancelled.html:5
+#: aleksis/core/templates/socialaccount/login_cancelled.html:6
+#: aleksis/core/templates/socialaccount/login_cancelled.html:13
+msgid "Login cancelled"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/login_cancelled.html:15
+#, python-format
+msgid ""
+"\n"
+"            You decided to cancel logging in to our site using one of your existing accounts. If this was a mistake, please proceed to <a href=\"%(login_url)s\">sign in</a>.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/signup.html:12
+#, python-format
+msgid ""
+"You are about to use your %(provider_name)s account to login to\n"
+"        %(site_name)s. As a final step, please complete the following form:"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/snippets/provider_list.html:12
+#, python-format
+msgid ""
+"\n"
+"                Login with %(name)s\n"
+"              "
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/snippets/provider_list.html:21
+#, python-format
+msgid ""
+"\n"
+"            Login with %(name)s\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/snippets/provider_list.html:30
+msgid ""
+"\n"
+"          No third-party account providers available.\n"
+"        "
+msgstr ""
+
+#: aleksis/core/templates/templated_email/data_checks.email:4
 msgid "The system detected some new problems with your data."
 msgstr ""
 
-#: templates/templated_email/data_checks.email:8
-#: templates/templated_email/data_checks.email:24
+#: aleksis/core/templates/templated_email/data_checks.email:8
+#: aleksis/core/templates/templated_email/data_checks.email:24
+#: aleksis/core/templates/templated_email/person_changed.email:8
+#: aleksis/core/templates/templated_email/person_changed.email:20
 msgid "Hello,"
 msgstr ""
 
-#: templates/templated_email/data_checks.email:10
+#: aleksis/core/templates/templated_email/data_checks.email:10
 msgid ""
 "\n"
 "  the system detected some new problems with your data.\n"
@@ -1742,7 +2465,7 @@ msgid ""
 " "
 msgstr ""
 
-#: templates/templated_email/data_checks.email:26
+#: aleksis/core/templates/templated_email/data_checks.email:26
 msgid ""
 "\n"
 "   the system detected some new problems with your data.\n"
@@ -1750,44 +2473,44 @@ msgid ""
 "  "
 msgstr ""
 
-#: templates/templated_email/data_checks.email:34
+#: aleksis/core/templates/templated_email/data_checks.email:34
 #, fuzzy
 #| msgid "Site description"
 msgid "Problem description"
 msgstr "Descriptio paginae"
 
-#: templates/templated_email/data_checks.email:35
+#: aleksis/core/templates/templated_email/data_checks.email:35
 msgid "Count of objects with new problems"
 msgstr ""
 
-#: templates/templated_email/notification.email:3
+#: aleksis/core/templates/templated_email/notification.email:3
 #, fuzzy
 #| msgid "Notification"
 msgid "New notification for"
 msgstr "Nuntius"
 
-#: templates/templated_email/notification.email:6
-#: templates/templated_email/notification.email:27
+#: aleksis/core/templates/templated_email/notification.email:6
+#: aleksis/core/templates/templated_email/notification.email:27
 #, fuzzy, python-format
 #| msgid "Notifications"
 msgid "Dear %(notification_user)s,"
 msgstr "Nuntii"
 
-#: templates/templated_email/notification.email:8
-#: templates/templated_email/notification.email:29
+#: aleksis/core/templates/templated_email/notification.email:8
+#: aleksis/core/templates/templated_email/notification.email:29
 #, fuzzy
 #| msgid "Notification"
 msgid "we got a new notification for you:"
 msgstr "Nuntius"
 
-#: templates/templated_email/notification.email:15
-#: templates/templated_email/notification.email:35
+#: aleksis/core/templates/templated_email/notification.email:15
+#: aleksis/core/templates/templated_email/notification.email:35
 #, fuzzy
 #| msgid "Edit school information"
 msgid "More information"
 msgstr "Muta informationes scolae"
 
-#: templates/templated_email/notification.email:18
+#: aleksis/core/templates/templated_email/notification.email:18
 #, python-format
 msgid ""
 "\n"
@@ -1795,12 +2518,7 @@ msgid ""
 "    "
 msgstr ""
 
-#: templates/templated_email/notification.email:22
-#: templates/templated_email/notification.email:46
-msgid "Your AlekSIS team"
-msgstr ""
-
-#: templates/templated_email/notification.email:40
+#: aleksis/core/templates/templated_email/notification.email:40
 #, python-format
 msgid ""
 "\n"
@@ -1808,24 +2526,41 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/two_factor/_base_focus.html:6
-#: templates/two_factor/core/otp_required.html:22
-#: templates/two_factor/core/setup.html:5
-#: templates/two_factor/profile/profile.html:87
-msgid "Enable Two-Factor Authentication"
+#: aleksis/core/templates/templated_email/person_changed.email:4
+#, python-format
+msgid "%(person)s changed their data!"
 msgstr ""
 
-#: templates/two_factor/_wizard_actions.html:6
-msgid "Cancel"
+#: aleksis/core/templates/templated_email/person_changed.email:10
+#, python-format
+msgid ""
+"\n"
+"   the person %(person)s recently changed the following fields:\n"
+" "
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:5
-#: templates/two_factor/core/backup_tokens.html:9
-#: templates/two_factor/profile/profile.html:46
+#: aleksis/core/templates/templated_email/person_changed.email:22
+#, python-format
+msgid ""
+"\n"
+"    the person %(person)s recently changed the following fields:\n"
+"  "
+msgstr ""
+
+#: aleksis/core/templates/two_factor/_base_focus.html:6
+#: aleksis/core/templates/two_factor/core/otp_required.html:22
+#: aleksis/core/templates/two_factor/core/setup.html:5
+#: aleksis/core/templates/two_factor/profile/profile.html:87
+msgid "Enable Two-Factor Authentication"
+msgstr ""
+
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:5
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:9
+#: aleksis/core/templates/two_factor/profile/profile.html:46
 msgid "Backup Tokens"
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:14
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:14
 msgid ""
 "\n"
 "        Backup tokens can be used when your primary and backup\n"
@@ -1836,115 +2571,129 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:33
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:33
 msgid ""
 "\n"
 "          Print these tokens and keep them somewhere safe.\n"
 "        "
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:39
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:39
 msgid "You don't have any backup codes yet."
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:45
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:45
 msgid "Back to Account Security"
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:49
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:49
 msgid "Generate Tokens"
 msgstr ""
 
-#: templates/two_factor/core/login.html:16
-msgid "You have no permission to view this page. Please login with an other account."
+#: aleksis/core/templates/two_factor/core/login.html:20
+msgid "Login with username and password"
 msgstr ""
 
-#: templates/two_factor/core/login.html:24
-msgid "Please login to see this page."
+#: aleksis/core/templates/two_factor/core/login.html:28
+msgid ""
+"You have no permission to view this page. Please login with an other\n"
+"                    account."
 msgstr ""
 
-#: templates/two_factor/core/login.html:30
-msgid "Login with username and password"
+#: aleksis/core/templates/two_factor/core/login.html:36
+msgid "Please login to see this page."
 msgstr ""
 
-#: templates/two_factor/core/login.html:40
+#: aleksis/core/templates/two_factor/core/login.html:46
 msgid ""
-"We are calling your phone right now, please enter the\n"
-"              digits you hear."
+"\n"
+"                        We are calling your phone right now, please enter the\n"
+"                        digits you hear.\n"
+"                      "
 msgstr ""
 
-#: templates/two_factor/core/login.html:43
+#: aleksis/core/templates/two_factor/core/login.html:51
 msgid ""
-"We sent you a text message, please enter the tokens we\n"
-"              sent."
+"\n"
+"                        We sent you a text message, please enter the tokens we\n"
+"                        sent.\n"
+"                      "
 msgstr ""
 
-#: templates/two_factor/core/login.html:46
+#: aleksis/core/templates/two_factor/core/login.html:56
 msgid ""
-"Please enter the tokens generated by your token\n"
-"              generator."
+"\n"
+"                        Please enter the tokens generated by your token\n"
+"                        generator.\n"
+"                      "
 msgstr ""
 
-#: templates/two_factor/core/login.html:50
+#: aleksis/core/templates/two_factor/core/login.html:62
 msgid ""
-"Use this form for entering backup tokens for logging in.\n"
-"            These tokens have been generated for you to print and keep safe. Please\n"
-"            enter one of these backup tokens to login to your account."
+"\n"
+"                      Use this form for entering backup tokens for logging in.\n"
+"                      These tokens have been generated for you to print and keep safe. Please\n"
+"                      enter one of these backup tokens to login to your account.\n"
+"                    "
 msgstr ""
 
-#: templates/two_factor/core/login.html:68
+#: aleksis/core/templates/two_factor/core/login.html:90
+msgid "Device currently not available?"
+msgstr ""
+
+#: aleksis/core/templates/two_factor/core/login.html:92
 msgid "Or, alternatively, use one of your backup phones:"
 msgstr ""
 
-#: templates/two_factor/core/login.html:78
+#: aleksis/core/templates/two_factor/core/login.html:102
 msgid "As a last resort, you can use a backup token:"
 msgstr ""
 
-#: templates/two_factor/core/login.html:81
+#: aleksis/core/templates/two_factor/core/login.html:105
 msgid "Use Backup Token"
 msgstr ""
 
-#: templates/two_factor/core/login.html:93
+#: aleksis/core/templates/two_factor/core/login.html:116
 msgid "Use alternative login options"
 msgstr ""
 
-#: templates/two_factor/core/otp_required.html:9
+#: aleksis/core/templates/two_factor/core/otp_required.html:9
 msgid "Permission Denied"
 msgstr ""
 
-#: templates/two_factor/core/otp_required.html:10
+#: aleksis/core/templates/two_factor/core/otp_required.html:10
 msgid ""
 "The page you requested, enforces users to verify using\n"
 "          two-factor authentication for security reasons. You need to enable these\n"
 "          security features in order to access this page."
 msgstr ""
 
-#: templates/two_factor/core/otp_required.html:14
+#: aleksis/core/templates/two_factor/core/otp_required.html:14
 msgid ""
 "Two-factor authentication is not enabled for your\n"
 "          account. Enable two-factor authentication for enhanced account\n"
 "          security."
 msgstr ""
 
-#: templates/two_factor/core/phone_register.html:5
-#: templates/two_factor/core/phone_register.html:9
+#: aleksis/core/templates/two_factor/core/phone_register.html:5
+#: aleksis/core/templates/two_factor/core/phone_register.html:9
 msgid "Add Backup Phone"
 msgstr ""
 
-#: templates/two_factor/core/phone_register.html:12
+#: aleksis/core/templates/two_factor/core/phone_register.html:12
 msgid ""
 "You'll be adding a backup phone number to your\n"
 "      account. This number will be used if your primary method of\n"
 "      registration is not available."
 msgstr ""
 
-#: templates/two_factor/core/phone_register.html:16
+#: aleksis/core/templates/two_factor/core/phone_register.html:16
 msgid ""
 "We've sent a token to your phone number. Please\n"
 "      enter the token you've received."
 msgstr ""
 
-#: templates/two_factor/core/setup.html:9
+#: aleksis/core/templates/two_factor/core/setup.html:9
 msgid ""
 "\n"
 "        You are about to take your account security to the\n"
@@ -1953,14 +2702,14 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:17
+#: aleksis/core/templates/two_factor/core/setup.html:17
 msgid ""
 "\n"
 "        Please select which authentication method you would like to use:\n"
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:23
+#: aleksis/core/templates/two_factor/core/setup.html:23
 msgid ""
 "\n"
 "        To start using a token generator, please use your\n"
@@ -1969,7 +2718,7 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:34
+#: aleksis/core/templates/two_factor/core/setup.html:34
 msgid ""
 "\n"
 "        Please enter the phone number you wish to receive the\n"
@@ -1977,7 +2726,7 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:41
+#: aleksis/core/templates/two_factor/core/setup.html:41
 msgid ""
 "\n"
 "        Please enter the phone number you wish to be called on.\n"
@@ -1985,21 +2734,21 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:50
+#: aleksis/core/templates/two_factor/core/setup.html:50
 msgid ""
 "\n"
 "            We are calling your phone right now, please enter the digits you hear.\n"
 "          "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:56
+#: aleksis/core/templates/two_factor/core/setup.html:56
 msgid ""
 "\n"
 "            We sent you a text message, please enter the tokens we sent.\n"
 "          "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:63
+#: aleksis/core/templates/two_factor/core/setup.html:63
 msgid ""
 "\n"
 "          We've encountered an issue with the selected authentication method. Please\n"
@@ -2009,7 +2758,7 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:73
+#: aleksis/core/templates/two_factor/core/setup.html:73
 msgid ""
 "\n"
 "        To identify and verify your YubiKey, please insert a\n"
@@ -2018,29 +2767,29 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:5
-#: templates/two_factor/core/setup_complete.html:9
+#: aleksis/core/templates/two_factor/core/setup_complete.html:5
+#: aleksis/core/templates/two_factor/core/setup_complete.html:9
 msgid "Two-Factor Authentication successfully enabled"
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:14
+#: aleksis/core/templates/two_factor/core/setup_complete.html:14
 msgid ""
 "\n"
 "        Congratulations, you've successfully enabled two-factor authentication.\n"
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:24
-#: templates/two_factor/core/setup_complete.html:44
+#: aleksis/core/templates/two_factor/core/setup_complete.html:24
+#: aleksis/core/templates/two_factor/core/setup_complete.html:44
 msgid "Back to Profile"
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:28
-#: templates/two_factor/core/setup_complete.html:48
+#: aleksis/core/templates/two_factor/core/setup_complete.html:28
+#: aleksis/core/templates/two_factor/core/setup_complete.html:48
 msgid "Generate backup codes"
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:34
+#: aleksis/core/templates/two_factor/core/setup_complete.html:34
 msgid ""
 "\n"
 "          However, it might happen that you don't have access to\n"
@@ -2049,65 +2798,65 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:52
-#: templates/two_factor/profile/profile.html:41
+#: aleksis/core/templates/two_factor/core/setup_complete.html:52
+#: aleksis/core/templates/two_factor/profile/profile.html:41
 msgid "Add Phone Number"
 msgstr ""
 
-#: templates/two_factor/profile/disable.html:5
-#: templates/two_factor/profile/disable.html:9
-#: templates/two_factor/profile/profile.html:63
-#: templates/two_factor/profile/profile.html:73
+#: aleksis/core/templates/two_factor/profile/disable.html:5
+#: aleksis/core/templates/two_factor/profile/disable.html:9
+#: aleksis/core/templates/two_factor/profile/profile.html:63
+#: aleksis/core/templates/two_factor/profile/profile.html:73
 msgid "Disable Two-Factor Authentication"
 msgstr ""
 
-#: templates/two_factor/profile/disable.html:12
+#: aleksis/core/templates/two_factor/profile/disable.html:12
 msgid "You are about to disable two-factor authentication. This weakens your account security, are you sure?"
 msgstr ""
 
-#: templates/two_factor/profile/disable.html:26
+#: aleksis/core/templates/two_factor/profile/disable.html:26
 msgid "Disable"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:5
-#: templates/two_factor/profile/profile.html:10
+#: aleksis/core/templates/two_factor/profile/profile.html:5
+#: aleksis/core/templates/two_factor/profile/profile.html:10
 msgid "Account Security"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:15
+#: aleksis/core/templates/two_factor/profile/profile.html:15
 msgid "Tokens will be generated by your token generator."
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:17
+#: aleksis/core/templates/two_factor/profile/profile.html:17
 #, python-format
 msgid "Primary method: %(primary)s"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:19
+#: aleksis/core/templates/two_factor/profile/profile.html:19
 msgid "Tokens will be generated by your YubiKey."
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:23
+#: aleksis/core/templates/two_factor/profile/profile.html:23
 msgid "Backup Phone Numbers"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:24
+#: aleksis/core/templates/two_factor/profile/profile.html:24
 msgid ""
 "If your primary method is not available, we are able to\n"
 "        send backup tokens to the phone numbers listed below."
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:33
+#: aleksis/core/templates/two_factor/profile/profile.html:33
 msgid "Unregister"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:48
+#: aleksis/core/templates/two_factor/profile/profile.html:48
 msgid ""
 "If you don't have any device with you, you can access\n"
 "        your account using backup tokens."
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:50
+#: aleksis/core/templates/two_factor/profile/profile.html:50
 #, python-format
 msgid ""
 "\n"
@@ -2120,11 +2869,11 @@ msgid_plural ""
 msgstr[0] ""
 msgstr[1] ""
 
-#: templates/two_factor/profile/profile.html:59
+#: aleksis/core/templates/two_factor/profile/profile.html:59
 msgid "Show Codes"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:65
+#: aleksis/core/templates/two_factor/profile/profile.html:65
 msgid ""
 "\n"
 "        However we strongly discourage you to do so, you can\n"
@@ -2132,7 +2881,7 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:78
+#: aleksis/core/templates/two_factor/profile/profile.html:78
 msgid ""
 "\n"
 "        Two-factor authentication is not enabled for your\n"
@@ -2141,117 +2890,167 @@ msgid ""
 "      "
 msgstr ""
 
-#: util/notifications.py:65
+#: aleksis/core/util/notifications.py:65
 msgid "E-Mail"
 msgstr ""
 
-#: util/notifications.py:66
+#: aleksis/core/util/notifications.py:66
 msgid "SMS"
 msgstr ""
 
-#: views.py:141
+#: aleksis/core/util/pdf.py:105
+msgid "Progress: Generate PDF file"
+msgstr ""
+
+#: aleksis/core/util/pdf.py:106
+msgid "Generating PDF file …"
+msgstr ""
+
+#: aleksis/core/util/pdf.py:107
+msgid "The PDF file has been generated successfully."
+msgstr ""
+
+#: aleksis/core/util/pdf.py:108
+msgid "There was a problem while generating the PDF file."
+msgstr ""
+
+#: aleksis/core/util/pdf.py:111
+msgid "Download PDF"
+msgstr ""
+
+#: aleksis/core/views.py:251
 msgid "The school term has been created."
 msgstr ""
 
-#: views.py:153
+#: aleksis/core/views.py:263
 msgid "The school term has been saved."
 msgstr ""
 
-#: views.py:298
+#: aleksis/core/views.py:387
 msgid "The child groups were successfully saved."
 msgstr ""
 
-#: views.py:336
+#: aleksis/core/views.py:406 aleksis/core/views.py:416
 msgid "The person has been saved."
 msgstr ""
 
-#: views.py:375
+#: aleksis/core/views.py:466
 msgid "The group has been saved."
 msgstr ""
 
-#: views.py:467
+#: aleksis/core/views.py:563
 msgid "The announcement has been saved."
 msgstr ""
 
-#: views.py:483
+#: aleksis/core/views.py:579
 msgid "The announcement has been deleted."
 msgstr ""
 
-#: views.py:562
+#: aleksis/core/views.py:663
 msgid "The preferences have been saved successfully."
 msgstr ""
 
-#: views.py:586
+#: aleksis/core/views.py:687
 msgid "The person has been deleted."
 msgstr ""
 
-#: views.py:600
+#: aleksis/core/views.py:701
 msgid "The group has been deleted."
 msgstr ""
 
-#: views.py:632
+#: aleksis/core/views.py:733
 msgid "The additional_field has been saved."
 msgstr ""
 
-#: views.py:666
+#: aleksis/core/views.py:767
 msgid "The additional field has been deleted."
 msgstr ""
 
-#: views.py:691
+#: aleksis/core/views.py:792
 msgid "The group type has been saved."
 msgstr ""
 
-#: views.py:721
+#: aleksis/core/views.py:822
 msgid "The group type has been deleted."
 msgstr ""
 
-#: views.py:749
-msgid "The data check has been started. Please note that it may take a while before you are able to fetch the data on this page."
+#: aleksis/core/views.py:855
+msgid "Progress: Run data checks"
+msgstr ""
+
+#: aleksis/core/views.py:856
+#, fuzzy
+#| msgid "System status"
+msgid "Run data checks …"
+msgstr "Status systemae"
+
+#: aleksis/core/views.py:857
+msgid "The data checks were run successfully."
 msgstr ""
 
-#: views.py:754
-msgid "The data check has finished."
+#: aleksis/core/views.py:858
+msgid "There was a problem while running data checks."
 msgstr ""
 
-#: views.py:769
+#: aleksis/core/views.py:874
 #, python-brace-format
 msgid "The solve option '{solve_option_obj.verbose_name}' "
 msgstr ""
 
-#: views.py:811
+#: aleksis/core/views.py:916
 msgid "The dashboard widget has been saved."
 msgstr ""
 
-#: views.py:841
+#: aleksis/core/views.py:946
 msgid "The dashboard widget has been created."
 msgstr ""
 
-#: views.py:851
+#: aleksis/core/views.py:956
 msgid "The dashboard widget has been deleted."
 msgstr ""
 
-#: views.py:914
+#: aleksis/core/views.py:1023
 msgid "Your dashboard configuration has been saved successfully."
 msgstr ""
 
-#: views.py:916
+#: aleksis/core/views.py:1025
 msgid "The configuration of the default dashboard has been saved successfully."
 msgstr ""
 
+#: aleksis/core/views.py:1153
+msgid "The third-party account could not be disconnected because it is the only login method available."
+msgstr ""
+
+#: aleksis/core/views.py:1160
+msgid "The third-party account has been successfully disconnected."
+msgstr ""
+
+#~ msgid "Persons and accounts"
+#~ msgstr "Personae et computi"
+
+#, fuzzy
+#~| msgid "Persons and accounts"
+#~ msgid "Can link persons to accounts"
+#~ msgstr "Personae et computi"
+
+#, fuzzy
+#~| msgid "Persons and accounts"
+#~ msgid "Link persons to accounts"
+#~ msgstr "Personae et computi"
+
+#, fuzzy
+#~| msgid "Impersonation"
+#~ msgid "Impersonate user"
+#~ msgstr "Simulare aliquem"
+
 #~ msgid "School logo"
 #~ msgstr "Imago scolae"
 
 #~ msgid "Manage school"
 #~ msgstr "Administra scholam"
 
-#~ msgid "Edit school information"
-#~ msgstr "Muta informationes scolae"
-
 #~ msgid "Official name"
 #~ msgstr "Officialis nomen"
 
-#~ msgid "School"
-#~ msgstr "Scola"
-
 #~ msgid "Schools"
 #~ msgstr "Scholae"
diff --git a/aleksis/core/locale/la/LC_MESSAGES/djangojs.po b/aleksis/core/locale/la/LC_MESSAGES/djangojs.po
index ea24fc3ff92afc211898fb8b426dccb97fce56ed..dc6d028e5442a3666f35a7cb46e39e5c01cdd3f9 100644
--- a/aleksis/core/locale/la/LC_MESSAGES/djangojs.po
+++ b/aleksis/core/locale/la/LC_MESSAGES/djangojs.po
@@ -8,7 +8,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2021-01-11 21:30+0100\n"
+"POT-Creation-Date: 2021-10-28 16:18+0200\n"
 "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
@@ -17,18 +17,18 @@ msgstr ""
 "Content-Type: text/plain; charset=UTF-8\n"
 "Content-Transfer-Encoding: 8bit\n"
 
-#: static/js/main.js:15
+#: aleksis/core/static/js/main.js:15
 msgid "Today"
 msgstr ""
 
-#: static/js/main.js:16
+#: aleksis/core/static/js/main.js:16
 msgid "Cancel"
 msgstr ""
 
-#: static/js/main.js:17
+#: aleksis/core/static/js/main.js:17
 msgid "OK"
 msgstr ""
 
-#: static/js/main.js:118
+#: aleksis/core/static/js/main.js:127
 msgid "This page may contain outdated information since there is no internet connection."
 msgstr ""
diff --git a/aleksis/core/locale/nb_NO/LC_MESSAGES/django.po b/aleksis/core/locale/nb_NO/LC_MESSAGES/django.po
index 6b4be05da6b0a53f60a6ffa060739c419ea4af3a..ae2f8d194fcacd2bae0d16bb3350dfbaa978aaee 100644
--- a/aleksis/core/locale/nb_NO/LC_MESSAGES/django.po
+++ b/aleksis/core/locale/nb_NO/LC_MESSAGES/django.po
@@ -8,7 +8,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: AlekSIS (School Information System) 0.1\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2021-01-11 21:30+0100\n"
+"POT-Creation-Date: 2021-10-28 16:18+0200\n"
 "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
@@ -17,822 +17,1113 @@ msgstr ""
 "Content-Type: text/plain; charset=UTF-8\n"
 "Content-Transfer-Encoding: 8bit\n"
 
-#: data_checks.py:53
+#: aleksis/core/apps.py:150
+msgid "OpenID Connect scope"
+msgstr ""
+
+#: aleksis/core/apps.py:151
+msgid "Given name, family name, link to profile and picture if existing."
+msgstr ""
+
+#: aleksis/core/apps.py:152
+msgid "Full home postal address"
+msgstr ""
+
+#: aleksis/core/apps.py:153
+msgid "Email address"
+msgstr ""
+
+#: aleksis/core/apps.py:154
+msgid "Home and mobile phone"
+msgstr ""
+
+#: aleksis/core/data_checks.py:55
 msgid "Ignore problem"
 msgstr ""
 
-#: data_checks.py:174
+#: aleksis/core/data_checks.py:184
 #, python-brace-format
 msgid "Solve option '{solve_option_obj.verbose_name}' "
 msgstr ""
 
-#: filters.py:37 templates/core/base.html:77 templates/core/group/list.html:20
-#: templates/core/person/list.html:24 templates/search/search.html:7
-#: templates/search/search.html:22
-msgid "Search"
+#: aleksis/core/data_checks.py:291
+msgid "Deactivate DashboardWidget"
 msgstr ""
 
-#: filters.py:53
-msgid "Search by name"
+#: aleksis/core/data_checks.py:303
+msgid "Ensure that there are no broken DashboardWidgets."
 msgstr ""
 
-#: filters.py:65
-msgid "Search by contact details"
+#: aleksis/core/data_checks.py:304
+msgid "The DashboardWidget was reported broken automatically."
 msgstr ""
 
-#: forms.py:54
-msgid "You cannot set a new username when also selecting an existing user."
+#: aleksis/core/filters.py:37 aleksis/core/templates/core/base.html:83
+#: aleksis/core/templates/core/base.html:84
+#: aleksis/core/templates/core/group/list.html:20
+#: aleksis/core/templates/core/person/list.html:24
+#: aleksis/core/templates/search/search.html:7
+#: aleksis/core/templates/search/search.html:22
+msgid "Search"
 msgstr ""
 
-#: forms.py:58
-msgid "This username is already in use."
+#: aleksis/core/filters.py:53
+msgid "Search by name"
 msgstr ""
 
-#: forms.py:82
+#: aleksis/core/filters.py:65
+msgid "Search by contact details"
+msgstr ""
+
+#: aleksis/core/forms.py:41 aleksis/core/forms.py:387
 msgid "Base data"
 msgstr ""
 
-#: forms.py:88
+#: aleksis/core/forms.py:47
 msgid "Address"
 msgstr ""
 
-#: forms.py:89
+#: aleksis/core/forms.py:48
 msgid "Contact data"
 msgstr ""
 
-#: forms.py:91
+#: aleksis/core/forms.py:50
 msgid "Advanced personal data"
 msgstr ""
 
-#: forms.py:134
+#: aleksis/core/forms.py:93
 msgid "New user"
 msgstr ""
 
-#: forms.py:134
+#: aleksis/core/forms.py:93
 msgid "Create a new account"
 msgstr ""
 
-#: forms.py:146 models.py:102
+#: aleksis/core/forms.py:124
+msgid "You cannot set a new username when also selecting an existing user."
+msgstr ""
+
+#: aleksis/core/forms.py:128
+msgid "This username is already in use."
+msgstr ""
+
+#: aleksis/core/forms.py:145 aleksis/core/models.py:117
 msgid "School term"
 msgstr ""
 
-#: forms.py:147
+#: aleksis/core/forms.py:146
 msgid "Common data"
 msgstr ""
 
-#: forms.py:148 forms.py:197 menus.py:169 models.py:116
-#: templates/core/person/list.html:8 templates/core/person/list.html:9
+#: aleksis/core/forms.py:147 aleksis/core/forms.py:196
+#: aleksis/core/menus.py:238 aleksis/core/models.py:140
+#: aleksis/core/templates/core/person/list.html:8
+#: aleksis/core/templates/core/person/list.html:9
 msgid "Persons"
 msgstr ""
 
-#: forms.py:149
+#: aleksis/core/forms.py:148
 msgid "Additional data"
 msgstr ""
 
-#: forms.py:189 forms.py:192 models.py:45
+#: aleksis/core/forms.py:188 aleksis/core/forms.py:191
+#: aleksis/core/models.py:60
 msgid "Date"
 msgstr ""
 
-#: forms.py:190 forms.py:193 models.py:53
+#: aleksis/core/forms.py:189 aleksis/core/forms.py:192
+#: aleksis/core/models.py:68
 msgid "Time"
 msgstr ""
 
-#: forms.py:210 menus.py:177 models.py:338 templates/core/group/list.html:8
-#: templates/core/group/list.html:9 templates/core/person/full.html:144
+#: aleksis/core/forms.py:209 aleksis/core/menus.py:249
+#: aleksis/core/models.py:398 aleksis/core/templates/core/group/list.html:8
+#: aleksis/core/templates/core/group/list.html:9
+#: aleksis/core/templates/core/person/full.html:144
 msgid "Groups"
 msgstr ""
 
-#: forms.py:220
+#: aleksis/core/forms.py:219
 msgid "From when until when should the announcement be displayed?"
 msgstr ""
 
-#: forms.py:223
+#: aleksis/core/forms.py:222
 msgid "Who should see the announcement?"
 msgstr ""
 
-#: forms.py:224
+#: aleksis/core/forms.py:223
 msgid "Write your announcement:"
 msgstr ""
 
-#: forms.py:263
+#: aleksis/core/forms.py:262
 msgid "You are not allowed to create announcements which are only valid in the past."
 msgstr ""
 
-#: forms.py:267
+#: aleksis/core/forms.py:266
 msgid "The from date and time must be earlier then the until date and time."
 msgstr ""
 
-#: forms.py:276
+#: aleksis/core/forms.py:275
 msgid "You need at least one recipient."
 msgstr ""
 
-#: health_checks.py:15
+#: aleksis/core/forms.py:389
+msgid "Account data"
+msgstr ""
+
+#: aleksis/core/forms.py:391
+msgid "Consents"
+msgstr ""
+
+#: aleksis/core/forms.py:396
+msgid "Password"
+msgstr ""
+
+#: aleksis/core/forms.py:402
+msgid "Password (again)"
+msgstr ""
+
+#: aleksis/core/forms.py:411
+#, python-brace-format
+msgid "I have read the <a href='{privacy_policy}'>Privacy policy</a> and agree with them."
+msgstr ""
+
+#: aleksis/core/forms.py:435
+msgid "You must type the same password each time."
+msgstr ""
+
+#: aleksis/core/forms.py:580
+msgid "No valid selection."
+msgstr ""
+
+#: aleksis/core/health_checks.py:21
 msgid "There are unresolved data problems."
 msgstr ""
 
-#: menus.py:7 templates/two_factor/core/login.html:6
-#: templates/two_factor/core/login.html:10
-#: templates/two_factor/core/login.html:86
+#: aleksis/core/health_checks.py:38
+msgid "The backup folder doesn't exist."
+msgstr ""
+
+#: aleksis/core/health_checks.py:47
+#, python-brace-format
+msgid "Last backup {time_gone_since_backup}!"
+msgstr ""
+
+#: aleksis/core/health_checks.py:49
+msgid "No backup found!"
+msgstr ""
+
+#: aleksis/core/health_checks.py:76
+msgid "No backup result found!"
+msgstr ""
+
+#: aleksis/core/health_checks.py:78
+#, python-brace-format
+msgid "{task.status} - {task.result}"
+msgstr ""
+
+#: aleksis/core/menus.py:9 aleksis/core/templates/two_factor/core/login.html:6
+#: aleksis/core/templates/two_factor/core/login.html:22
+#: aleksis/core/templates/two_factor/core/login.html:76
 msgid "Login"
 msgstr ""
 
-#: menus.py:13
+#: aleksis/core/menus.py:15 aleksis/core/templates/account/signup.html:20
+#: aleksis/core/templates/socialaccount/signup.html:23
+msgid "Sign up"
+msgstr ""
+
+#: aleksis/core/menus.py:24
 msgid "Dashboard"
 msgstr ""
 
-#: menus.py:19
+#: aleksis/core/menus.py:32 aleksis/core/models.py:605
+#: aleksis/core/preferences.py:26
+#: aleksis/core/templates/core/notifications.html:4
+#: aleksis/core/templates/core/notifications.html:5
+msgid "Notifications"
+msgstr ""
+
+#: aleksis/core/menus.py:41
 msgid "Account"
 msgstr ""
 
-#: menus.py:26
+#: aleksis/core/menus.py:48
 msgid "Stop impersonation"
 msgstr ""
 
-#: menus.py:35 templates/core/base.html:56
+#: aleksis/core/menus.py:57 aleksis/core/templates/core/base.html:62
 msgid "Logout"
 msgstr ""
 
-#: menus.py:41
+#: aleksis/core/menus.py:63
 msgid "2FA"
 msgstr ""
 
-#: menus.py:47
+#: aleksis/core/menus.py:69
+#: aleksis/core/templates/account/password_change.html:5
+#: aleksis/core/templates/account/password_change.html:6
+#: aleksis/core/templates/account/password_change.html:19
+#: aleksis/core/templates/account/password_reset_from_key.html:5
+#: aleksis/core/templates/account/password_reset_from_key.html:42
+#: aleksis/core/templates/account/password_reset_from_key.html:46
+#: aleksis/core/templates/account/password_reset_from_key_done.html:5
+#: aleksis/core/templates/account/password_reset_from_key_done.html:6
+msgid "Change password"
+msgstr ""
+
+#: aleksis/core/menus.py:81
 msgid "Me"
 msgstr ""
 
-#: menus.py:56 templates/dynamic_preferences/form.html:5
+#: aleksis/core/menus.py:90
+#: aleksis/core/templates/dynamic_preferences/form.html:5
 msgid "Preferences"
 msgstr ""
 
-#: menus.py:67
+#: aleksis/core/menus.py:99
+msgid "Third-party accounts"
+msgstr ""
+
+#: aleksis/core/menus.py:108
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:5
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:6
+msgid "Authorized applications"
+msgstr ""
+
+#: aleksis/core/menus.py:119
 msgid "Admin"
 msgstr ""
 
-#: menus.py:75 models.py:602 templates/core/announcement/list.html:7
-#: templates/core/announcement/list.html:8
+#: aleksis/core/menus.py:127 aleksis/core/models.py:704
+#: aleksis/core/templates/core/announcement/list.html:7
+#: aleksis/core/templates/core/announcement/list.html:8
 msgid "Announcements"
 msgstr ""
 
-#: menus.py:86 models.py:103 templates/core/school_term/list.html:8
-#: templates/core/school_term/list.html:9
+#: aleksis/core/menus.py:138 aleksis/core/models.py:118
+#: aleksis/core/templates/core/school_term/list.html:8
+#: aleksis/core/templates/core/school_term/list.html:9
 msgid "School terms"
 msgstr ""
 
-#: menus.py:97 templates/core/dashboard_widget/list.html:8
-#: templates/core/dashboard_widget/list.html:9
+#: aleksis/core/menus.py:149
+#: aleksis/core/templates/core/dashboard_widget/list.html:8
+#: aleksis/core/templates/core/dashboard_widget/list.html:9
 msgid "Dashboard widgets"
 msgstr ""
 
-#: menus.py:108 templates/core/management/data_management.html:6
-#: templates/core/management/data_management.html:7
+#: aleksis/core/menus.py:160
+#: aleksis/core/templates/core/management/data_management.html:6
+#: aleksis/core/templates/core/management/data_management.html:7
 msgid "Data management"
 msgstr ""
 
-#: menus.py:116 templates/core/pages/system_status.html:5
-#: templates/core/pages/system_status.html:7
+#: aleksis/core/menus.py:171
+#: aleksis/core/templates/core/pages/system_status.html:5
+#: aleksis/core/templates/core/pages/system_status.html:7
 msgid "System status"
 msgstr ""
 
-#: menus.py:127
+#: aleksis/core/menus.py:182
 msgid "Impersonation"
 msgstr ""
 
-#: menus.py:135
+#: aleksis/core/menus.py:193
 msgid "Configuration"
 msgstr ""
 
-#: menus.py:146 templates/core/data_check/list.html:9
-#: templates/core/data_check/list.html:10
+#: aleksis/core/menus.py:204 aleksis/core/templates/core/data_check/list.html:9
+#: aleksis/core/templates/core/data_check/list.html:10
 msgid "Data checks"
 msgstr ""
 
-#: menus.py:152
+#: aleksis/core/menus.py:210
 msgid "Backend Admin"
 msgstr ""
 
-#: menus.py:160
-msgid "People"
+#: aleksis/core/menus.py:216
+#: aleksis/core/templates/oauth2_provider/application_detail.html:5
+#: aleksis/core/templates/oauth2_provider/application_list.html:5
+msgid "OAuth2 Applications"
 msgstr ""
 
-#: menus.py:185 models.py:812 templates/core/group_type/list.html:8
-#: templates/core/group_type/list.html:9
-msgid "Group types"
+#: aleksis/core/menus.py:229
+msgid "People"
 msgstr ""
 
-#: menus.py:196
-msgid "Persons and accounts"
+#: aleksis/core/menus.py:260 aleksis/core/models.py:958
+#: aleksis/core/templates/core/group_type/list.html:8
+#: aleksis/core/templates/core/group_type/list.html:9
+msgid "Group types"
 msgstr ""
 
-#: menus.py:207
+#: aleksis/core/menus.py:271
 msgid "Groups and child groups"
 msgstr ""
 
-#: menus.py:218 models.py:385 templates/core/additional_field/list.html:8
-#: templates/core/additional_field/list.html:9
+#: aleksis/core/menus.py:282 aleksis/core/models.py:446
+#: aleksis/core/templates/core/additional_field/list.html:8
+#: aleksis/core/templates/core/additional_field/list.html:9
 msgid "Additional fields"
 msgstr ""
 
-#: menus.py:233 templates/core/group/child_groups.html:7
-#: templates/core/group/child_groups.html:9
+#: aleksis/core/menus.py:297
+#: aleksis/core/templates/core/group/child_groups.html:7
+#: aleksis/core/templates/core/group/child_groups.html:9
 msgid "Assign child groups to groups"
 msgstr ""
 
-#: mixins.py:384
+#: aleksis/core/mixins.py:498
 msgid "Linked school term"
 msgstr ""
 
-#: models.py:43
+#: aleksis/core/models.py:58
 msgid "Boolean (Yes/No)"
 msgstr ""
 
-#: models.py:44
+#: aleksis/core/models.py:59
 msgid "Text (one line)"
 msgstr ""
 
-#: models.py:46
+#: aleksis/core/models.py:61
 msgid "Date and time"
 msgstr ""
 
-#: models.py:47
+#: aleksis/core/models.py:62
 msgid "Decimal number"
 msgstr ""
 
-#: models.py:48 models.py:157
+#: aleksis/core/models.py:63 aleksis/core/models.py:186
 msgid "E-mail address"
 msgstr ""
 
-#: models.py:49
+#: aleksis/core/models.py:64
 msgid "Integer"
 msgstr ""
 
-#: models.py:50
+#: aleksis/core/models.py:65
 msgid "IP address"
 msgstr ""
 
-#: models.py:51
+#: aleksis/core/models.py:66
 msgid "Boolean or empty (Yes/No/Neither)"
 msgstr ""
 
-#: models.py:52
+#: aleksis/core/models.py:67
 msgid "Text (multi-line)"
 msgstr ""
 
-#: models.py:54
+#: aleksis/core/models.py:69
 msgid "URL / Link"
 msgstr ""
 
-#: models.py:66 models.py:785
+#: aleksis/core/models.py:81 aleksis/core/models.py:927
 msgid "Name"
 msgstr ""
 
-#: models.py:68
+#: aleksis/core/models.py:83
 msgid "Start date"
 msgstr ""
 
-#: models.py:69
+#: aleksis/core/models.py:84
 msgid "End date"
 msgstr ""
 
-#: models.py:88
+#: aleksis/core/models.py:103
 msgid "The start date must be earlier than the end date."
 msgstr ""
 
-#: models.py:95
+#: aleksis/core/models.py:110
 msgid "There is already a school term for this time or a part of this time."
 msgstr ""
 
-#: models.py:115 models.py:744 templates/core/person/accounts.html:41
+#: aleksis/core/models.py:139 aleksis/core/models.py:876
 msgid "Person"
 msgstr ""
 
-#: models.py:118
+#: aleksis/core/models.py:142
 msgid "Can view address"
 msgstr ""
 
-#: models.py:119
+#: aleksis/core/models.py:143
 msgid "Can view contact details"
 msgstr ""
 
-#: models.py:120
+#: aleksis/core/models.py:144
 msgid "Can view photo"
 msgstr ""
 
-#: models.py:121
+#: aleksis/core/models.py:145
 msgid "Can view persons groups"
 msgstr ""
 
-#: models.py:122
+#: aleksis/core/models.py:146
 msgid "Can view personal details"
 msgstr ""
 
-#: models.py:127
+#: aleksis/core/models.py:156
 msgid "female"
 msgstr ""
 
-#: models.py:127
+#: aleksis/core/models.py:156
 msgid "male"
 msgstr ""
 
-#: models.py:135
+#: aleksis/core/models.py:164
 msgid "Linked user"
 msgstr ""
 
-#: models.py:137
+#: aleksis/core/models.py:166
 msgid "Is person active?"
 msgstr ""
 
-#: models.py:139
+#: aleksis/core/models.py:168
 msgid "First name"
 msgstr ""
 
-#: models.py:140
+#: aleksis/core/models.py:169
 msgid "Last name"
 msgstr ""
 
-#: models.py:142
+#: aleksis/core/models.py:171
 msgid "Additional name(s)"
 msgstr ""
 
-#: models.py:146 models.py:354
+#: aleksis/core/models.py:175 aleksis/core/models.py:415
 msgid "Short name"
 msgstr ""
 
-#: models.py:149
+#: aleksis/core/models.py:178
 msgid "Street"
 msgstr ""
 
-#: models.py:150
+#: aleksis/core/models.py:179
 msgid "Street number"
 msgstr ""
 
-#: models.py:151
+#: aleksis/core/models.py:180
 msgid "Postal code"
 msgstr ""
 
-#: models.py:152
+#: aleksis/core/models.py:181
 msgid "Place"
 msgstr ""
 
-#: models.py:154
+#: aleksis/core/models.py:183
 msgid "Home phone"
 msgstr ""
 
-#: models.py:155
+#: aleksis/core/models.py:184
 msgid "Mobile phone"
 msgstr ""
 
-#: models.py:159
+#: aleksis/core/models.py:188
 msgid "Date of birth"
 msgstr ""
 
-#: models.py:160
+#: aleksis/core/models.py:189
 msgid "Sex"
 msgstr ""
 
-#: models.py:162
+#: aleksis/core/models.py:191
 msgid "Photo"
 msgstr ""
 
-#: models.py:166 templates/core/person/full.html:137
+#: aleksis/core/models.py:195 aleksis/core/templates/core/person/full.html:137
 msgid "Guardians / Parents"
 msgstr ""
 
-#: models.py:173
+#: aleksis/core/models.py:202
 msgid "Primary group"
 msgstr ""
 
-#: models.py:176 models.py:461 models.py:485 models.py:570 models.py:805
-#: templates/core/person/full.html:120
+#: aleksis/core/models.py:205 aleksis/core/models.py:563
+#: aleksis/core/models.py:587 aleksis/core/models.py:672
+#: aleksis/core/models.py:951 aleksis/core/templates/core/person/full.html:120
 msgid "Description"
 msgstr ""
 
-#: models.py:313
+#: aleksis/core/models.py:370
 msgid "Title of field"
 msgstr ""
 
-#: models.py:315
+#: aleksis/core/models.py:372
 msgid "Type of field"
 msgstr ""
 
-#: models.py:322
+#: aleksis/core/models.py:379
 msgid "Addtitional field for groups"
 msgstr ""
 
-#: models.py:323
+#: aleksis/core/models.py:380
 msgid "Addtitional fields for groups"
 msgstr ""
 
-#: models.py:337
+#: aleksis/core/models.py:397
 msgid "Group"
 msgstr ""
 
-#: models.py:340
+#: aleksis/core/models.py:400
 msgid "Can assign child groups to groups"
 msgstr ""
 
-#: models.py:341
+#: aleksis/core/models.py:401
 msgid "Can view statistics about group."
 msgstr ""
 
-#: models.py:352
+#: aleksis/core/models.py:413
 msgid "Long name"
 msgstr ""
 
-#: models.py:362 templates/core/group/full.html:85
+#: aleksis/core/models.py:423 aleksis/core/templates/core/group/full.html:85
 msgid "Members"
 msgstr ""
 
-#: models.py:365 templates/core/group/full.html:82
+#: aleksis/core/models.py:426 aleksis/core/templates/core/group/full.html:82
 msgid "Owners"
 msgstr ""
 
-#: models.py:372 templates/core/group/full.html:55
+#: aleksis/core/models.py:433 aleksis/core/templates/core/group/full.html:55
 msgid "Parent groups"
 msgstr ""
 
-#: models.py:380
+#: aleksis/core/models.py:441
 msgid "Type of group"
 msgstr ""
 
-#: models.py:457
+#: aleksis/core/models.py:559
 msgid "User"
 msgstr ""
 
-#: models.py:460 models.py:484 models.py:569
-#: templates/core/announcement/list.html:18
+#: aleksis/core/models.py:562 aleksis/core/models.py:586
+#: aleksis/core/models.py:671
+#: aleksis/core/templates/core/announcement/list.html:18
 msgid "Title"
 msgstr ""
 
-#: models.py:463
+#: aleksis/core/models.py:565
 msgid "Application"
 msgstr ""
 
-#: models.py:469
+#: aleksis/core/models.py:571
 msgid "Activity"
 msgstr ""
 
-#: models.py:470
+#: aleksis/core/models.py:572
 msgid "Activities"
 msgstr ""
 
-#: models.py:476
+#: aleksis/core/models.py:578
 msgid "Sender"
 msgstr ""
 
-#: models.py:481
+#: aleksis/core/models.py:583
 msgid "Recipient"
 msgstr ""
 
-#: models.py:486 models.py:786
+#: aleksis/core/models.py:588 aleksis/core/models.py:928
 msgid "Link"
 msgstr ""
 
-#: models.py:488
+#: aleksis/core/models.py:590
 msgid "Read"
 msgstr ""
 
-#: models.py:489
+#: aleksis/core/models.py:591
 msgid "Sent"
 msgstr ""
 
-#: models.py:502
+#: aleksis/core/models.py:604
 msgid "Notification"
 msgstr ""
 
-#: models.py:503
-msgid "Notifications"
-msgstr ""
-
-#: models.py:571
+#: aleksis/core/models.py:673
 msgid "Link to detailed view"
 msgstr ""
 
-#: models.py:574
+#: aleksis/core/models.py:676
 msgid "Date and time from when to show"
 msgstr ""
 
-#: models.py:577
+#: aleksis/core/models.py:679
 msgid "Date and time until when to show"
 msgstr ""
 
-#: models.py:601
+#: aleksis/core/models.py:703
 msgid "Announcement"
 msgstr ""
 
-#: models.py:639
+#: aleksis/core/models.py:741
 msgid "Announcement recipient"
 msgstr ""
 
-#: models.py:640
+#: aleksis/core/models.py:742
 msgid "Announcement recipients"
 msgstr ""
 
-#: models.py:690
+#: aleksis/core/models.py:797
 msgid "Widget Title"
 msgstr ""
 
-#: models.py:691
+#: aleksis/core/models.py:798
 msgid "Activate Widget"
 msgstr ""
 
-#: models.py:694
+#: aleksis/core/models.py:799
+msgid "Widget is broken"
+msgstr ""
+
+#: aleksis/core/models.py:802
 msgid "Size on mobile devices"
 msgstr ""
 
-#: models.py:695
+#: aleksis/core/models.py:803
 msgid "<= 600 px, 12 columns"
 msgstr ""
 
-#: models.py:700
+#: aleksis/core/models.py:808
 msgid "Size on tablet devices"
 msgstr ""
 
-#: models.py:701
+#: aleksis/core/models.py:809
 msgid "> 600 px, 12 columns"
 msgstr ""
 
-#: models.py:706
+#: aleksis/core/models.py:814
 msgid "Size on desktop devices"
 msgstr ""
 
-#: models.py:707
+#: aleksis/core/models.py:815
 msgid "> 992 px, 12 columns"
 msgstr ""
 
-#: models.py:712
+#: aleksis/core/models.py:820
 msgid "Size on large desktop devices"
 msgstr ""
 
-#: models.py:713
+#: aleksis/core/models.py:821
 msgid "> 1200 px>, 12 columns"
 msgstr ""
 
-#: models.py:734
+#: aleksis/core/models.py:852
 msgid "Can edit default dashboard"
 msgstr ""
 
-#: models.py:735
+#: aleksis/core/models.py:853
 msgid "Dashboard Widget"
 msgstr ""
 
-#: models.py:736
+#: aleksis/core/models.py:854
 msgid "Dashboard Widgets"
 msgstr ""
 
-#: models.py:741
+#: aleksis/core/models.py:860
+msgid "URL"
+msgstr ""
+
+#: aleksis/core/models.py:861
+msgid "Icon URL"
+msgstr ""
+
+#: aleksis/core/models.py:867
+msgid "External link widget"
+msgstr ""
+
+#: aleksis/core/models.py:868
+msgid "External link widgets"
+msgstr ""
+
+#: aleksis/core/models.py:873
 msgid "Dashboard widget"
 msgstr ""
 
-#: models.py:746
+#: aleksis/core/models.py:878
 msgid "Order"
 msgstr ""
 
-#: models.py:747
+#: aleksis/core/models.py:879
 msgid "Part of the default dashboard"
 msgstr ""
 
-#: models.py:755
+#: aleksis/core/models.py:894
 msgid "Dashboard widget order"
 msgstr ""
 
-#: models.py:756
+#: aleksis/core/models.py:895
 msgid "Dashboard widget orders"
 msgstr ""
 
-#: models.py:762
+#: aleksis/core/models.py:901
 msgid "Menu ID"
 msgstr ""
 
-#: models.py:775
+#: aleksis/core/models.py:914
 msgid "Custom menu"
 msgstr ""
 
-#: models.py:776
+#: aleksis/core/models.py:915
 msgid "Custom menus"
 msgstr ""
 
-#: models.py:783
+#: aleksis/core/models.py:925
 msgid "Menu"
 msgstr ""
 
-#: models.py:787
+#: aleksis/core/models.py:929
 msgid "Icon"
 msgstr ""
 
-#: models.py:793
+#: aleksis/core/models.py:935
 msgid "Custom menu item"
 msgstr ""
 
-#: models.py:794
+#: aleksis/core/models.py:936
 msgid "Custom menu items"
 msgstr ""
 
-#: models.py:804
+#: aleksis/core/models.py:950
 msgid "Title of type"
 msgstr ""
 
-#: models.py:811 templates/core/group/full.html:47
+#: aleksis/core/models.py:957 aleksis/core/templates/core/group/full.html:47
 msgid "Group type"
 msgstr ""
 
-#: models.py:821
+#: aleksis/core/models.py:971
 msgid "Can view system status"
 msgstr ""
 
-#: models.py:822
-msgid "Can link persons to accounts"
-msgstr ""
-
-#: models.py:823
+#: aleksis/core/models.py:972
 msgid "Can manage data"
 msgstr ""
 
-#: models.py:824
+#: aleksis/core/models.py:973
 msgid "Can impersonate"
 msgstr ""
 
-#: models.py:825
+#: aleksis/core/models.py:974
 msgid "Can use search"
 msgstr ""
 
-#: models.py:826
+#: aleksis/core/models.py:975
 msgid "Can change site preferences"
 msgstr ""
 
-#: models.py:827
+#: aleksis/core/models.py:976
 msgid "Can change person preferences"
 msgstr ""
 
-#: models.py:828
+#: aleksis/core/models.py:977
 msgid "Can change group preferences"
 msgstr ""
 
-#: models.py:864
+#: aleksis/core/models.py:978
+msgid "Can add oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:979
+msgid "Can list oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:980
+msgid "Can view oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:981
+msgid "Can update oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:982
+msgid "Can delete oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:983
+msgid "Can test PDF generation"
+msgstr ""
+
+#: aleksis/core/models.py:1019
 msgid "Related data check task"
 msgstr ""
 
-#: models.py:872
+#: aleksis/core/models.py:1027
 msgid "Issue solved"
 msgstr ""
 
-#: models.py:873
+#: aleksis/core/models.py:1028
 msgid "Notification sent"
 msgstr ""
 
-#: models.py:886
+#: aleksis/core/models.py:1041
 msgid "Data check result"
 msgstr ""
 
-#: models.py:887
+#: aleksis/core/models.py:1042
 msgid "Data check results"
 msgstr ""
 
-#: models.py:889
+#: aleksis/core/models.py:1044
 msgid "Can run data checks"
 msgstr ""
 
-#: models.py:890
+#: aleksis/core/models.py:1045
 msgid "Can solve data check problems"
 msgstr ""
 
-#: preferences.py:27
+#: aleksis/core/models.py:1060
+msgid "Owner"
+msgstr ""
+
+#: aleksis/core/models.py:1064
+msgid "File expires at"
+msgstr ""
+
+#: aleksis/core/models.py:1066
+msgid "Generated HTML file"
+msgstr ""
+
+#: aleksis/core/models.py:1068
+msgid "Generated PDF file"
+msgstr ""
+
+#: aleksis/core/models.py:1075
+msgid "PDF file"
+msgstr ""
+
+#: aleksis/core/models.py:1076
+msgid "PDF files"
+msgstr ""
+
+#: aleksis/core/models.py:1081
+msgid "Task result"
+msgstr ""
+
+#: aleksis/core/models.py:1084
+msgid "Task user"
+msgstr ""
+
+#: aleksis/core/models.py:1096
+msgid "Task user assignment"
+msgstr ""
+
+#: aleksis/core/models.py:1097
+msgid "Task user assignments"
+msgstr ""
+
+#: aleksis/core/preferences.py:22
+msgid "General"
+msgstr ""
+
+#: aleksis/core/preferences.py:23
+msgid "School"
+msgstr ""
+
+#: aleksis/core/preferences.py:24
+msgid "Theme"
+msgstr ""
+
+#: aleksis/core/preferences.py:25
+msgid "Mail"
+msgstr ""
+
+#: aleksis/core/preferences.py:27
+msgid "Footer"
+msgstr ""
+
+#: aleksis/core/preferences.py:28
+msgid "Accounts"
+msgstr ""
+
+#: aleksis/core/preferences.py:29
 msgid "Authentication"
 msgstr ""
 
-#: preferences.py:28
+#: aleksis/core/preferences.py:30
 msgid "Internationalisation"
 msgstr ""
 
-#: preferences.py:37
+#: aleksis/core/preferences.py:41
 msgid "Site title"
 msgstr ""
 
-#: preferences.py:46
+#: aleksis/core/preferences.py:52
 msgid "Site description"
 msgstr ""
 
-#: preferences.py:55
+#: aleksis/core/preferences.py:63
 msgid "Primary colour"
 msgstr ""
 
-#: preferences.py:64
+#: aleksis/core/preferences.py:75
 msgid "Secondary colour"
 msgstr ""
 
-#: preferences.py:72
+#: aleksis/core/preferences.py:86
 msgid "Logo"
 msgstr ""
 
-#: preferences.py:80
+#: aleksis/core/preferences.py:96
 msgid "Favicon"
 msgstr ""
 
-#: preferences.py:88
+#: aleksis/core/preferences.py:106
 msgid "PWA-Icon"
 msgstr ""
 
-#: preferences.py:97
+#: aleksis/core/preferences.py:117
 msgid "Mail out name"
 msgstr ""
 
-#: preferences.py:106
+#: aleksis/core/preferences.py:128
 msgid "Mail out address"
 msgstr ""
 
-#: preferences.py:116
+#: aleksis/core/preferences.py:140
 msgid "Link to privacy policy"
 msgstr ""
 
-#: preferences.py:126
+#: aleksis/core/preferences.py:152
 msgid "Link to imprint"
 msgstr ""
 
-#: preferences.py:136
+#: aleksis/core/preferences.py:164
 msgid "Name format for addressing"
 msgstr ""
 
-#: preferences.py:150
+#: aleksis/core/preferences.py:180
 msgid "Channels to use for notifications"
 msgstr ""
 
-#: preferences.py:160
+#: aleksis/core/preferences.py:192
 msgid "Regular expression to match primary group, e.g. '^Class .*'"
 msgstr ""
 
-#: preferences.py:169
+#: aleksis/core/preferences.py:203
 msgid "Field on person to match primary group against"
 msgstr ""
 
-#: preferences.py:181
+#: aleksis/core/preferences.py:215
+msgid "Automatically create new persons for new users"
+msgstr ""
+
+#: aleksis/core/preferences.py:224
+msgid "Automatically link existing persons to new users by their e-mail address"
+msgstr ""
+
+#: aleksis/core/preferences.py:235
 msgid "Display name of the school"
 msgstr ""
 
-#: preferences.py:190
+#: aleksis/core/preferences.py:246
 msgid "Official name of the school, e.g. as given by supervisory authority"
 msgstr ""
 
-#: preferences.py:198
-msgid "Enabled custom authentication backends"
+#: aleksis/core/preferences.py:254
+msgid "Allow users to change their passwords"
+msgstr ""
+
+#: aleksis/core/preferences.py:262
+msgid "Enable signup"
 msgstr ""
 
-#: preferences.py:211
+#: aleksis/core/preferences.py:273
 msgid "Available languages"
 msgstr ""
 
-#: preferences.py:223
+#: aleksis/core/preferences.py:285
 msgid "Send emails if data checks detect problems"
 msgstr ""
 
-#: preferences.py:234
+#: aleksis/core/preferences.py:296
 msgid "Email recipients for data checks problem emails"
 msgstr ""
 
-#: preferences.py:245
+#: aleksis/core/preferences.py:307
 msgid "Email recipient groups for data checks problem emails"
 msgstr ""
 
-#: settings.py:322
-msgid "English"
+#: aleksis/core/preferences.py:316
+msgid "Show dashboard to users without login"
 msgstr ""
 
-#: settings.py:323
-msgid "German"
+#: aleksis/core/preferences.py:325
+msgid "Allow users to edit their dashboard"
+msgstr ""
+
+#: aleksis/core/preferences.py:336
+msgid "Fields on person model which are editable by themselves."
+msgstr ""
+
+#: aleksis/core/preferences.py:350
+msgid "Editable fields on person model which should trigger a notification on change"
+msgstr ""
+
+#: aleksis/core/preferences.py:363
+msgid "Contact for notification if a person changes their data"
+msgstr ""
+
+#: aleksis/core/preferences.py:373
+msgid "PDF file expiration duration"
 msgstr ""
 
-#: settings.py:324
-msgid "French"
+#: aleksis/core/preferences.py:374
+msgid "in minutes"
 msgstr ""
 
-#: settings.py:325
-msgid "Norwegian (bokmål)"
-msgstr "Norsk (bokmål)"
+#: aleksis/core/preferences.py:384
+msgid "Automatically update the dashboard and its widgets"
+msgstr ""
+
+#: aleksis/core/preferences.py:394
+msgid "Automatically update the dashboard and its widgets sitewide"
+msgstr ""
 
-#: tables.py:19 templates/core/announcement/list.html:36
-#: templates/core/group/full.html:24 templates/core/person/full.html:23
+#: aleksis/core/settings.py:452
+msgid "English"
+msgstr ""
+
+#: aleksis/core/settings.py:453
+msgid "German"
+msgstr ""
+
+#: aleksis/core/tables.py:19
+#: aleksis/core/templates/core/announcement/list.html:36
+#: aleksis/core/templates/core/group/full.html:24
+#: aleksis/core/templates/core/person/full.html:23
+#: aleksis/core/templates/oauth2_provider/application_detail.html:17
 msgid "Edit"
 msgstr ""
 
-#: tables.py:21 tables.py:89 templates/core/announcement/list.html:22
+#: aleksis/core/tables.py:21 aleksis/core/tables.py:89
+#: aleksis/core/templates/core/announcement/list.html:22
 msgid "Actions"
 msgstr ""
 
-#: tables.py:56 tables.py:57 tables.py:71 tables.py:87
-#: templates/core/announcement/list.html:42 templates/core/group/full.html:31
-#: templates/core/pages/delete.html:22 templates/core/person/full.html:30
+#: aleksis/core/tables.py:56 aleksis/core/tables.py:57
+#: aleksis/core/tables.py:71 aleksis/core/tables.py:87
+#: aleksis/core/templates/core/announcement/list.html:42
+#: aleksis/core/templates/core/group/full.html:31
+#: aleksis/core/templates/core/pages/delete.html:22
+#: aleksis/core/templates/core/person/full.html:30
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:20
+#: aleksis/core/templates/oauth2_provider/application_detail.html:21
 msgid "Delete"
 msgstr ""
 
-#: templates/403.html:14 templates/404.html:10 templates/500.html:10
+#: aleksis/core/templates/403.html:14 aleksis/core/templates/404.html:10
+#: aleksis/core/templates/500.html:10
+#: aleksis/core/templates/oauth2_provider/authorize.html:47
+#: aleksis/core/templates/oauth2_provider/authorized-oob.html:24
 msgid "Error"
 msgstr ""
 
-#: templates/403.html:14
+#: aleksis/core/templates/403.html:14
 msgid ""
 "You are not allowed to access the requested page or\n"
 "          object."
 msgstr ""
 
-#: templates/403.html:19 templates/404.html:17
+#: aleksis/core/templates/403.html:19 aleksis/core/templates/404.html:17
 msgid ""
 "\n"
 "            If you think this is an error in AlekSIS, please contact your site\n"
@@ -840,13 +1131,13 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/404.html:10
+#: aleksis/core/templates/404.html:10
 msgid ""
 "The requested page or object was not\n"
 "          found."
 msgstr ""
 
-#: templates/404.html:13
+#: aleksis/core/templates/404.html:13
 msgid ""
 "\n"
 "            If you were redirected by a link on an external page,\n"
@@ -854,13 +1145,13 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/500.html:10
+#: aleksis/core/templates/500.html:10
 msgid ""
 "An unexpected error has\n"
 "          occured."
 msgstr ""
 
-#: templates/500.html:13
+#: aleksis/core/templates/500.html:13
 msgid ""
 "\n"
 "            Your site administrators will automatically be notified about this\n"
@@ -868,167 +1159,394 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/503.html:10
+#: aleksis/core/templates/503.html:10
 msgid ""
 "The maintenance mode is currently enabled. Please try again\n"
 "          later."
 msgstr ""
 
-#: templates/503.html:13
+#: aleksis/core/templates/503.html:13
 msgid ""
 "\n"
 "            This page is currently unavailable. If this error persists, contact your site administrators:\n"
 "          "
 msgstr ""
 
-#: templates/core/additional_field/edit.html:6
-#: templates/core/additional_field/edit.html:7
+#: aleksis/core/templates/account/account_inactive.html:5
+#: aleksis/core/templates/account/account_inactive.html:6
+msgid "Account inactive"
+msgstr ""
+
+#: aleksis/core/templates/account/account_inactive.html:13
+msgid "Account inactive."
+msgstr ""
+
+#: aleksis/core/templates/account/account_inactive.html:15
+msgid ""
+"\n"
+"            This account is currently inactive. If you think this is an\n"
+"            error, please contact one of your site administrators.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/email/base_message.txt:5
+msgid "Hello!"
+msgstr ""
+
+#: aleksis/core/templates/account/email/base_message.txt:9
+#: aleksis/core/templates/templated_email/notification.email:22
+#: aleksis/core/templates/templated_email/notification.email:46
+msgid "Your AlekSIS team"
+msgstr ""
+
+#: aleksis/core/templates/account/email_confirm.html:5
+#: aleksis/core/templates/account/email_confirm.html:6
+#: aleksis/core/templates/account/email_confirm.html:17
+msgid "Confirm"
+msgstr ""
+
+#: aleksis/core/templates/account/email_confirm.html:12
+#, python-format
+msgid "Please confirm that <a href=\"mailto:%(email)s\">%(email)s</a> is an e-mail address for user %(user_display)s."
+msgstr ""
+
+#: aleksis/core/templates/account/email_confirm.html:25
+#, python-format
+msgid "This e-mail confirmation link expired or is invalid. Please <a href=\"%(email_url)s\">issue a new e-mail confirmation request</a>."
+msgstr ""
+
+#: aleksis/core/templates/account/password_change.html:12
+msgid "Forgot your current password? Click here to reset it:"
+msgstr ""
+
+#: aleksis/core/templates/account/password_change.html:12
+msgid "Forgot Password?"
+msgstr ""
+
+#: aleksis/core/templates/account/password_change_disabled.html:5
+#: aleksis/core/templates/account/password_change_disabled.html:6
+msgid "Changing of password disabled"
+msgstr ""
+
+#: aleksis/core/templates/account/password_change_disabled.html:13
+msgid "Changing of password disabled."
+msgstr ""
+
+#: aleksis/core/templates/account/password_change_disabled.html:15
+msgid ""
+"\n"
+"            Users are not allowed to edit their own passwords. If you think\n"
+"            this is an error please contact one of your site administrators.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset.html:5
+#: aleksis/core/templates/account/password_reset.html:15
+#: aleksis/core/templates/account/password_reset.html:23
+#: aleksis/core/templates/account/password_reset_done.html:5
+#: aleksis/core/templates/account/verification_email_required.html:5
+#: aleksis/core/templates/account/verification_email_required.html:6
+#: aleksis/core/templates/two_factor/core/login.html:81
+msgid "Reset password"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset.html:17
+msgid "Forgotten your password? Enter your e-mail address below, and we'll send you an e-mail allowing you to reset it."
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset.html:30
+msgid ""
+"Please contact one of your site administrators, if you\n"
+"        have any trouble resetting your password:"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_done.html:15
+msgid "Password reset mail sent"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_done.html:18
+#: aleksis/core/templates/account/verification_email_required.html:16
+msgid ""
+"\n"
+"            We have sent you an e-mail. Please contact one of your site\n"
+"            administrators if you do not receive it within a few minutes.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:15
+msgid "Bad token"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:19
+#, python-format
+msgid ""
+"\n"
+"              The password reset link was invalid, possibly because it has already been used. Please request a <a href=\"%(passwd_reset_url)s\"\n"
+"              class=\"blue-text text-lighten-2\">new password reset</a>.\n"
+"            "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:25
+msgid ""
+"\n"
+"              If this issue persists, please contact one of your site\n"
+"              administrators\n"
+"            "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:56
+#: aleksis/core/templates/account/password_reset_from_key_done.html:15
+msgid ""
+"\n"
+"            Your password is now changed!\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:61
+msgid "Back to login"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key_done.html:13
+msgid "Password changed!"
+msgstr ""
+
+#: aleksis/core/templates/account/password_set.html:5
+#: aleksis/core/templates/account/password_set.html:6
+#: aleksis/core/templates/account/password_set.html:12
+msgid "Set password"
+msgstr ""
+
+#: aleksis/core/templates/account/signup.html:5
+#: aleksis/core/templates/account/signup.html:6
+#: aleksis/core/templates/socialaccount/signup.html:5
+#: aleksis/core/templates/socialaccount/signup.html:6
+msgid "Signup"
+msgstr ""
+
+#: aleksis/core/templates/account/signup.html:12
+#, python-format
+msgid "Already have an account? Then please <a href=\"%(login_url)s\">sign in</a>."
+msgstr ""
+
+#: aleksis/core/templates/account/signup_closed.html:5
+#: aleksis/core/templates/account/signup_closed.html:6
+msgid "Signup closed"
+msgstr ""
+
+#: aleksis/core/templates/account/signup_closed.html:13
+msgid "Signup closed."
+msgstr ""
+
+#: aleksis/core/templates/account/signup_closed.html:15
+msgid ""
+"\n"
+"            This sign up is currently closed. If you think this is an\n"
+"            error, please contact one of your site administrators.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/verification_email_required.html:14
+msgid "Password reset mail sent!"
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:5
+#: aleksis/core/templates/account/verification_sent.html:6
+msgid "Verify your email address"
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:14
+msgid "Verify your email!"
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:16
+msgid ""
+"\n"
+"            This part of the site requires us to verify that you are who you claim to be.\n"
+"            For this purpose, we require that you verify ownership of your e-mail address.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:22
+msgid ""
+"\n"
+"            We have sent an e-mail to you for verification.\n"
+"            Please click on the link inside this e-mail. Please\n"
+"            contact us if you do not receive it within a few minutes.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:30
+#, python-format
+msgid "<strong>Note:</strong> you can still <a href=\"%(email_url)s\">change your e-mail address</a>"
+msgstr ""
+
+#: aleksis/core/templates/core/additional_field/edit.html:6
+#: aleksis/core/templates/core/additional_field/edit.html:7
 msgid "Edit additional field"
 msgstr ""
 
-#: templates/core/additional_field/list.html:14
+#: aleksis/core/templates/core/additional_field/list.html:14
 msgid "Create additional field"
 msgstr ""
 
-#: templates/core/announcement/form.html:14
-#: templates/core/announcement/form.html:21
+#: aleksis/core/templates/core/announcement/form.html:14
+#: aleksis/core/templates/core/announcement/form.html:21
 msgid "Edit announcement"
 msgstr ""
 
-#: templates/core/announcement/form.html:16
+#: aleksis/core/templates/core/announcement/form.html:16
 msgid "Publish announcement"
 msgstr ""
 
-#: templates/core/announcement/form.html:23
-#: templates/core/announcement/list.html:13
+#: aleksis/core/templates/core/announcement/form.html:23
+#: aleksis/core/templates/core/announcement/list.html:13
 msgid "Publish new announcement"
 msgstr ""
 
-#: templates/core/announcement/form.html:34
+#: aleksis/core/templates/core/announcement/form.html:34
 msgid "Save und publish announcement"
 msgstr ""
 
-#: templates/core/announcement/list.html:19
+#: aleksis/core/templates/core/announcement/list.html:19
 msgid "Valid from"
 msgstr ""
 
-#: templates/core/announcement/list.html:20
+#: aleksis/core/templates/core/announcement/list.html:20
 msgid "Valid until"
 msgstr ""
 
-#: templates/core/announcement/list.html:21
+#: aleksis/core/templates/core/announcement/list.html:21
 msgid "Recipients"
 msgstr ""
 
-#: templates/core/announcement/list.html:50
+#: aleksis/core/templates/core/announcement/list.html:50
 msgid "There are no announcements."
 msgstr ""
 
-#: templates/core/base.html:54
+#: aleksis/core/templates/core/base.html:60
 msgid "Logged in as"
 msgstr ""
 
-#: templates/core/base.html:147
+#: aleksis/core/templates/core/base.html:154
 msgid "About AlekSIS — The Free School Information System"
 msgstr ""
 
-#: templates/core/base.html:155
+#: aleksis/core/templates/core/base.html:162
 msgid "Impress"
 msgstr ""
 
-#: templates/core/base.html:163
+#: aleksis/core/templates/core/base.html:170
 msgid "Privacy Policy"
 msgstr ""
 
-#: templates/core/base_print.html:64
+#: aleksis/core/templates/core/base_print.html:72
 msgid "Powered by AlekSIS"
 msgstr ""
 
-#: templates/core/dashboard_widget/create.html:8
-#: templates/core/dashboard_widget/create.html:12
+#: aleksis/core/templates/core/dashboard_widget/create.html:8
+#: aleksis/core/templates/core/dashboard_widget/create.html:12
 #, python-format
 msgid "Create %(widget)s"
 msgstr ""
 
-#: templates/core/dashboard_widget/edit.html:8
-#: templates/core/dashboard_widget/edit.html:12
+#: aleksis/core/templates/core/dashboard_widget/dashboardwidget_broken.html:10
+msgid "This widget is currently not available."
+msgstr ""
+
+#: aleksis/core/templates/core/dashboard_widget/dashboardwidget_broken.html:14
+#, python-format
+msgid ""
+"\n"
+"            There is a problem getting the widget \"%(title)s\".\n"
+"            There is no need for you to take any action.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/core/dashboard_widget/edit.html:8
+#: aleksis/core/templates/core/dashboard_widget/edit.html:12
 #, python-format
 msgid "Edit %(widget)s"
 msgstr ""
 
-#: templates/core/dashboard_widget/list.html:17
+#: aleksis/core/templates/core/dashboard_widget/list.html:15
+msgid "Create dashboard widget"
+msgstr ""
+
+#: aleksis/core/templates/core/dashboard_widget/list.html:22
 #, python-format
 msgid "Create %(name)s"
 msgstr ""
 
-#: templates/core/dashboard_widget/list.html:25
-#: templates/core/edit_dashboard.html:8 templates/core/edit_dashboard.html:15
+#: aleksis/core/templates/core/dashboard_widget/list.html:32
+#: aleksis/core/templates/core/edit_dashboard.html:8
+#: aleksis/core/templates/core/edit_dashboard.html:15
 msgid "Edit default dashboard"
 msgstr ""
 
-#: templates/core/data_check/list.html:15
+#: aleksis/core/templates/core/data_check/list.html:15
 msgid "Check data again"
 msgstr ""
 
-#: templates/core/data_check/list.html:22
+#: aleksis/core/templates/core/data_check/list.html:22
 msgid "The system detected some problems with your data."
 msgstr ""
 
-#: templates/core/data_check/list.html:23
+#: aleksis/core/templates/core/data_check/list.html:23
 msgid ""
 "Please go through all data and check whether some extra action is\n"
 "          needed."
 msgstr ""
 
-#: templates/core/data_check/list.html:31
+#: aleksis/core/templates/core/data_check/list.html:31
 msgid "Everything is fine."
 msgstr ""
 
-#: templates/core/data_check/list.html:32
+#: aleksis/core/templates/core/data_check/list.html:32
 msgid "The system hasn't detected any problems with your data."
 msgstr ""
 
-#: templates/core/data_check/list.html:40
+#: aleksis/core/templates/core/data_check/list.html:40
 msgid "Detected problems"
 msgstr ""
 
-#: templates/core/data_check/list.html:45
+#: aleksis/core/templates/core/data_check/list.html:45
 msgid "Affected object"
 msgstr ""
 
-#: templates/core/data_check/list.html:46
+#: aleksis/core/templates/core/data_check/list.html:46
 msgid "Detected problem"
 msgstr ""
 
-#: templates/core/data_check/list.html:47
+#: aleksis/core/templates/core/data_check/list.html:47
 msgid "Show details"
 msgstr ""
 
-#: templates/core/data_check/list.html:48
+#: aleksis/core/templates/core/data_check/list.html:48
 msgid "Options to solve the problem"
 msgstr ""
 
-#: templates/core/data_check/list.html:62
+#: aleksis/core/templates/core/data_check/list.html:62
 msgid "Show object"
 msgstr ""
 
-#: templates/core/data_check/list.html:84
+#: aleksis/core/templates/core/data_check/list.html:84
 msgid "Registered checks"
 msgstr ""
 
-#: templates/core/data_check/list.html:88
+#: aleksis/core/templates/core/data_check/list.html:88
 msgid ""
 "\n"
 "            The system will check for the following problems:\n"
 "          "
 msgstr ""
 
-#: templates/core/edit_dashboard.html:6 templates/core/edit_dashboard.html:13
-#: templates/core/index.html:14
+#: aleksis/core/templates/core/edit_dashboard.html:6
+#: aleksis/core/templates/core/edit_dashboard.html:13
+#: aleksis/core/templates/core/index.html:17
 msgid "Edit dashboard"
 msgstr ""
 
-#: templates/core/edit_dashboard.html:24
+#: aleksis/core/templates/core/edit_dashboard.html:24
 msgid ""
 "\n"
 "          On this page you can arrange your personal dashboard. You can drag any items from \"Available widgets\" to \"Your\n"
@@ -1037,7 +1555,7 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/core/edit_dashboard.html:30
+#: aleksis/core/templates/core/edit_dashboard.html:30
 msgid ""
 "\n"
 "          On this page you can arrange the default dashboard which is shown when a user doesn't arrange his own\n"
@@ -1046,19 +1564,19 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/core/edit_dashboard.html:48
+#: aleksis/core/templates/core/edit_dashboard.html:48
 msgid "Available widgets"
 msgstr ""
 
-#: templates/core/edit_dashboard.html:57
+#: aleksis/core/templates/core/edit_dashboard.html:57
 msgid "Your dashboard"
 msgstr ""
 
-#: templates/core/edit_dashboard.html:59
+#: aleksis/core/templates/core/edit_dashboard.html:59
 msgid "Default dashboard"
 msgstr ""
 
-#: templates/core/group/child_groups.html:18
+#: aleksis/core/templates/core/group/child_groups.html:18
 msgid ""
 "\n"
 "          You can use this to assign child groups to groups. Please use the filters below to select groups you want to\n"
@@ -1066,38 +1584,38 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/core/group/child_groups.html:31
+#: aleksis/core/templates/core/group/child_groups.html:31
 msgid "Update selection"
 msgstr ""
 
-#: templates/core/group/child_groups.html:35
+#: aleksis/core/templates/core/group/child_groups.html:35
 msgid "Clear all filters"
 msgstr ""
 
-#: templates/core/group/child_groups.html:39
+#: aleksis/core/templates/core/group/child_groups.html:39
 msgid "Currently selected groups"
 msgstr ""
 
-#: templates/core/group/child_groups.html:52
+#: aleksis/core/templates/core/group/child_groups.html:52
 msgid "Start assigning child groups for this groups"
 msgstr ""
 
-#: templates/core/group/child_groups.html:61
+#: aleksis/core/templates/core/group/child_groups.html:61
 msgid ""
 "\n"
 "            Please select some groups in order to go on with assigning.\n"
 "          "
 msgstr ""
 
-#: templates/core/group/child_groups.html:72
+#: aleksis/core/templates/core/group/child_groups.html:72
 msgid "Current group:"
 msgstr ""
 
-#: templates/core/group/child_groups.html:78
+#: aleksis/core/templates/core/group/child_groups.html:78
 msgid "Please be careful!"
 msgstr ""
 
-#: templates/core/group/child_groups.html:79
+#: aleksis/core/templates/core/group/child_groups.html:79
 msgid ""
 "\n"
 "            If you click \"Back\" or \"Next\" the current group assignments are not saved.\n"
@@ -1106,127 +1624,135 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/core/group/child_groups.html:93
-#: templates/core/group/child_groups.html:128
-#: templates/two_factor/_wizard_actions.html:15
-#: templates/two_factor/_wizard_actions.html:20
+#: aleksis/core/templates/core/group/child_groups.html:93
+#: aleksis/core/templates/core/group/child_groups.html:128
+#: aleksis/core/templates/oauth2_provider/application_detail.html:9
+#: aleksis/core/templates/two_factor/_wizard_actions.html:15
+#: aleksis/core/templates/two_factor/_wizard_actions.html:20
 msgid "Back"
 msgstr ""
 
-#: templates/core/group/child_groups.html:99
-#: templates/core/group/child_groups.html:134
-#: templates/two_factor/_wizard_actions.html:26
+#: aleksis/core/templates/core/group/child_groups.html:99
+#: aleksis/core/templates/core/group/child_groups.html:134
+#: aleksis/core/templates/two_factor/_wizard_actions.html:26
 msgid "Next"
 msgstr ""
 
-#: templates/core/group/child_groups.html:106
-#: templates/core/group/child_groups.html:141
-#: templates/core/partials/save_button.html:3
+#: aleksis/core/templates/core/group/child_groups.html:106
+#: aleksis/core/templates/core/group/child_groups.html:141
+#: aleksis/core/templates/core/partials/save_button.html:3
 msgid "Save"
 msgstr ""
 
-#: templates/core/group/child_groups.html:112
-#: templates/core/group/child_groups.html:147
+#: aleksis/core/templates/core/group/child_groups.html:112
+#: aleksis/core/templates/core/group/child_groups.html:147
 msgid "Save and next"
 msgstr ""
 
-#: templates/core/group/edit.html:11 templates/core/group/edit.html:12
+#: aleksis/core/templates/core/group/edit.html:11
+#: aleksis/core/templates/core/group/edit.html:12
 msgid "Edit group"
 msgstr ""
 
-#: templates/core/group/full.html:38 templates/core/person/full.html:37
+#: aleksis/core/templates/core/group/full.html:38
+#: aleksis/core/templates/core/person/full.html:37
 msgid "Change preferences"
 msgstr ""
 
-#: templates/core/group/full.html:64
+#: aleksis/core/templates/core/group/full.html:64
 msgid "Statistics"
 msgstr ""
 
-#: templates/core/group/full.html:67
+#: aleksis/core/templates/core/group/full.html:67
 msgid "Count of members"
 msgstr ""
 
-#: templates/core/group/full.html:71
+#: aleksis/core/templates/core/group/full.html:71
 msgid "Average age"
 msgstr ""
 
-#: templates/core/group/full.html:76
+#: aleksis/core/templates/core/group/full.html:76
 msgid "Age range"
 msgstr ""
 
-#: templates/core/group/full.html:76
+#: aleksis/core/templates/core/group/full.html:76
 msgid "years to"
 msgstr ""
 
-#: templates/core/group/full.html:76
+#: aleksis/core/templates/core/group/full.html:76
 msgid "years "
 msgstr ""
 
-#: templates/core/group/list.html:14
+#: aleksis/core/templates/core/group/list.html:14
 msgid "Create group"
 msgstr ""
 
-#: templates/core/group/list.html:17
+#: aleksis/core/templates/core/group/list.html:17
 msgid "Filter groups"
 msgstr ""
 
-#: templates/core/group/list.html:24 templates/core/person/list.html:28
+#: aleksis/core/templates/core/group/list.html:24
+#: aleksis/core/templates/core/person/list.html:28
 msgid "Clear"
 msgstr ""
 
-#: templates/core/group/list.html:28
+#: aleksis/core/templates/core/group/list.html:28
 msgid "Selected groups"
 msgstr ""
 
-#: templates/core/group_type/edit.html:6 templates/core/group_type/edit.html:7
+#: aleksis/core/templates/core/group_type/edit.html:6
+#: aleksis/core/templates/core/group_type/edit.html:7
 msgid "Edit group type"
 msgstr ""
 
-#: templates/core/group_type/list.html:14
+#: aleksis/core/templates/core/group_type/list.html:14
 msgid "Create group type"
 msgstr ""
 
-#: templates/core/index.html:4
+#: aleksis/core/templates/core/index.html:4
 msgid "Home"
 msgstr ""
 
-#: templates/core/index.html:50
+#: aleksis/core/templates/core/index.html:49
 msgid ""
 "\n"
-"          You didn't customise your dashboard so that you see the system default. Please click on \"Edit dashboard\" to\n"
-"          customise your personal dashboard.\n"
-"        "
+"        You didn't customise your dashboard so that you see the system default. Please click on \"Edit dashboard\" to\n"
+"        customise your personal dashboard.\n"
+"      "
 msgstr ""
 
-#: templates/core/index.html:59
+#: aleksis/core/templates/core/index.html:59
 msgid "Last activities"
 msgstr ""
 
-#: templates/core/index.html:77
+#: aleksis/core/templates/core/index.html:77
 msgid "No activities available yet."
 msgstr ""
 
-#: templates/core/index.html:82
+#: aleksis/core/templates/core/index.html:82
 msgid "Recent notifications"
 msgstr ""
 
-#: templates/core/index.html:98
+#: aleksis/core/templates/core/index.html:98
+#: aleksis/core/templates/core/notifications.html:23
 msgid "More information →"
 msgstr ""
 
-#: templates/core/index.html:105
+#: aleksis/core/templates/core/index.html:105
+#: aleksis/core/templates/core/notifications.html:30
 msgid "No notifications available yet."
 msgstr ""
 
-#: templates/core/pages/about.html:6 templates/core/pages/about.html:15
+#: aleksis/core/templates/core/pages/about.html:6
+#: aleksis/core/templates/core/pages/about.html:15
 msgid "About AlekSIS"
 msgstr ""
 
-#: templates/core/pages/about.html:7
+#: aleksis/core/templates/core/pages/about.html:7
 msgid "AlekSIS – The Free School Information System"
 msgstr ""
 
-#: templates/core/pages/about.html:17
+#: aleksis/core/templates/core/pages/about.html:17
 msgid ""
 "\n"
 "              This platform is powered by AlekSIS, a web-based school information system (SIS) which can be used\n"
@@ -1235,19 +1761,19 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/pages/about.html:25
+#: aleksis/core/templates/core/pages/about.html:25
 msgid "Website of AlekSIS"
 msgstr ""
 
-#: templates/core/pages/about.html:26
+#: aleksis/core/templates/core/pages/about.html:26
 msgid "Source code"
 msgstr ""
 
-#: templates/core/pages/about.html:35
+#: aleksis/core/templates/core/pages/about.html:35
 msgid "Licence information"
 msgstr ""
 
-#: templates/core/pages/about.html:37
+#: aleksis/core/templates/core/pages/about.html:37
 msgid ""
 "\n"
 "              The core and the official apps of AlekSIS are licenced under the EUPL, version 1.2 or later. For licence\n"
@@ -1256,23 +1782,23 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/pages/about.html:45
+#: aleksis/core/templates/core/pages/about.html:45
 msgid "Free/Open Source Licence"
 msgstr ""
 
-#: templates/core/pages/about.html:46
+#: aleksis/core/templates/core/pages/about.html:46
 msgid "Other Licence"
 msgstr ""
 
-#: templates/core/pages/about.html:50
+#: aleksis/core/templates/core/pages/about.html:50
 msgid "Full licence text"
 msgstr ""
 
-#: templates/core/pages/about.html:51
+#: aleksis/core/templates/core/pages/about.html:51
 msgid "More information about the EUPL"
 msgstr ""
 
-#: templates/core/pages/about.html:90
+#: aleksis/core/templates/core/pages/about.html:90
 #, python-format
 msgid ""
 "\n"
@@ -1280,12 +1806,12 @@ msgid ""
 "                  "
 msgstr ""
 
-#: templates/core/pages/delete.html:6
+#: aleksis/core/templates/core/pages/delete.html:6
 #, python-format
 msgid "Delete %(object_name)s"
 msgstr ""
 
-#: templates/core/pages/delete.html:13
+#: aleksis/core/templates/core/pages/delete.html:13
 #, python-format
 msgid ""
 "\n"
@@ -1293,102 +1819,114 @@ msgid ""
 "    "
 msgstr ""
 
-#: templates/core/pages/progress.html:27
+#: aleksis/core/templates/core/pages/progress.html:27
 msgid ""
 "\n"
 "              Without activated JavaScript the progress status can't be updated.\n"
 "            "
 msgstr ""
 
-#: templates/core/pages/progress.html:47
-#: templates/two_factor/core/otp_required.html:19
+#: aleksis/core/templates/core/pages/progress.html:47
+#: aleksis/core/templates/two_factor/core/otp_required.html:19
 msgid "Go back"
 msgstr ""
 
-#: templates/core/pages/system_status.html:12
+#: aleksis/core/templates/core/pages/system_status.html:12
 msgid "System checks"
 msgstr ""
 
-#: templates/core/pages/system_status.html:21
+#: aleksis/core/templates/core/pages/system_status.html:21
 msgid "Maintenance mode enabled"
 msgstr ""
 
-#: templates/core/pages/system_status.html:23
+#: aleksis/core/templates/core/pages/system_status.html:23
 msgid ""
 "\n"
 "                Only admin and visitors from internal IPs can access thesite.\n"
 "              "
 msgstr ""
 
-#: templates/core/pages/system_status.html:34
+#: aleksis/core/templates/core/pages/system_status.html:34
 msgid "Maintenance mode disabled"
 msgstr ""
 
-#: templates/core/pages/system_status.html:35
+#: aleksis/core/templates/core/pages/system_status.html:35
 msgid "Everyone can access the site."
 msgstr ""
 
-#: templates/core/pages/system_status.html:45
+#: aleksis/core/templates/core/pages/system_status.html:45
 msgid "Debug mode enabled"
 msgstr ""
 
-#: templates/core/pages/system_status.html:47
+#: aleksis/core/templates/core/pages/system_status.html:47
 msgid ""
 "\n"
 "                The web server throws back debug information on errors. Do not use in production!\n"
 "              "
 msgstr ""
 
-#: templates/core/pages/system_status.html:54
+#: aleksis/core/templates/core/pages/system_status.html:54
 msgid "Debug mode disabled"
 msgstr ""
 
-#: templates/core/pages/system_status.html:56
+#: aleksis/core/templates/core/pages/system_status.html:56
 msgid ""
 "\n"
 "                Debug mode is disabled. Default error pages are displayed on errors.\n"
 "              "
 msgstr ""
 
-#: templates/core/pages/system_status.html:69
+#: aleksis/core/templates/core/pages/system_status.html:69
 msgid "System health checks"
 msgstr ""
 
-#: templates/core/pages/system_status.html:75
+#: aleksis/core/templates/core/pages/system_status.html:75
 msgid "Service"
 msgstr ""
 
-#: templates/core/pages/system_status.html:76
-#: templates/core/pages/system_status.html:115
+#: aleksis/core/templates/core/pages/system_status.html:76
+#: aleksis/core/templates/core/pages/system_status.html:115
 msgid "Status"
 msgstr ""
 
-#: templates/core/pages/system_status.html:77
+#: aleksis/core/templates/core/pages/system_status.html:77
 msgid "Time taken"
 msgstr ""
 
-#: templates/core/pages/system_status.html:96
+#: aleksis/core/templates/core/pages/system_status.html:96
 msgid "seconds"
 msgstr ""
 
-#: templates/core/pages/system_status.html:107
+#: aleksis/core/templates/core/pages/system_status.html:107
 msgid "Celery task results"
 msgstr ""
 
-#: templates/core/pages/system_status.html:112
+#: aleksis/core/templates/core/pages/system_status.html:112
 msgid "Task"
 msgstr ""
 
-#: templates/core/pages/system_status.html:113
+#: aleksis/core/templates/core/pages/system_status.html:113
 msgid "ID"
 msgstr ""
 
-#: templates/core/pages/system_status.html:114
+#: aleksis/core/templates/core/pages/system_status.html:114
 msgid "Date done"
 msgstr ""
 
-#: templates/core/partials/announcements.html:9
-#: templates/core/partials/announcements.html:36
+#: aleksis/core/templates/core/pages/test_pdf.html:7
+#: aleksis/core/templates/core/pages/test_pdf.html:8
+msgid "Test PDF generation"
+msgstr ""
+
+#: aleksis/core/templates/core/pages/test_pdf.html:14
+msgid ""
+"\n"
+"        This simple view can be used to ensure the correct function of the built-in PDF generation system.\n"
+"      "
+msgstr ""
+
+#: aleksis/core/templates/core/partials/announcements.html:8
+#: aleksis/core/templates/core/partials/announcements.html:35
 #, python-format
 msgid ""
 "\n"
@@ -1396,7 +1934,7 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/partials/announcements.html:13
+#: aleksis/core/templates/core/partials/announcements.html:12
 #, python-format
 msgid ""
 "\n"
@@ -1404,7 +1942,7 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/partials/announcements.html:40
+#: aleksis/core/templates/core/partials/announcements.html:39
 #, python-format
 msgid ""
 "\n"
@@ -1412,23 +1950,23 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/partials/crud_events.html:15
+#: aleksis/core/templates/core/partials/crud_events.html:15
 msgid "Changed by"
 msgstr ""
 
-#: templates/core/partials/crud_events.html:15
+#: aleksis/core/templates/core/partials/crud_events.html:15
 msgid "Unknown"
 msgstr ""
 
-#: templates/core/partials/language_form.html:15
+#: aleksis/core/templates/core/partials/language_form.html:15
 msgid "Language"
 msgstr ""
 
-#: templates/core/partials/language_form.html:27
+#: aleksis/core/templates/core/partials/language_form.html:27
 msgid "Select language"
 msgstr ""
 
-#: templates/core/partials/no_person.html:12
+#: aleksis/core/templates/core/partials/no_person.html:12
 msgid ""
 "\n"
 "            Your administrator account is not linked to any person. Therefore,\n"
@@ -1436,7 +1974,7 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/core/partials/no_person.html:19
+#: aleksis/core/templates/core/partials/no_person.html:19
 msgid ""
 "\n"
 "            Your user account is not linked to a person. This means you\n"
@@ -1445,105 +1983,178 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/core/person/accounts.html:12
-#: templates/core/person/accounts.html:14
-msgid "Link persons to accounts"
-msgstr ""
-
-#: templates/core/person/accounts.html:21
-msgid ""
-"\n"
-"        You can use this form to assign user accounts to persons. Use the\n"
-"        dropdowns to select existing accounts; use the text fields to create new\n"
-"        accounts on-the-fly. The latter will create a new account with the\n"
-"        entered username and copy all other details from the person.\n"
-"      "
-msgstr ""
-
-#: templates/core/person/accounts.html:36
-#: templates/core/person/accounts.html:60
-msgid "Update"
-msgstr ""
-
-#: templates/core/person/accounts.html:42
-msgid "Existing account"
-msgstr ""
-
-#: templates/core/person/accounts.html:43
-msgid "New account"
+#: aleksis/core/templates/core/person/create.html:12
+#: aleksis/core/templates/core/person/create.html:13
+#: aleksis/core/templates/core/person/list.html:17
+msgid "Create person"
 msgstr ""
 
-#: templates/core/person/edit.html:12 templates/core/person/edit.html:13
+#: aleksis/core/templates/core/person/edit.html:12
+#: aleksis/core/templates/core/person/edit.html:13
 msgid "Edit person"
 msgstr ""
 
-#: templates/core/person/full.html:44
+#: aleksis/core/templates/core/person/full.html:44
+#: aleksis/core/templates/impersonate/list_users.html:7
+#: aleksis/core/templates/impersonate/list_users.html:8
 msgid "Impersonate"
 msgstr ""
 
-#: templates/core/person/full.html:50
+#: aleksis/core/templates/core/person/full.html:50
 msgid "Contact details"
 msgstr ""
 
-#: templates/core/person/full.html:130
+#: aleksis/core/templates/core/person/full.html:130
 msgid "Children"
 msgstr ""
 
-#: templates/core/person/list.html:17
-msgid "Create person"
-msgstr ""
-
-#: templates/core/person/list.html:21
+#: aleksis/core/templates/core/person/list.html:21
 msgid "Filter persons"
 msgstr ""
 
-#: templates/core/person/list.html:32
+#: aleksis/core/templates/core/person/list.html:32
 msgid "Selected persons"
 msgstr ""
 
-#: templates/core/school_term/create.html:6
-#: templates/core/school_term/create.html:7
-#: templates/core/school_term/list.html:14
+#: aleksis/core/templates/core/school_term/create.html:6
+#: aleksis/core/templates/core/school_term/create.html:7
+#: aleksis/core/templates/core/school_term/list.html:14
 msgid "Create school term"
 msgstr ""
 
-#: templates/core/school_term/edit.html:6
-#: templates/core/school_term/edit.html:7
+#: aleksis/core/templates/core/school_term/edit.html:6
+#: aleksis/core/templates/core/school_term/edit.html:7
 msgid "Edit school term"
 msgstr ""
 
-#: templates/dynamic_preferences/form.html:9
+#: aleksis/core/templates/dynamic_preferences/form.html:9
 msgid "Site preferences"
 msgstr ""
 
-#: templates/dynamic_preferences/form.html:11
+#: aleksis/core/templates/dynamic_preferences/form.html:11
 msgid "My preferences"
 msgstr ""
 
-#: templates/dynamic_preferences/form.html:13
+#: aleksis/core/templates/dynamic_preferences/form.html:13
 #, python-format
 msgid "Preferences for %(instance)s"
 msgstr ""
 
-#: templates/dynamic_preferences/form.html:25
+#: aleksis/core/templates/dynamic_preferences/form.html:25
 msgid "Save preferences"
 msgstr ""
 
-#: templates/impersonate/list_users.html:8
-msgid "Impersonate user"
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:5
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:6
+msgid "Delete application"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:12
+#, python-format
+msgid "Are you sure to delete the application %(application_name)s?"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:24
+#: aleksis/core/templates/oauth2_provider/application_form.html:18
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:24
+#: aleksis/core/templates/two_factor/_wizard_actions.html:6
+msgid "Cancel"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:27
+msgid "Client id"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:35
+msgid "Client secret"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:43
+msgid "Client type"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:51
+msgid "Authorization Grant Type"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:59
+msgid "Redirect URIs"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_form.html:5
+msgid "Create OAuth2 Application"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_form.html:10
+msgid "Edit application"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_list.html:8
+msgid "OAuth2 applications"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_list.html:12
+msgid "Register new applications"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_list.html:23
+msgid "No applications defined."
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorize.html:5
+#: aleksis/core/templates/oauth2_provider/authorize.html:16
+msgid "Authorize"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorize.html:18
+msgid "The application requests access to the following scopes:"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorize.html:33
+msgid "Allow"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorize.html:36
+msgid "Disallow"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-oob.html:12
+msgid "Success!"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-oob.html:14
+msgid "Please return to your application and enter this code:"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:5
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:6
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:23
+msgid "Revoke access"
 msgstr ""
 
-#: templates/offline.html:5
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:12
+msgid "Are you sure to revoke the access for this application?"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:20
+msgid "Revoke"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:33
+msgid "No authorized applications."
+msgstr ""
+
+#: aleksis/core/templates/offline.html:5
 msgid "Network error"
 msgstr ""
 
-#: templates/offline.html:8
+#: aleksis/core/templates/offline.html:8
 msgid ""
 "No internet\n"
 "    connection."
 msgstr ""
 
-#: templates/offline.html:12
+#: aleksis/core/templates/offline.html:12
 msgid ""
 "\n"
 "      There was an error accessing this page. You probably don't have an internet connection. Check to see if your WiFi\n"
@@ -1552,36 +2163,116 @@ msgid ""
 "    "
 msgstr ""
 
-#: templates/search/search.html:8
+#: aleksis/core/templates/search/search.html:8
 msgid "Global Search"
 msgstr ""
 
-#: templates/search/search.html:15
+#: aleksis/core/templates/search/search.html:15
 msgid "Search Term"
 msgstr ""
 
-#: templates/search/search.html:26
+#: aleksis/core/templates/search/search.html:26
 msgid "Results"
 msgstr ""
 
-#: templates/search/search.html:38
+#: aleksis/core/templates/search/search.html:38
 msgid "No search results could be found to your search."
 msgstr ""
 
-#: templates/search/search.html:87
+#: aleksis/core/templates/search/search.html:87
 msgid "Please enter a search term above."
 msgstr ""
 
-#: templates/templated_email/data_checks.email:4
+#: aleksis/core/templates/socialaccount/authentication_error.html:5
+#: aleksis/core/templates/socialaccount/authentication_error.html:6
+msgid "Third-party Account Login Failure"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/authentication_error.html:13
+msgid "Third-party Account Login Failure."
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/authentication_error.html:15
+msgid ""
+"\n"
+"            An error occurred while attempting to login via your third-party account.\n"
+"            Please contact one of your site administrators.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/connections.html:5
+#: aleksis/core/templates/socialaccount/connections.html:6
+msgid "Connections"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/connections.html:24
+msgid "Remove"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/connections.html:34
+msgid "You currently have no third-party accounts connected to this account."
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/connections.html:37
+msgid "Add a Third-party Account"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/login_cancelled.html:5
+#: aleksis/core/templates/socialaccount/login_cancelled.html:6
+#: aleksis/core/templates/socialaccount/login_cancelled.html:13
+msgid "Login cancelled"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/login_cancelled.html:15
+#, python-format
+msgid ""
+"\n"
+"            You decided to cancel logging in to our site using one of your existing accounts. If this was a mistake, please proceed to <a href=\"%(login_url)s\">sign in</a>.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/signup.html:12
+#, python-format
+msgid ""
+"You are about to use your %(provider_name)s account to login to\n"
+"        %(site_name)s. As a final step, please complete the following form:"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/snippets/provider_list.html:12
+#, python-format
+msgid ""
+"\n"
+"                Login with %(name)s\n"
+"              "
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/snippets/provider_list.html:21
+#, python-format
+msgid ""
+"\n"
+"            Login with %(name)s\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/snippets/provider_list.html:30
+msgid ""
+"\n"
+"          No third-party account providers available.\n"
+"        "
+msgstr ""
+
+#: aleksis/core/templates/templated_email/data_checks.email:4
 msgid "The system detected some new problems with your data."
 msgstr ""
 
-#: templates/templated_email/data_checks.email:8
-#: templates/templated_email/data_checks.email:24
+#: aleksis/core/templates/templated_email/data_checks.email:8
+#: aleksis/core/templates/templated_email/data_checks.email:24
+#: aleksis/core/templates/templated_email/person_changed.email:8
+#: aleksis/core/templates/templated_email/person_changed.email:20
 msgid "Hello,"
 msgstr ""
 
-#: templates/templated_email/data_checks.email:10
+#: aleksis/core/templates/templated_email/data_checks.email:10
 msgid ""
 "\n"
 "  the system detected some new problems with your data.\n"
@@ -1589,7 +2280,7 @@ msgid ""
 " "
 msgstr ""
 
-#: templates/templated_email/data_checks.email:26
+#: aleksis/core/templates/templated_email/data_checks.email:26
 msgid ""
 "\n"
 "   the system detected some new problems with your data.\n"
@@ -1597,35 +2288,35 @@ msgid ""
 "  "
 msgstr ""
 
-#: templates/templated_email/data_checks.email:34
+#: aleksis/core/templates/templated_email/data_checks.email:34
 msgid "Problem description"
 msgstr ""
 
-#: templates/templated_email/data_checks.email:35
+#: aleksis/core/templates/templated_email/data_checks.email:35
 msgid "Count of objects with new problems"
 msgstr ""
 
-#: templates/templated_email/notification.email:3
+#: aleksis/core/templates/templated_email/notification.email:3
 msgid "New notification for"
 msgstr ""
 
-#: templates/templated_email/notification.email:6
-#: templates/templated_email/notification.email:27
+#: aleksis/core/templates/templated_email/notification.email:6
+#: aleksis/core/templates/templated_email/notification.email:27
 #, python-format
 msgid "Dear %(notification_user)s,"
 msgstr ""
 
-#: templates/templated_email/notification.email:8
-#: templates/templated_email/notification.email:29
+#: aleksis/core/templates/templated_email/notification.email:8
+#: aleksis/core/templates/templated_email/notification.email:29
 msgid "we got a new notification for you:"
 msgstr ""
 
-#: templates/templated_email/notification.email:15
-#: templates/templated_email/notification.email:35
+#: aleksis/core/templates/templated_email/notification.email:15
+#: aleksis/core/templates/templated_email/notification.email:35
 msgid "More information"
 msgstr ""
 
-#: templates/templated_email/notification.email:18
+#: aleksis/core/templates/templated_email/notification.email:18
 #, python-format
 msgid ""
 "\n"
@@ -1633,12 +2324,7 @@ msgid ""
 "    "
 msgstr ""
 
-#: templates/templated_email/notification.email:22
-#: templates/templated_email/notification.email:46
-msgid "Your AlekSIS team"
-msgstr ""
-
-#: templates/templated_email/notification.email:40
+#: aleksis/core/templates/templated_email/notification.email:40
 #, python-format
 msgid ""
 "\n"
@@ -1646,24 +2332,41 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/two_factor/_base_focus.html:6
-#: templates/two_factor/core/otp_required.html:22
-#: templates/two_factor/core/setup.html:5
-#: templates/two_factor/profile/profile.html:87
-msgid "Enable Two-Factor Authentication"
+#: aleksis/core/templates/templated_email/person_changed.email:4
+#, python-format
+msgid "%(person)s changed their data!"
 msgstr ""
 
-#: templates/two_factor/_wizard_actions.html:6
-msgid "Cancel"
+#: aleksis/core/templates/templated_email/person_changed.email:10
+#, python-format
+msgid ""
+"\n"
+"   the person %(person)s recently changed the following fields:\n"
+" "
+msgstr ""
+
+#: aleksis/core/templates/templated_email/person_changed.email:22
+#, python-format
+msgid ""
+"\n"
+"    the person %(person)s recently changed the following fields:\n"
+"  "
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:5
-#: templates/two_factor/core/backup_tokens.html:9
-#: templates/two_factor/profile/profile.html:46
+#: aleksis/core/templates/two_factor/_base_focus.html:6
+#: aleksis/core/templates/two_factor/core/otp_required.html:22
+#: aleksis/core/templates/two_factor/core/setup.html:5
+#: aleksis/core/templates/two_factor/profile/profile.html:87
+msgid "Enable Two-Factor Authentication"
+msgstr ""
+
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:5
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:9
+#: aleksis/core/templates/two_factor/profile/profile.html:46
 msgid "Backup Tokens"
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:14
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:14
 msgid ""
 "\n"
 "        Backup tokens can be used when your primary and backup\n"
@@ -1674,115 +2377,129 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:33
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:33
 msgid ""
 "\n"
 "          Print these tokens and keep them somewhere safe.\n"
 "        "
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:39
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:39
 msgid "You don't have any backup codes yet."
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:45
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:45
 msgid "Back to Account Security"
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:49
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:49
 msgid "Generate Tokens"
 msgstr ""
 
-#: templates/two_factor/core/login.html:16
-msgid "You have no permission to view this page. Please login with an other account."
+#: aleksis/core/templates/two_factor/core/login.html:20
+msgid "Login with username and password"
 msgstr ""
 
-#: templates/two_factor/core/login.html:24
-msgid "Please login to see this page."
+#: aleksis/core/templates/two_factor/core/login.html:28
+msgid ""
+"You have no permission to view this page. Please login with an other\n"
+"                    account."
 msgstr ""
 
-#: templates/two_factor/core/login.html:30
-msgid "Login with username and password"
+#: aleksis/core/templates/two_factor/core/login.html:36
+msgid "Please login to see this page."
 msgstr ""
 
-#: templates/two_factor/core/login.html:40
+#: aleksis/core/templates/two_factor/core/login.html:46
 msgid ""
-"We are calling your phone right now, please enter the\n"
-"              digits you hear."
+"\n"
+"                        We are calling your phone right now, please enter the\n"
+"                        digits you hear.\n"
+"                      "
 msgstr ""
 
-#: templates/two_factor/core/login.html:43
+#: aleksis/core/templates/two_factor/core/login.html:51
 msgid ""
-"We sent you a text message, please enter the tokens we\n"
-"              sent."
+"\n"
+"                        We sent you a text message, please enter the tokens we\n"
+"                        sent.\n"
+"                      "
 msgstr ""
 
-#: templates/two_factor/core/login.html:46
+#: aleksis/core/templates/two_factor/core/login.html:56
 msgid ""
-"Please enter the tokens generated by your token\n"
-"              generator."
+"\n"
+"                        Please enter the tokens generated by your token\n"
+"                        generator.\n"
+"                      "
 msgstr ""
 
-#: templates/two_factor/core/login.html:50
+#: aleksis/core/templates/two_factor/core/login.html:62
 msgid ""
-"Use this form for entering backup tokens for logging in.\n"
-"            These tokens have been generated for you to print and keep safe. Please\n"
-"            enter one of these backup tokens to login to your account."
+"\n"
+"                      Use this form for entering backup tokens for logging in.\n"
+"                      These tokens have been generated for you to print and keep safe. Please\n"
+"                      enter one of these backup tokens to login to your account.\n"
+"                    "
+msgstr ""
+
+#: aleksis/core/templates/two_factor/core/login.html:90
+msgid "Device currently not available?"
 msgstr ""
 
-#: templates/two_factor/core/login.html:68
+#: aleksis/core/templates/two_factor/core/login.html:92
 msgid "Or, alternatively, use one of your backup phones:"
 msgstr ""
 
-#: templates/two_factor/core/login.html:78
+#: aleksis/core/templates/two_factor/core/login.html:102
 msgid "As a last resort, you can use a backup token:"
 msgstr ""
 
-#: templates/two_factor/core/login.html:81
+#: aleksis/core/templates/two_factor/core/login.html:105
 msgid "Use Backup Token"
 msgstr ""
 
-#: templates/two_factor/core/login.html:93
+#: aleksis/core/templates/two_factor/core/login.html:116
 msgid "Use alternative login options"
 msgstr ""
 
-#: templates/two_factor/core/otp_required.html:9
+#: aleksis/core/templates/two_factor/core/otp_required.html:9
 msgid "Permission Denied"
 msgstr ""
 
-#: templates/two_factor/core/otp_required.html:10
+#: aleksis/core/templates/two_factor/core/otp_required.html:10
 msgid ""
 "The page you requested, enforces users to verify using\n"
 "          two-factor authentication for security reasons. You need to enable these\n"
 "          security features in order to access this page."
 msgstr ""
 
-#: templates/two_factor/core/otp_required.html:14
+#: aleksis/core/templates/two_factor/core/otp_required.html:14
 msgid ""
 "Two-factor authentication is not enabled for your\n"
 "          account. Enable two-factor authentication for enhanced account\n"
 "          security."
 msgstr ""
 
-#: templates/two_factor/core/phone_register.html:5
-#: templates/two_factor/core/phone_register.html:9
+#: aleksis/core/templates/two_factor/core/phone_register.html:5
+#: aleksis/core/templates/two_factor/core/phone_register.html:9
 msgid "Add Backup Phone"
 msgstr ""
 
-#: templates/two_factor/core/phone_register.html:12
+#: aleksis/core/templates/two_factor/core/phone_register.html:12
 msgid ""
 "You'll be adding a backup phone number to your\n"
 "      account. This number will be used if your primary method of\n"
 "      registration is not available."
 msgstr ""
 
-#: templates/two_factor/core/phone_register.html:16
+#: aleksis/core/templates/two_factor/core/phone_register.html:16
 msgid ""
 "We've sent a token to your phone number. Please\n"
 "      enter the token you've received."
 msgstr ""
 
-#: templates/two_factor/core/setup.html:9
+#: aleksis/core/templates/two_factor/core/setup.html:9
 msgid ""
 "\n"
 "        You are about to take your account security to the\n"
@@ -1791,14 +2508,14 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:17
+#: aleksis/core/templates/two_factor/core/setup.html:17
 msgid ""
 "\n"
 "        Please select which authentication method you would like to use:\n"
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:23
+#: aleksis/core/templates/two_factor/core/setup.html:23
 msgid ""
 "\n"
 "        To start using a token generator, please use your\n"
@@ -1807,7 +2524,7 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:34
+#: aleksis/core/templates/two_factor/core/setup.html:34
 msgid ""
 "\n"
 "        Please enter the phone number you wish to receive the\n"
@@ -1815,7 +2532,7 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:41
+#: aleksis/core/templates/two_factor/core/setup.html:41
 msgid ""
 "\n"
 "        Please enter the phone number you wish to be called on.\n"
@@ -1823,21 +2540,21 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:50
+#: aleksis/core/templates/two_factor/core/setup.html:50
 msgid ""
 "\n"
 "            We are calling your phone right now, please enter the digits you hear.\n"
 "          "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:56
+#: aleksis/core/templates/two_factor/core/setup.html:56
 msgid ""
 "\n"
 "            We sent you a text message, please enter the tokens we sent.\n"
 "          "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:63
+#: aleksis/core/templates/two_factor/core/setup.html:63
 msgid ""
 "\n"
 "          We've encountered an issue with the selected authentication method. Please\n"
@@ -1847,7 +2564,7 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:73
+#: aleksis/core/templates/two_factor/core/setup.html:73
 msgid ""
 "\n"
 "        To identify and verify your YubiKey, please insert a\n"
@@ -1856,29 +2573,29 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:5
-#: templates/two_factor/core/setup_complete.html:9
+#: aleksis/core/templates/two_factor/core/setup_complete.html:5
+#: aleksis/core/templates/two_factor/core/setup_complete.html:9
 msgid "Two-Factor Authentication successfully enabled"
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:14
+#: aleksis/core/templates/two_factor/core/setup_complete.html:14
 msgid ""
 "\n"
 "        Congratulations, you've successfully enabled two-factor authentication.\n"
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:24
-#: templates/two_factor/core/setup_complete.html:44
+#: aleksis/core/templates/two_factor/core/setup_complete.html:24
+#: aleksis/core/templates/two_factor/core/setup_complete.html:44
 msgid "Back to Profile"
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:28
-#: templates/two_factor/core/setup_complete.html:48
+#: aleksis/core/templates/two_factor/core/setup_complete.html:28
+#: aleksis/core/templates/two_factor/core/setup_complete.html:48
 msgid "Generate backup codes"
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:34
+#: aleksis/core/templates/two_factor/core/setup_complete.html:34
 msgid ""
 "\n"
 "          However, it might happen that you don't have access to\n"
@@ -1887,65 +2604,65 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:52
-#: templates/two_factor/profile/profile.html:41
+#: aleksis/core/templates/two_factor/core/setup_complete.html:52
+#: aleksis/core/templates/two_factor/profile/profile.html:41
 msgid "Add Phone Number"
 msgstr ""
 
-#: templates/two_factor/profile/disable.html:5
-#: templates/two_factor/profile/disable.html:9
-#: templates/two_factor/profile/profile.html:63
-#: templates/two_factor/profile/profile.html:73
+#: aleksis/core/templates/two_factor/profile/disable.html:5
+#: aleksis/core/templates/two_factor/profile/disable.html:9
+#: aleksis/core/templates/two_factor/profile/profile.html:63
+#: aleksis/core/templates/two_factor/profile/profile.html:73
 msgid "Disable Two-Factor Authentication"
 msgstr ""
 
-#: templates/two_factor/profile/disable.html:12
+#: aleksis/core/templates/two_factor/profile/disable.html:12
 msgid "You are about to disable two-factor authentication. This weakens your account security, are you sure?"
 msgstr ""
 
-#: templates/two_factor/profile/disable.html:26
+#: aleksis/core/templates/two_factor/profile/disable.html:26
 msgid "Disable"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:5
-#: templates/two_factor/profile/profile.html:10
+#: aleksis/core/templates/two_factor/profile/profile.html:5
+#: aleksis/core/templates/two_factor/profile/profile.html:10
 msgid "Account Security"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:15
+#: aleksis/core/templates/two_factor/profile/profile.html:15
 msgid "Tokens will be generated by your token generator."
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:17
+#: aleksis/core/templates/two_factor/profile/profile.html:17
 #, python-format
 msgid "Primary method: %(primary)s"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:19
+#: aleksis/core/templates/two_factor/profile/profile.html:19
 msgid "Tokens will be generated by your YubiKey."
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:23
+#: aleksis/core/templates/two_factor/profile/profile.html:23
 msgid "Backup Phone Numbers"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:24
+#: aleksis/core/templates/two_factor/profile/profile.html:24
 msgid ""
 "If your primary method is not available, we are able to\n"
 "        send backup tokens to the phone numbers listed below."
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:33
+#: aleksis/core/templates/two_factor/profile/profile.html:33
 msgid "Unregister"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:48
+#: aleksis/core/templates/two_factor/profile/profile.html:48
 msgid ""
 "If you don't have any device with you, you can access\n"
 "        your account using backup tokens."
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:50
+#: aleksis/core/templates/two_factor/profile/profile.html:50
 #, python-format
 msgid ""
 "\n"
@@ -1958,11 +2675,11 @@ msgid_plural ""
 msgstr[0] ""
 msgstr[1] ""
 
-#: templates/two_factor/profile/profile.html:59
+#: aleksis/core/templates/two_factor/profile/profile.html:59
 msgid "Show Codes"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:65
+#: aleksis/core/templates/two_factor/profile/profile.html:65
 msgid ""
 "\n"
 "        However we strongly discourage you to do so, you can\n"
@@ -1970,7 +2687,7 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:78
+#: aleksis/core/templates/two_factor/profile/profile.html:78
 msgid ""
 "\n"
 "        Two-factor authentication is not enabled for your\n"
@@ -1979,99 +2696,138 @@ msgid ""
 "      "
 msgstr ""
 
-#: util/notifications.py:65
+#: aleksis/core/util/notifications.py:65
 msgid "E-Mail"
 msgstr ""
 
-#: util/notifications.py:66
+#: aleksis/core/util/notifications.py:66
 msgid "SMS"
 msgstr ""
 
-#: views.py:141
+#: aleksis/core/util/pdf.py:105
+msgid "Progress: Generate PDF file"
+msgstr ""
+
+#: aleksis/core/util/pdf.py:106
+msgid "Generating PDF file …"
+msgstr ""
+
+#: aleksis/core/util/pdf.py:107
+msgid "The PDF file has been generated successfully."
+msgstr ""
+
+#: aleksis/core/util/pdf.py:108
+msgid "There was a problem while generating the PDF file."
+msgstr ""
+
+#: aleksis/core/util/pdf.py:111
+msgid "Download PDF"
+msgstr ""
+
+#: aleksis/core/views.py:251
 msgid "The school term has been created."
 msgstr ""
 
-#: views.py:153
+#: aleksis/core/views.py:263
 msgid "The school term has been saved."
 msgstr ""
 
-#: views.py:298
+#: aleksis/core/views.py:387
 msgid "The child groups were successfully saved."
 msgstr ""
 
-#: views.py:336
+#: aleksis/core/views.py:406 aleksis/core/views.py:416
 msgid "The person has been saved."
 msgstr ""
 
-#: views.py:375
+#: aleksis/core/views.py:466
 msgid "The group has been saved."
 msgstr ""
 
-#: views.py:467
+#: aleksis/core/views.py:563
 msgid "The announcement has been saved."
 msgstr ""
 
-#: views.py:483
+#: aleksis/core/views.py:579
 msgid "The announcement has been deleted."
 msgstr ""
 
-#: views.py:562
+#: aleksis/core/views.py:663
 msgid "The preferences have been saved successfully."
 msgstr ""
 
-#: views.py:586
+#: aleksis/core/views.py:687
 msgid "The person has been deleted."
 msgstr ""
 
-#: views.py:600
+#: aleksis/core/views.py:701
 msgid "The group has been deleted."
 msgstr ""
 
-#: views.py:632
+#: aleksis/core/views.py:733
 msgid "The additional_field has been saved."
 msgstr ""
 
-#: views.py:666
+#: aleksis/core/views.py:767
 msgid "The additional field has been deleted."
 msgstr ""
 
-#: views.py:691
+#: aleksis/core/views.py:792
 msgid "The group type has been saved."
 msgstr ""
 
-#: views.py:721
+#: aleksis/core/views.py:822
 msgid "The group type has been deleted."
 msgstr ""
 
-#: views.py:749
-msgid "The data check has been started. Please note that it may take a while before you are able to fetch the data on this page."
+#: aleksis/core/views.py:855
+msgid "Progress: Run data checks"
 msgstr ""
 
-#: views.py:754
-msgid "The data check has finished."
+#: aleksis/core/views.py:856
+msgid "Run data checks …"
 msgstr ""
 
-#: views.py:769
+#: aleksis/core/views.py:857
+msgid "The data checks were run successfully."
+msgstr ""
+
+#: aleksis/core/views.py:858
+msgid "There was a problem while running data checks."
+msgstr ""
+
+#: aleksis/core/views.py:874
 #, python-brace-format
 msgid "The solve option '{solve_option_obj.verbose_name}' "
 msgstr ""
 
-#: views.py:811
+#: aleksis/core/views.py:916
 msgid "The dashboard widget has been saved."
 msgstr ""
 
-#: views.py:841
+#: aleksis/core/views.py:946
 msgid "The dashboard widget has been created."
 msgstr ""
 
-#: views.py:851
+#: aleksis/core/views.py:956
 msgid "The dashboard widget has been deleted."
 msgstr ""
 
-#: views.py:914
+#: aleksis/core/views.py:1023
 msgid "Your dashboard configuration has been saved successfully."
 msgstr ""
 
-#: views.py:916
+#: aleksis/core/views.py:1025
 msgid "The configuration of the default dashboard has been saved successfully."
 msgstr ""
+
+#: aleksis/core/views.py:1153
+msgid "The third-party account could not be disconnected because it is the only login method available."
+msgstr ""
+
+#: aleksis/core/views.py:1160
+msgid "The third-party account has been successfully disconnected."
+msgstr ""
+
+#~ msgid "Norwegian (bokmål)"
+#~ msgstr "Norsk (bokmål)"
diff --git a/aleksis/core/locale/nb_NO/LC_MESSAGES/djangojs.po b/aleksis/core/locale/nb_NO/LC_MESSAGES/djangojs.po
index ea24fc3ff92afc211898fb8b426dccb97fce56ed..dc6d028e5442a3666f35a7cb46e39e5c01cdd3f9 100644
--- a/aleksis/core/locale/nb_NO/LC_MESSAGES/djangojs.po
+++ b/aleksis/core/locale/nb_NO/LC_MESSAGES/djangojs.po
@@ -8,7 +8,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2021-01-11 21:30+0100\n"
+"POT-Creation-Date: 2021-10-28 16:18+0200\n"
 "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
@@ -17,18 +17,18 @@ msgstr ""
 "Content-Type: text/plain; charset=UTF-8\n"
 "Content-Transfer-Encoding: 8bit\n"
 
-#: static/js/main.js:15
+#: aleksis/core/static/js/main.js:15
 msgid "Today"
 msgstr ""
 
-#: static/js/main.js:16
+#: aleksis/core/static/js/main.js:16
 msgid "Cancel"
 msgstr ""
 
-#: static/js/main.js:17
+#: aleksis/core/static/js/main.js:17
 msgid "OK"
 msgstr ""
 
-#: static/js/main.js:118
+#: aleksis/core/static/js/main.js:127
 msgid "This page may contain outdated information since there is no internet connection."
 msgstr ""
diff --git a/aleksis/core/locale/tr_TR/LC_MESSAGES/django.po b/aleksis/core/locale/tr_TR/LC_MESSAGES/django.po
index 7ecf671cf7dd018f3d159265a13605f8f050e518..c1f28664922a7b5cbdac80d557c812092765b16f 100644
--- a/aleksis/core/locale/tr_TR/LC_MESSAGES/django.po
+++ b/aleksis/core/locale/tr_TR/LC_MESSAGES/django.po
@@ -8,7 +8,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: AlekSIS (School Information System) 0.1\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2021-01-11 21:30+0100\n"
+"POT-Creation-Date: 2021-10-28 16:18+0200\n"
 "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
@@ -17,822 +17,1113 @@ msgstr ""
 "Content-Type: text/plain; charset=UTF-8\n"
 "Content-Transfer-Encoding: 8bit\n"
 
-#: data_checks.py:53
+#: aleksis/core/apps.py:150
+msgid "OpenID Connect scope"
+msgstr ""
+
+#: aleksis/core/apps.py:151
+msgid "Given name, family name, link to profile and picture if existing."
+msgstr ""
+
+#: aleksis/core/apps.py:152
+msgid "Full home postal address"
+msgstr ""
+
+#: aleksis/core/apps.py:153
+msgid "Email address"
+msgstr ""
+
+#: aleksis/core/apps.py:154
+msgid "Home and mobile phone"
+msgstr ""
+
+#: aleksis/core/data_checks.py:55
 msgid "Ignore problem"
 msgstr ""
 
-#: data_checks.py:174
+#: aleksis/core/data_checks.py:184
 #, python-brace-format
 msgid "Solve option '{solve_option_obj.verbose_name}' "
 msgstr ""
 
-#: filters.py:37 templates/core/base.html:77 templates/core/group/list.html:20
-#: templates/core/person/list.html:24 templates/search/search.html:7
-#: templates/search/search.html:22
-msgid "Search"
+#: aleksis/core/data_checks.py:291
+msgid "Deactivate DashboardWidget"
 msgstr ""
 
-#: filters.py:53
-msgid "Search by name"
+#: aleksis/core/data_checks.py:303
+msgid "Ensure that there are no broken DashboardWidgets."
 msgstr ""
 
-#: filters.py:65
-msgid "Search by contact details"
+#: aleksis/core/data_checks.py:304
+msgid "The DashboardWidget was reported broken automatically."
 msgstr ""
 
-#: forms.py:54
-msgid "You cannot set a new username when also selecting an existing user."
+#: aleksis/core/filters.py:37 aleksis/core/templates/core/base.html:83
+#: aleksis/core/templates/core/base.html:84
+#: aleksis/core/templates/core/group/list.html:20
+#: aleksis/core/templates/core/person/list.html:24
+#: aleksis/core/templates/search/search.html:7
+#: aleksis/core/templates/search/search.html:22
+msgid "Search"
 msgstr ""
 
-#: forms.py:58
-msgid "This username is already in use."
+#: aleksis/core/filters.py:53
+msgid "Search by name"
 msgstr ""
 
-#: forms.py:82
+#: aleksis/core/filters.py:65
+msgid "Search by contact details"
+msgstr ""
+
+#: aleksis/core/forms.py:41 aleksis/core/forms.py:387
 msgid "Base data"
 msgstr ""
 
-#: forms.py:88
+#: aleksis/core/forms.py:47
 msgid "Address"
 msgstr ""
 
-#: forms.py:89
+#: aleksis/core/forms.py:48
 msgid "Contact data"
 msgstr ""
 
-#: forms.py:91
+#: aleksis/core/forms.py:50
 msgid "Advanced personal data"
 msgstr ""
 
-#: forms.py:134
+#: aleksis/core/forms.py:93
 msgid "New user"
 msgstr ""
 
-#: forms.py:134
+#: aleksis/core/forms.py:93
 msgid "Create a new account"
 msgstr ""
 
-#: forms.py:146 models.py:102
+#: aleksis/core/forms.py:124
+msgid "You cannot set a new username when also selecting an existing user."
+msgstr ""
+
+#: aleksis/core/forms.py:128
+msgid "This username is already in use."
+msgstr ""
+
+#: aleksis/core/forms.py:145 aleksis/core/models.py:117
 msgid "School term"
 msgstr ""
 
-#: forms.py:147
+#: aleksis/core/forms.py:146
 msgid "Common data"
 msgstr ""
 
-#: forms.py:148 forms.py:197 menus.py:169 models.py:116
-#: templates/core/person/list.html:8 templates/core/person/list.html:9
+#: aleksis/core/forms.py:147 aleksis/core/forms.py:196
+#: aleksis/core/menus.py:238 aleksis/core/models.py:140
+#: aleksis/core/templates/core/person/list.html:8
+#: aleksis/core/templates/core/person/list.html:9
 msgid "Persons"
 msgstr ""
 
-#: forms.py:149
+#: aleksis/core/forms.py:148
 msgid "Additional data"
 msgstr ""
 
-#: forms.py:189 forms.py:192 models.py:45
+#: aleksis/core/forms.py:188 aleksis/core/forms.py:191
+#: aleksis/core/models.py:60
 msgid "Date"
 msgstr ""
 
-#: forms.py:190 forms.py:193 models.py:53
+#: aleksis/core/forms.py:189 aleksis/core/forms.py:192
+#: aleksis/core/models.py:68
 msgid "Time"
 msgstr ""
 
-#: forms.py:210 menus.py:177 models.py:338 templates/core/group/list.html:8
-#: templates/core/group/list.html:9 templates/core/person/full.html:144
+#: aleksis/core/forms.py:209 aleksis/core/menus.py:249
+#: aleksis/core/models.py:398 aleksis/core/templates/core/group/list.html:8
+#: aleksis/core/templates/core/group/list.html:9
+#: aleksis/core/templates/core/person/full.html:144
 msgid "Groups"
 msgstr ""
 
-#: forms.py:220
+#: aleksis/core/forms.py:219
 msgid "From when until when should the announcement be displayed?"
 msgstr ""
 
-#: forms.py:223
+#: aleksis/core/forms.py:222
 msgid "Who should see the announcement?"
 msgstr ""
 
-#: forms.py:224
+#: aleksis/core/forms.py:223
 msgid "Write your announcement:"
 msgstr ""
 
-#: forms.py:263
+#: aleksis/core/forms.py:262
 msgid "You are not allowed to create announcements which are only valid in the past."
 msgstr ""
 
-#: forms.py:267
+#: aleksis/core/forms.py:266
 msgid "The from date and time must be earlier then the until date and time."
 msgstr ""
 
-#: forms.py:276
+#: aleksis/core/forms.py:275
 msgid "You need at least one recipient."
 msgstr ""
 
-#: health_checks.py:15
+#: aleksis/core/forms.py:389
+msgid "Account data"
+msgstr ""
+
+#: aleksis/core/forms.py:391
+msgid "Consents"
+msgstr ""
+
+#: aleksis/core/forms.py:396
+msgid "Password"
+msgstr ""
+
+#: aleksis/core/forms.py:402
+msgid "Password (again)"
+msgstr ""
+
+#: aleksis/core/forms.py:411
+#, python-brace-format
+msgid "I have read the <a href='{privacy_policy}'>Privacy policy</a> and agree with them."
+msgstr ""
+
+#: aleksis/core/forms.py:435
+msgid "You must type the same password each time."
+msgstr ""
+
+#: aleksis/core/forms.py:580
+msgid "No valid selection."
+msgstr ""
+
+#: aleksis/core/health_checks.py:21
 msgid "There are unresolved data problems."
 msgstr ""
 
-#: menus.py:7 templates/two_factor/core/login.html:6
-#: templates/two_factor/core/login.html:10
-#: templates/two_factor/core/login.html:86
+#: aleksis/core/health_checks.py:38
+msgid "The backup folder doesn't exist."
+msgstr ""
+
+#: aleksis/core/health_checks.py:47
+#, python-brace-format
+msgid "Last backup {time_gone_since_backup}!"
+msgstr ""
+
+#: aleksis/core/health_checks.py:49
+msgid "No backup found!"
+msgstr ""
+
+#: aleksis/core/health_checks.py:76
+msgid "No backup result found!"
+msgstr ""
+
+#: aleksis/core/health_checks.py:78
+#, python-brace-format
+msgid "{task.status} - {task.result}"
+msgstr ""
+
+#: aleksis/core/menus.py:9 aleksis/core/templates/two_factor/core/login.html:6
+#: aleksis/core/templates/two_factor/core/login.html:22
+#: aleksis/core/templates/two_factor/core/login.html:76
 msgid "Login"
 msgstr ""
 
-#: menus.py:13
+#: aleksis/core/menus.py:15 aleksis/core/templates/account/signup.html:20
+#: aleksis/core/templates/socialaccount/signup.html:23
+msgid "Sign up"
+msgstr ""
+
+#: aleksis/core/menus.py:24
 msgid "Dashboard"
 msgstr ""
 
-#: menus.py:19
+#: aleksis/core/menus.py:32 aleksis/core/models.py:605
+#: aleksis/core/preferences.py:26
+#: aleksis/core/templates/core/notifications.html:4
+#: aleksis/core/templates/core/notifications.html:5
+msgid "Notifications"
+msgstr ""
+
+#: aleksis/core/menus.py:41
 msgid "Account"
 msgstr ""
 
-#: menus.py:26
+#: aleksis/core/menus.py:48
 msgid "Stop impersonation"
 msgstr ""
 
-#: menus.py:35 templates/core/base.html:56
+#: aleksis/core/menus.py:57 aleksis/core/templates/core/base.html:62
 msgid "Logout"
 msgstr ""
 
-#: menus.py:41
+#: aleksis/core/menus.py:63
 msgid "2FA"
 msgstr ""
 
-#: menus.py:47
+#: aleksis/core/menus.py:69
+#: aleksis/core/templates/account/password_change.html:5
+#: aleksis/core/templates/account/password_change.html:6
+#: aleksis/core/templates/account/password_change.html:19
+#: aleksis/core/templates/account/password_reset_from_key.html:5
+#: aleksis/core/templates/account/password_reset_from_key.html:42
+#: aleksis/core/templates/account/password_reset_from_key.html:46
+#: aleksis/core/templates/account/password_reset_from_key_done.html:5
+#: aleksis/core/templates/account/password_reset_from_key_done.html:6
+msgid "Change password"
+msgstr ""
+
+#: aleksis/core/menus.py:81
 msgid "Me"
 msgstr ""
 
-#: menus.py:56 templates/dynamic_preferences/form.html:5
+#: aleksis/core/menus.py:90
+#: aleksis/core/templates/dynamic_preferences/form.html:5
 msgid "Preferences"
 msgstr ""
 
-#: menus.py:67
+#: aleksis/core/menus.py:99
+msgid "Third-party accounts"
+msgstr ""
+
+#: aleksis/core/menus.py:108
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:5
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:6
+msgid "Authorized applications"
+msgstr ""
+
+#: aleksis/core/menus.py:119
 msgid "Admin"
 msgstr ""
 
-#: menus.py:75 models.py:602 templates/core/announcement/list.html:7
-#: templates/core/announcement/list.html:8
+#: aleksis/core/menus.py:127 aleksis/core/models.py:704
+#: aleksis/core/templates/core/announcement/list.html:7
+#: aleksis/core/templates/core/announcement/list.html:8
 msgid "Announcements"
 msgstr ""
 
-#: menus.py:86 models.py:103 templates/core/school_term/list.html:8
-#: templates/core/school_term/list.html:9
+#: aleksis/core/menus.py:138 aleksis/core/models.py:118
+#: aleksis/core/templates/core/school_term/list.html:8
+#: aleksis/core/templates/core/school_term/list.html:9
 msgid "School terms"
 msgstr ""
 
-#: menus.py:97 templates/core/dashboard_widget/list.html:8
-#: templates/core/dashboard_widget/list.html:9
+#: aleksis/core/menus.py:149
+#: aleksis/core/templates/core/dashboard_widget/list.html:8
+#: aleksis/core/templates/core/dashboard_widget/list.html:9
 msgid "Dashboard widgets"
 msgstr ""
 
-#: menus.py:108 templates/core/management/data_management.html:6
-#: templates/core/management/data_management.html:7
+#: aleksis/core/menus.py:160
+#: aleksis/core/templates/core/management/data_management.html:6
+#: aleksis/core/templates/core/management/data_management.html:7
 msgid "Data management"
 msgstr ""
 
-#: menus.py:116 templates/core/pages/system_status.html:5
-#: templates/core/pages/system_status.html:7
+#: aleksis/core/menus.py:171
+#: aleksis/core/templates/core/pages/system_status.html:5
+#: aleksis/core/templates/core/pages/system_status.html:7
 msgid "System status"
 msgstr ""
 
-#: menus.py:127
+#: aleksis/core/menus.py:182
 msgid "Impersonation"
 msgstr ""
 
-#: menus.py:135
+#: aleksis/core/menus.py:193
 msgid "Configuration"
 msgstr ""
 
-#: menus.py:146 templates/core/data_check/list.html:9
-#: templates/core/data_check/list.html:10
+#: aleksis/core/menus.py:204 aleksis/core/templates/core/data_check/list.html:9
+#: aleksis/core/templates/core/data_check/list.html:10
 msgid "Data checks"
 msgstr ""
 
-#: menus.py:152
+#: aleksis/core/menus.py:210
 msgid "Backend Admin"
 msgstr ""
 
-#: menus.py:160
-msgid "People"
+#: aleksis/core/menus.py:216
+#: aleksis/core/templates/oauth2_provider/application_detail.html:5
+#: aleksis/core/templates/oauth2_provider/application_list.html:5
+msgid "OAuth2 Applications"
 msgstr ""
 
-#: menus.py:185 models.py:812 templates/core/group_type/list.html:8
-#: templates/core/group_type/list.html:9
-msgid "Group types"
+#: aleksis/core/menus.py:229
+msgid "People"
 msgstr ""
 
-#: menus.py:196
-msgid "Persons and accounts"
+#: aleksis/core/menus.py:260 aleksis/core/models.py:958
+#: aleksis/core/templates/core/group_type/list.html:8
+#: aleksis/core/templates/core/group_type/list.html:9
+msgid "Group types"
 msgstr ""
 
-#: menus.py:207
+#: aleksis/core/menus.py:271
 msgid "Groups and child groups"
 msgstr ""
 
-#: menus.py:218 models.py:385 templates/core/additional_field/list.html:8
-#: templates/core/additional_field/list.html:9
+#: aleksis/core/menus.py:282 aleksis/core/models.py:446
+#: aleksis/core/templates/core/additional_field/list.html:8
+#: aleksis/core/templates/core/additional_field/list.html:9
 msgid "Additional fields"
 msgstr ""
 
-#: menus.py:233 templates/core/group/child_groups.html:7
-#: templates/core/group/child_groups.html:9
+#: aleksis/core/menus.py:297
+#: aleksis/core/templates/core/group/child_groups.html:7
+#: aleksis/core/templates/core/group/child_groups.html:9
 msgid "Assign child groups to groups"
 msgstr ""
 
-#: mixins.py:384
+#: aleksis/core/mixins.py:498
 msgid "Linked school term"
 msgstr ""
 
-#: models.py:43
+#: aleksis/core/models.py:58
 msgid "Boolean (Yes/No)"
 msgstr ""
 
-#: models.py:44
+#: aleksis/core/models.py:59
 msgid "Text (one line)"
 msgstr ""
 
-#: models.py:46
+#: aleksis/core/models.py:61
 msgid "Date and time"
 msgstr ""
 
-#: models.py:47
+#: aleksis/core/models.py:62
 msgid "Decimal number"
 msgstr ""
 
-#: models.py:48 models.py:157
+#: aleksis/core/models.py:63 aleksis/core/models.py:186
 msgid "E-mail address"
 msgstr ""
 
-#: models.py:49
+#: aleksis/core/models.py:64
 msgid "Integer"
 msgstr ""
 
-#: models.py:50
+#: aleksis/core/models.py:65
 msgid "IP address"
 msgstr ""
 
-#: models.py:51
+#: aleksis/core/models.py:66
 msgid "Boolean or empty (Yes/No/Neither)"
 msgstr ""
 
-#: models.py:52
+#: aleksis/core/models.py:67
 msgid "Text (multi-line)"
 msgstr ""
 
-#: models.py:54
+#: aleksis/core/models.py:69
 msgid "URL / Link"
 msgstr ""
 
-#: models.py:66 models.py:785
+#: aleksis/core/models.py:81 aleksis/core/models.py:927
 msgid "Name"
 msgstr ""
 
-#: models.py:68
+#: aleksis/core/models.py:83
 msgid "Start date"
 msgstr ""
 
-#: models.py:69
+#: aleksis/core/models.py:84
 msgid "End date"
 msgstr ""
 
-#: models.py:88
+#: aleksis/core/models.py:103
 msgid "The start date must be earlier than the end date."
 msgstr ""
 
-#: models.py:95
+#: aleksis/core/models.py:110
 msgid "There is already a school term for this time or a part of this time."
 msgstr ""
 
-#: models.py:115 models.py:744 templates/core/person/accounts.html:41
+#: aleksis/core/models.py:139 aleksis/core/models.py:876
 msgid "Person"
 msgstr ""
 
-#: models.py:118
+#: aleksis/core/models.py:142
 msgid "Can view address"
 msgstr ""
 
-#: models.py:119
+#: aleksis/core/models.py:143
 msgid "Can view contact details"
 msgstr ""
 
-#: models.py:120
+#: aleksis/core/models.py:144
 msgid "Can view photo"
 msgstr ""
 
-#: models.py:121
+#: aleksis/core/models.py:145
 msgid "Can view persons groups"
 msgstr ""
 
-#: models.py:122
+#: aleksis/core/models.py:146
 msgid "Can view personal details"
 msgstr ""
 
-#: models.py:127
+#: aleksis/core/models.py:156
 msgid "female"
 msgstr ""
 
-#: models.py:127
+#: aleksis/core/models.py:156
 msgid "male"
 msgstr ""
 
-#: models.py:135
+#: aleksis/core/models.py:164
 msgid "Linked user"
 msgstr ""
 
-#: models.py:137
+#: aleksis/core/models.py:166
 msgid "Is person active?"
 msgstr ""
 
-#: models.py:139
+#: aleksis/core/models.py:168
 msgid "First name"
 msgstr ""
 
-#: models.py:140
+#: aleksis/core/models.py:169
 msgid "Last name"
 msgstr ""
 
-#: models.py:142
+#: aleksis/core/models.py:171
 msgid "Additional name(s)"
 msgstr ""
 
-#: models.py:146 models.py:354
+#: aleksis/core/models.py:175 aleksis/core/models.py:415
 msgid "Short name"
 msgstr ""
 
-#: models.py:149
+#: aleksis/core/models.py:178
 msgid "Street"
 msgstr ""
 
-#: models.py:150
+#: aleksis/core/models.py:179
 msgid "Street number"
 msgstr ""
 
-#: models.py:151
+#: aleksis/core/models.py:180
 msgid "Postal code"
 msgstr ""
 
-#: models.py:152
+#: aleksis/core/models.py:181
 msgid "Place"
 msgstr ""
 
-#: models.py:154
+#: aleksis/core/models.py:183
 msgid "Home phone"
 msgstr ""
 
-#: models.py:155
+#: aleksis/core/models.py:184
 msgid "Mobile phone"
 msgstr ""
 
-#: models.py:159
+#: aleksis/core/models.py:188
 msgid "Date of birth"
 msgstr ""
 
-#: models.py:160
+#: aleksis/core/models.py:189
 msgid "Sex"
 msgstr ""
 
-#: models.py:162
+#: aleksis/core/models.py:191
 msgid "Photo"
 msgstr ""
 
-#: models.py:166 templates/core/person/full.html:137
+#: aleksis/core/models.py:195 aleksis/core/templates/core/person/full.html:137
 msgid "Guardians / Parents"
 msgstr ""
 
-#: models.py:173
+#: aleksis/core/models.py:202
 msgid "Primary group"
 msgstr ""
 
-#: models.py:176 models.py:461 models.py:485 models.py:570 models.py:805
-#: templates/core/person/full.html:120
+#: aleksis/core/models.py:205 aleksis/core/models.py:563
+#: aleksis/core/models.py:587 aleksis/core/models.py:672
+#: aleksis/core/models.py:951 aleksis/core/templates/core/person/full.html:120
 msgid "Description"
 msgstr ""
 
-#: models.py:313
+#: aleksis/core/models.py:370
 msgid "Title of field"
 msgstr ""
 
-#: models.py:315
+#: aleksis/core/models.py:372
 msgid "Type of field"
 msgstr ""
 
-#: models.py:322
+#: aleksis/core/models.py:379
 msgid "Addtitional field for groups"
 msgstr ""
 
-#: models.py:323
+#: aleksis/core/models.py:380
 msgid "Addtitional fields for groups"
 msgstr ""
 
-#: models.py:337
+#: aleksis/core/models.py:397
 msgid "Group"
 msgstr ""
 
-#: models.py:340
+#: aleksis/core/models.py:400
 msgid "Can assign child groups to groups"
 msgstr ""
 
-#: models.py:341
+#: aleksis/core/models.py:401
 msgid "Can view statistics about group."
 msgstr ""
 
-#: models.py:352
+#: aleksis/core/models.py:413
 msgid "Long name"
 msgstr ""
 
-#: models.py:362 templates/core/group/full.html:85
+#: aleksis/core/models.py:423 aleksis/core/templates/core/group/full.html:85
 msgid "Members"
 msgstr ""
 
-#: models.py:365 templates/core/group/full.html:82
+#: aleksis/core/models.py:426 aleksis/core/templates/core/group/full.html:82
 msgid "Owners"
 msgstr ""
 
-#: models.py:372 templates/core/group/full.html:55
+#: aleksis/core/models.py:433 aleksis/core/templates/core/group/full.html:55
 msgid "Parent groups"
 msgstr ""
 
-#: models.py:380
+#: aleksis/core/models.py:441
 msgid "Type of group"
 msgstr ""
 
-#: models.py:457
+#: aleksis/core/models.py:559
 msgid "User"
 msgstr ""
 
-#: models.py:460 models.py:484 models.py:569
-#: templates/core/announcement/list.html:18
+#: aleksis/core/models.py:562 aleksis/core/models.py:586
+#: aleksis/core/models.py:671
+#: aleksis/core/templates/core/announcement/list.html:18
 msgid "Title"
 msgstr ""
 
-#: models.py:463
+#: aleksis/core/models.py:565
 msgid "Application"
 msgstr ""
 
-#: models.py:469
+#: aleksis/core/models.py:571
 msgid "Activity"
 msgstr ""
 
-#: models.py:470
+#: aleksis/core/models.py:572
 msgid "Activities"
 msgstr ""
 
-#: models.py:476
+#: aleksis/core/models.py:578
 msgid "Sender"
 msgstr ""
 
-#: models.py:481
+#: aleksis/core/models.py:583
 msgid "Recipient"
 msgstr ""
 
-#: models.py:486 models.py:786
+#: aleksis/core/models.py:588 aleksis/core/models.py:928
 msgid "Link"
 msgstr ""
 
-#: models.py:488
+#: aleksis/core/models.py:590
 msgid "Read"
 msgstr ""
 
-#: models.py:489
+#: aleksis/core/models.py:591
 msgid "Sent"
 msgstr ""
 
-#: models.py:502
+#: aleksis/core/models.py:604
 msgid "Notification"
 msgstr ""
 
-#: models.py:503
-msgid "Notifications"
-msgstr ""
-
-#: models.py:571
+#: aleksis/core/models.py:673
 msgid "Link to detailed view"
 msgstr ""
 
-#: models.py:574
+#: aleksis/core/models.py:676
 msgid "Date and time from when to show"
 msgstr ""
 
-#: models.py:577
+#: aleksis/core/models.py:679
 msgid "Date and time until when to show"
 msgstr ""
 
-#: models.py:601
+#: aleksis/core/models.py:703
 msgid "Announcement"
 msgstr ""
 
-#: models.py:639
+#: aleksis/core/models.py:741
 msgid "Announcement recipient"
 msgstr ""
 
-#: models.py:640
+#: aleksis/core/models.py:742
 msgid "Announcement recipients"
 msgstr ""
 
-#: models.py:690
+#: aleksis/core/models.py:797
 msgid "Widget Title"
 msgstr ""
 
-#: models.py:691
+#: aleksis/core/models.py:798
 msgid "Activate Widget"
 msgstr ""
 
-#: models.py:694
+#: aleksis/core/models.py:799
+msgid "Widget is broken"
+msgstr ""
+
+#: aleksis/core/models.py:802
 msgid "Size on mobile devices"
 msgstr ""
 
-#: models.py:695
+#: aleksis/core/models.py:803
 msgid "<= 600 px, 12 columns"
 msgstr ""
 
-#: models.py:700
+#: aleksis/core/models.py:808
 msgid "Size on tablet devices"
 msgstr ""
 
-#: models.py:701
+#: aleksis/core/models.py:809
 msgid "> 600 px, 12 columns"
 msgstr ""
 
-#: models.py:706
+#: aleksis/core/models.py:814
 msgid "Size on desktop devices"
 msgstr ""
 
-#: models.py:707
+#: aleksis/core/models.py:815
 msgid "> 992 px, 12 columns"
 msgstr ""
 
-#: models.py:712
+#: aleksis/core/models.py:820
 msgid "Size on large desktop devices"
 msgstr ""
 
-#: models.py:713
+#: aleksis/core/models.py:821
 msgid "> 1200 px>, 12 columns"
 msgstr ""
 
-#: models.py:734
+#: aleksis/core/models.py:852
 msgid "Can edit default dashboard"
 msgstr ""
 
-#: models.py:735
+#: aleksis/core/models.py:853
 msgid "Dashboard Widget"
 msgstr ""
 
-#: models.py:736
+#: aleksis/core/models.py:854
 msgid "Dashboard Widgets"
 msgstr ""
 
-#: models.py:741
+#: aleksis/core/models.py:860
+msgid "URL"
+msgstr ""
+
+#: aleksis/core/models.py:861
+msgid "Icon URL"
+msgstr ""
+
+#: aleksis/core/models.py:867
+msgid "External link widget"
+msgstr ""
+
+#: aleksis/core/models.py:868
+msgid "External link widgets"
+msgstr ""
+
+#: aleksis/core/models.py:873
 msgid "Dashboard widget"
 msgstr ""
 
-#: models.py:746
+#: aleksis/core/models.py:878
 msgid "Order"
 msgstr ""
 
-#: models.py:747
+#: aleksis/core/models.py:879
 msgid "Part of the default dashboard"
 msgstr ""
 
-#: models.py:755
+#: aleksis/core/models.py:894
 msgid "Dashboard widget order"
 msgstr ""
 
-#: models.py:756
+#: aleksis/core/models.py:895
 msgid "Dashboard widget orders"
 msgstr ""
 
-#: models.py:762
+#: aleksis/core/models.py:901
 msgid "Menu ID"
 msgstr ""
 
-#: models.py:775
+#: aleksis/core/models.py:914
 msgid "Custom menu"
 msgstr ""
 
-#: models.py:776
+#: aleksis/core/models.py:915
 msgid "Custom menus"
 msgstr ""
 
-#: models.py:783
+#: aleksis/core/models.py:925
 msgid "Menu"
 msgstr ""
 
-#: models.py:787
+#: aleksis/core/models.py:929
 msgid "Icon"
 msgstr ""
 
-#: models.py:793
+#: aleksis/core/models.py:935
 msgid "Custom menu item"
 msgstr ""
 
-#: models.py:794
+#: aleksis/core/models.py:936
 msgid "Custom menu items"
 msgstr ""
 
-#: models.py:804
+#: aleksis/core/models.py:950
 msgid "Title of type"
 msgstr ""
 
-#: models.py:811 templates/core/group/full.html:47
+#: aleksis/core/models.py:957 aleksis/core/templates/core/group/full.html:47
 msgid "Group type"
 msgstr ""
 
-#: models.py:821
+#: aleksis/core/models.py:971
 msgid "Can view system status"
 msgstr ""
 
-#: models.py:822
-msgid "Can link persons to accounts"
-msgstr ""
-
-#: models.py:823
+#: aleksis/core/models.py:972
 msgid "Can manage data"
 msgstr ""
 
-#: models.py:824
+#: aleksis/core/models.py:973
 msgid "Can impersonate"
 msgstr ""
 
-#: models.py:825
+#: aleksis/core/models.py:974
 msgid "Can use search"
 msgstr ""
 
-#: models.py:826
+#: aleksis/core/models.py:975
 msgid "Can change site preferences"
 msgstr ""
 
-#: models.py:827
+#: aleksis/core/models.py:976
 msgid "Can change person preferences"
 msgstr ""
 
-#: models.py:828
+#: aleksis/core/models.py:977
 msgid "Can change group preferences"
 msgstr ""
 
-#: models.py:864
+#: aleksis/core/models.py:978
+msgid "Can add oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:979
+msgid "Can list oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:980
+msgid "Can view oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:981
+msgid "Can update oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:982
+msgid "Can delete oauth applications"
+msgstr ""
+
+#: aleksis/core/models.py:983
+msgid "Can test PDF generation"
+msgstr ""
+
+#: aleksis/core/models.py:1019
 msgid "Related data check task"
 msgstr ""
 
-#: models.py:872
+#: aleksis/core/models.py:1027
 msgid "Issue solved"
 msgstr ""
 
-#: models.py:873
+#: aleksis/core/models.py:1028
 msgid "Notification sent"
 msgstr ""
 
-#: models.py:886
+#: aleksis/core/models.py:1041
 msgid "Data check result"
 msgstr ""
 
-#: models.py:887
+#: aleksis/core/models.py:1042
 msgid "Data check results"
 msgstr ""
 
-#: models.py:889
+#: aleksis/core/models.py:1044
 msgid "Can run data checks"
 msgstr ""
 
-#: models.py:890
+#: aleksis/core/models.py:1045
 msgid "Can solve data check problems"
 msgstr ""
 
-#: preferences.py:27
+#: aleksis/core/models.py:1060
+msgid "Owner"
+msgstr ""
+
+#: aleksis/core/models.py:1064
+msgid "File expires at"
+msgstr ""
+
+#: aleksis/core/models.py:1066
+msgid "Generated HTML file"
+msgstr ""
+
+#: aleksis/core/models.py:1068
+msgid "Generated PDF file"
+msgstr ""
+
+#: aleksis/core/models.py:1075
+msgid "PDF file"
+msgstr ""
+
+#: aleksis/core/models.py:1076
+msgid "PDF files"
+msgstr ""
+
+#: aleksis/core/models.py:1081
+msgid "Task result"
+msgstr ""
+
+#: aleksis/core/models.py:1084
+msgid "Task user"
+msgstr ""
+
+#: aleksis/core/models.py:1096
+msgid "Task user assignment"
+msgstr ""
+
+#: aleksis/core/models.py:1097
+msgid "Task user assignments"
+msgstr ""
+
+#: aleksis/core/preferences.py:22
+msgid "General"
+msgstr ""
+
+#: aleksis/core/preferences.py:23
+msgid "School"
+msgstr ""
+
+#: aleksis/core/preferences.py:24
+msgid "Theme"
+msgstr ""
+
+#: aleksis/core/preferences.py:25
+msgid "Mail"
+msgstr ""
+
+#: aleksis/core/preferences.py:27
+msgid "Footer"
+msgstr ""
+
+#: aleksis/core/preferences.py:28
+msgid "Accounts"
+msgstr ""
+
+#: aleksis/core/preferences.py:29
 msgid "Authentication"
 msgstr ""
 
-#: preferences.py:28
+#: aleksis/core/preferences.py:30
 msgid "Internationalisation"
 msgstr ""
 
-#: preferences.py:37
+#: aleksis/core/preferences.py:41
 msgid "Site title"
 msgstr ""
 
-#: preferences.py:46
+#: aleksis/core/preferences.py:52
 msgid "Site description"
 msgstr ""
 
-#: preferences.py:55
+#: aleksis/core/preferences.py:63
 msgid "Primary colour"
 msgstr ""
 
-#: preferences.py:64
+#: aleksis/core/preferences.py:75
 msgid "Secondary colour"
 msgstr ""
 
-#: preferences.py:72
+#: aleksis/core/preferences.py:86
 msgid "Logo"
 msgstr ""
 
-#: preferences.py:80
+#: aleksis/core/preferences.py:96
 msgid "Favicon"
 msgstr ""
 
-#: preferences.py:88
+#: aleksis/core/preferences.py:106
 msgid "PWA-Icon"
 msgstr ""
 
-#: preferences.py:97
+#: aleksis/core/preferences.py:117
 msgid "Mail out name"
 msgstr ""
 
-#: preferences.py:106
+#: aleksis/core/preferences.py:128
 msgid "Mail out address"
 msgstr ""
 
-#: preferences.py:116
+#: aleksis/core/preferences.py:140
 msgid "Link to privacy policy"
 msgstr ""
 
-#: preferences.py:126
+#: aleksis/core/preferences.py:152
 msgid "Link to imprint"
 msgstr ""
 
-#: preferences.py:136
+#: aleksis/core/preferences.py:164
 msgid "Name format for addressing"
 msgstr ""
 
-#: preferences.py:150
+#: aleksis/core/preferences.py:180
 msgid "Channels to use for notifications"
 msgstr ""
 
-#: preferences.py:160
+#: aleksis/core/preferences.py:192
 msgid "Regular expression to match primary group, e.g. '^Class .*'"
 msgstr ""
 
-#: preferences.py:169
+#: aleksis/core/preferences.py:203
 msgid "Field on person to match primary group against"
 msgstr ""
 
-#: preferences.py:181
+#: aleksis/core/preferences.py:215
+msgid "Automatically create new persons for new users"
+msgstr ""
+
+#: aleksis/core/preferences.py:224
+msgid "Automatically link existing persons to new users by their e-mail address"
+msgstr ""
+
+#: aleksis/core/preferences.py:235
 msgid "Display name of the school"
 msgstr ""
 
-#: preferences.py:190
+#: aleksis/core/preferences.py:246
 msgid "Official name of the school, e.g. as given by supervisory authority"
 msgstr ""
 
-#: preferences.py:198
-msgid "Enabled custom authentication backends"
+#: aleksis/core/preferences.py:254
+msgid "Allow users to change their passwords"
+msgstr ""
+
+#: aleksis/core/preferences.py:262
+msgid "Enable signup"
 msgstr ""
 
-#: preferences.py:211
+#: aleksis/core/preferences.py:273
 msgid "Available languages"
 msgstr ""
 
-#: preferences.py:223
+#: aleksis/core/preferences.py:285
 msgid "Send emails if data checks detect problems"
 msgstr ""
 
-#: preferences.py:234
+#: aleksis/core/preferences.py:296
 msgid "Email recipients for data checks problem emails"
 msgstr ""
 
-#: preferences.py:245
+#: aleksis/core/preferences.py:307
 msgid "Email recipient groups for data checks problem emails"
 msgstr ""
 
-#: settings.py:322
-msgid "English"
+#: aleksis/core/preferences.py:316
+msgid "Show dashboard to users without login"
 msgstr ""
 
-#: settings.py:323
-msgid "German"
+#: aleksis/core/preferences.py:325
+msgid "Allow users to edit their dashboard"
 msgstr ""
 
-#: settings.py:324
-msgid "French"
+#: aleksis/core/preferences.py:336
+msgid "Fields on person model which are editable by themselves."
 msgstr ""
 
-#: settings.py:325
-msgid "Norwegian (bokmål)"
+#: aleksis/core/preferences.py:350
+msgid "Editable fields on person model which should trigger a notification on change"
 msgstr ""
 
-#: tables.py:19 templates/core/announcement/list.html:36
-#: templates/core/group/full.html:24 templates/core/person/full.html:23
+#: aleksis/core/preferences.py:363
+msgid "Contact for notification if a person changes their data"
+msgstr ""
+
+#: aleksis/core/preferences.py:373
+msgid "PDF file expiration duration"
+msgstr ""
+
+#: aleksis/core/preferences.py:374
+msgid "in minutes"
+msgstr ""
+
+#: aleksis/core/preferences.py:384
+msgid "Automatically update the dashboard and its widgets"
+msgstr ""
+
+#: aleksis/core/preferences.py:394
+msgid "Automatically update the dashboard and its widgets sitewide"
+msgstr ""
+
+#: aleksis/core/settings.py:452
+msgid "English"
+msgstr ""
+
+#: aleksis/core/settings.py:453
+msgid "German"
+msgstr ""
+
+#: aleksis/core/tables.py:19
+#: aleksis/core/templates/core/announcement/list.html:36
+#: aleksis/core/templates/core/group/full.html:24
+#: aleksis/core/templates/core/person/full.html:23
+#: aleksis/core/templates/oauth2_provider/application_detail.html:17
 msgid "Edit"
 msgstr ""
 
-#: tables.py:21 tables.py:89 templates/core/announcement/list.html:22
+#: aleksis/core/tables.py:21 aleksis/core/tables.py:89
+#: aleksis/core/templates/core/announcement/list.html:22
 msgid "Actions"
 msgstr ""
 
-#: tables.py:56 tables.py:57 tables.py:71 tables.py:87
-#: templates/core/announcement/list.html:42 templates/core/group/full.html:31
-#: templates/core/pages/delete.html:22 templates/core/person/full.html:30
+#: aleksis/core/tables.py:56 aleksis/core/tables.py:57
+#: aleksis/core/tables.py:71 aleksis/core/tables.py:87
+#: aleksis/core/templates/core/announcement/list.html:42
+#: aleksis/core/templates/core/group/full.html:31
+#: aleksis/core/templates/core/pages/delete.html:22
+#: aleksis/core/templates/core/person/full.html:30
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:20
+#: aleksis/core/templates/oauth2_provider/application_detail.html:21
 msgid "Delete"
 msgstr ""
 
-#: templates/403.html:14 templates/404.html:10 templates/500.html:10
+#: aleksis/core/templates/403.html:14 aleksis/core/templates/404.html:10
+#: aleksis/core/templates/500.html:10
+#: aleksis/core/templates/oauth2_provider/authorize.html:47
+#: aleksis/core/templates/oauth2_provider/authorized-oob.html:24
 msgid "Error"
 msgstr ""
 
-#: templates/403.html:14
+#: aleksis/core/templates/403.html:14
 msgid ""
 "You are not allowed to access the requested page or\n"
 "          object."
 msgstr ""
 
-#: templates/403.html:19 templates/404.html:17
+#: aleksis/core/templates/403.html:19 aleksis/core/templates/404.html:17
 msgid ""
 "\n"
 "            If you think this is an error in AlekSIS, please contact your site\n"
@@ -840,13 +1131,13 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/404.html:10
+#: aleksis/core/templates/404.html:10
 msgid ""
 "The requested page or object was not\n"
 "          found."
 msgstr ""
 
-#: templates/404.html:13
+#: aleksis/core/templates/404.html:13
 msgid ""
 "\n"
 "            If you were redirected by a link on an external page,\n"
@@ -854,13 +1145,13 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/500.html:10
+#: aleksis/core/templates/500.html:10
 msgid ""
 "An unexpected error has\n"
 "          occured."
 msgstr ""
 
-#: templates/500.html:13
+#: aleksis/core/templates/500.html:13
 msgid ""
 "\n"
 "            Your site administrators will automatically be notified about this\n"
@@ -868,167 +1159,394 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/503.html:10
+#: aleksis/core/templates/503.html:10
 msgid ""
 "The maintenance mode is currently enabled. Please try again\n"
 "          later."
 msgstr ""
 
-#: templates/503.html:13
+#: aleksis/core/templates/503.html:13
 msgid ""
 "\n"
 "            This page is currently unavailable. If this error persists, contact your site administrators:\n"
 "          "
 msgstr ""
 
-#: templates/core/additional_field/edit.html:6
-#: templates/core/additional_field/edit.html:7
+#: aleksis/core/templates/account/account_inactive.html:5
+#: aleksis/core/templates/account/account_inactive.html:6
+msgid "Account inactive"
+msgstr ""
+
+#: aleksis/core/templates/account/account_inactive.html:13
+msgid "Account inactive."
+msgstr ""
+
+#: aleksis/core/templates/account/account_inactive.html:15
+msgid ""
+"\n"
+"            This account is currently inactive. If you think this is an\n"
+"            error, please contact one of your site administrators.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/email/base_message.txt:5
+msgid "Hello!"
+msgstr ""
+
+#: aleksis/core/templates/account/email/base_message.txt:9
+#: aleksis/core/templates/templated_email/notification.email:22
+#: aleksis/core/templates/templated_email/notification.email:46
+msgid "Your AlekSIS team"
+msgstr ""
+
+#: aleksis/core/templates/account/email_confirm.html:5
+#: aleksis/core/templates/account/email_confirm.html:6
+#: aleksis/core/templates/account/email_confirm.html:17
+msgid "Confirm"
+msgstr ""
+
+#: aleksis/core/templates/account/email_confirm.html:12
+#, python-format
+msgid "Please confirm that <a href=\"mailto:%(email)s\">%(email)s</a> is an e-mail address for user %(user_display)s."
+msgstr ""
+
+#: aleksis/core/templates/account/email_confirm.html:25
+#, python-format
+msgid "This e-mail confirmation link expired or is invalid. Please <a href=\"%(email_url)s\">issue a new e-mail confirmation request</a>."
+msgstr ""
+
+#: aleksis/core/templates/account/password_change.html:12
+msgid "Forgot your current password? Click here to reset it:"
+msgstr ""
+
+#: aleksis/core/templates/account/password_change.html:12
+msgid "Forgot Password?"
+msgstr ""
+
+#: aleksis/core/templates/account/password_change_disabled.html:5
+#: aleksis/core/templates/account/password_change_disabled.html:6
+msgid "Changing of password disabled"
+msgstr ""
+
+#: aleksis/core/templates/account/password_change_disabled.html:13
+msgid "Changing of password disabled."
+msgstr ""
+
+#: aleksis/core/templates/account/password_change_disabled.html:15
+msgid ""
+"\n"
+"            Users are not allowed to edit their own passwords. If you think\n"
+"            this is an error please contact one of your site administrators.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset.html:5
+#: aleksis/core/templates/account/password_reset.html:15
+#: aleksis/core/templates/account/password_reset.html:23
+#: aleksis/core/templates/account/password_reset_done.html:5
+#: aleksis/core/templates/account/verification_email_required.html:5
+#: aleksis/core/templates/account/verification_email_required.html:6
+#: aleksis/core/templates/two_factor/core/login.html:81
+msgid "Reset password"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset.html:17
+msgid "Forgotten your password? Enter your e-mail address below, and we'll send you an e-mail allowing you to reset it."
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset.html:30
+msgid ""
+"Please contact one of your site administrators, if you\n"
+"        have any trouble resetting your password:"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_done.html:15
+msgid "Password reset mail sent"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_done.html:18
+#: aleksis/core/templates/account/verification_email_required.html:16
+msgid ""
+"\n"
+"            We have sent you an e-mail. Please contact one of your site\n"
+"            administrators if you do not receive it within a few minutes.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:15
+msgid "Bad token"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:19
+#, python-format
+msgid ""
+"\n"
+"              The password reset link was invalid, possibly because it has already been used. Please request a <a href=\"%(passwd_reset_url)s\"\n"
+"              class=\"blue-text text-lighten-2\">new password reset</a>.\n"
+"            "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:25
+msgid ""
+"\n"
+"              If this issue persists, please contact one of your site\n"
+"              administrators\n"
+"            "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:56
+#: aleksis/core/templates/account/password_reset_from_key_done.html:15
+msgid ""
+"\n"
+"            Your password is now changed!\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key.html:61
+msgid "Back to login"
+msgstr ""
+
+#: aleksis/core/templates/account/password_reset_from_key_done.html:13
+msgid "Password changed!"
+msgstr ""
+
+#: aleksis/core/templates/account/password_set.html:5
+#: aleksis/core/templates/account/password_set.html:6
+#: aleksis/core/templates/account/password_set.html:12
+msgid "Set password"
+msgstr ""
+
+#: aleksis/core/templates/account/signup.html:5
+#: aleksis/core/templates/account/signup.html:6
+#: aleksis/core/templates/socialaccount/signup.html:5
+#: aleksis/core/templates/socialaccount/signup.html:6
+msgid "Signup"
+msgstr ""
+
+#: aleksis/core/templates/account/signup.html:12
+#, python-format
+msgid "Already have an account? Then please <a href=\"%(login_url)s\">sign in</a>."
+msgstr ""
+
+#: aleksis/core/templates/account/signup_closed.html:5
+#: aleksis/core/templates/account/signup_closed.html:6
+msgid "Signup closed"
+msgstr ""
+
+#: aleksis/core/templates/account/signup_closed.html:13
+msgid "Signup closed."
+msgstr ""
+
+#: aleksis/core/templates/account/signup_closed.html:15
+msgid ""
+"\n"
+"            This sign up is currently closed. If you think this is an\n"
+"            error, please contact one of your site administrators.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/verification_email_required.html:14
+msgid "Password reset mail sent!"
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:5
+#: aleksis/core/templates/account/verification_sent.html:6
+msgid "Verify your email address"
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:14
+msgid "Verify your email!"
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:16
+msgid ""
+"\n"
+"            This part of the site requires us to verify that you are who you claim to be.\n"
+"            For this purpose, we require that you verify ownership of your e-mail address.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:22
+msgid ""
+"\n"
+"            We have sent an e-mail to you for verification.\n"
+"            Please click on the link inside this e-mail. Please\n"
+"            contact us if you do not receive it within a few minutes.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/account/verification_sent.html:30
+#, python-format
+msgid "<strong>Note:</strong> you can still <a href=\"%(email_url)s\">change your e-mail address</a>"
+msgstr ""
+
+#: aleksis/core/templates/core/additional_field/edit.html:6
+#: aleksis/core/templates/core/additional_field/edit.html:7
 msgid "Edit additional field"
 msgstr ""
 
-#: templates/core/additional_field/list.html:14
+#: aleksis/core/templates/core/additional_field/list.html:14
 msgid "Create additional field"
 msgstr ""
 
-#: templates/core/announcement/form.html:14
-#: templates/core/announcement/form.html:21
+#: aleksis/core/templates/core/announcement/form.html:14
+#: aleksis/core/templates/core/announcement/form.html:21
 msgid "Edit announcement"
 msgstr ""
 
-#: templates/core/announcement/form.html:16
+#: aleksis/core/templates/core/announcement/form.html:16
 msgid "Publish announcement"
 msgstr ""
 
-#: templates/core/announcement/form.html:23
-#: templates/core/announcement/list.html:13
+#: aleksis/core/templates/core/announcement/form.html:23
+#: aleksis/core/templates/core/announcement/list.html:13
 msgid "Publish new announcement"
 msgstr ""
 
-#: templates/core/announcement/form.html:34
+#: aleksis/core/templates/core/announcement/form.html:34
 msgid "Save und publish announcement"
 msgstr ""
 
-#: templates/core/announcement/list.html:19
+#: aleksis/core/templates/core/announcement/list.html:19
 msgid "Valid from"
 msgstr ""
 
-#: templates/core/announcement/list.html:20
+#: aleksis/core/templates/core/announcement/list.html:20
 msgid "Valid until"
 msgstr ""
 
-#: templates/core/announcement/list.html:21
+#: aleksis/core/templates/core/announcement/list.html:21
 msgid "Recipients"
 msgstr ""
 
-#: templates/core/announcement/list.html:50
+#: aleksis/core/templates/core/announcement/list.html:50
 msgid "There are no announcements."
 msgstr ""
 
-#: templates/core/base.html:54
+#: aleksis/core/templates/core/base.html:60
 msgid "Logged in as"
 msgstr ""
 
-#: templates/core/base.html:147
+#: aleksis/core/templates/core/base.html:154
 msgid "About AlekSIS — The Free School Information System"
 msgstr ""
 
-#: templates/core/base.html:155
+#: aleksis/core/templates/core/base.html:162
 msgid "Impress"
 msgstr ""
 
-#: templates/core/base.html:163
+#: aleksis/core/templates/core/base.html:170
 msgid "Privacy Policy"
 msgstr ""
 
-#: templates/core/base_print.html:64
+#: aleksis/core/templates/core/base_print.html:72
 msgid "Powered by AlekSIS"
 msgstr ""
 
-#: templates/core/dashboard_widget/create.html:8
-#: templates/core/dashboard_widget/create.html:12
+#: aleksis/core/templates/core/dashboard_widget/create.html:8
+#: aleksis/core/templates/core/dashboard_widget/create.html:12
 #, python-format
 msgid "Create %(widget)s"
 msgstr ""
 
-#: templates/core/dashboard_widget/edit.html:8
-#: templates/core/dashboard_widget/edit.html:12
+#: aleksis/core/templates/core/dashboard_widget/dashboardwidget_broken.html:10
+msgid "This widget is currently not available."
+msgstr ""
+
+#: aleksis/core/templates/core/dashboard_widget/dashboardwidget_broken.html:14
+#, python-format
+msgid ""
+"\n"
+"            There is a problem getting the widget \"%(title)s\".\n"
+"            There is no need for you to take any action.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/core/dashboard_widget/edit.html:8
+#: aleksis/core/templates/core/dashboard_widget/edit.html:12
 #, python-format
 msgid "Edit %(widget)s"
 msgstr ""
 
-#: templates/core/dashboard_widget/list.html:17
+#: aleksis/core/templates/core/dashboard_widget/list.html:15
+msgid "Create dashboard widget"
+msgstr ""
+
+#: aleksis/core/templates/core/dashboard_widget/list.html:22
 #, python-format
 msgid "Create %(name)s"
 msgstr ""
 
-#: templates/core/dashboard_widget/list.html:25
-#: templates/core/edit_dashboard.html:8 templates/core/edit_dashboard.html:15
+#: aleksis/core/templates/core/dashboard_widget/list.html:32
+#: aleksis/core/templates/core/edit_dashboard.html:8
+#: aleksis/core/templates/core/edit_dashboard.html:15
 msgid "Edit default dashboard"
 msgstr ""
 
-#: templates/core/data_check/list.html:15
+#: aleksis/core/templates/core/data_check/list.html:15
 msgid "Check data again"
 msgstr ""
 
-#: templates/core/data_check/list.html:22
+#: aleksis/core/templates/core/data_check/list.html:22
 msgid "The system detected some problems with your data."
 msgstr ""
 
-#: templates/core/data_check/list.html:23
+#: aleksis/core/templates/core/data_check/list.html:23
 msgid ""
 "Please go through all data and check whether some extra action is\n"
 "          needed."
 msgstr ""
 
-#: templates/core/data_check/list.html:31
+#: aleksis/core/templates/core/data_check/list.html:31
 msgid "Everything is fine."
 msgstr ""
 
-#: templates/core/data_check/list.html:32
+#: aleksis/core/templates/core/data_check/list.html:32
 msgid "The system hasn't detected any problems with your data."
 msgstr ""
 
-#: templates/core/data_check/list.html:40
+#: aleksis/core/templates/core/data_check/list.html:40
 msgid "Detected problems"
 msgstr ""
 
-#: templates/core/data_check/list.html:45
+#: aleksis/core/templates/core/data_check/list.html:45
 msgid "Affected object"
 msgstr ""
 
-#: templates/core/data_check/list.html:46
+#: aleksis/core/templates/core/data_check/list.html:46
 msgid "Detected problem"
 msgstr ""
 
-#: templates/core/data_check/list.html:47
+#: aleksis/core/templates/core/data_check/list.html:47
 msgid "Show details"
 msgstr ""
 
-#: templates/core/data_check/list.html:48
+#: aleksis/core/templates/core/data_check/list.html:48
 msgid "Options to solve the problem"
 msgstr ""
 
-#: templates/core/data_check/list.html:62
+#: aleksis/core/templates/core/data_check/list.html:62
 msgid "Show object"
 msgstr ""
 
-#: templates/core/data_check/list.html:84
+#: aleksis/core/templates/core/data_check/list.html:84
 msgid "Registered checks"
 msgstr ""
 
-#: templates/core/data_check/list.html:88
+#: aleksis/core/templates/core/data_check/list.html:88
 msgid ""
 "\n"
 "            The system will check for the following problems:\n"
 "          "
 msgstr ""
 
-#: templates/core/edit_dashboard.html:6 templates/core/edit_dashboard.html:13
-#: templates/core/index.html:14
+#: aleksis/core/templates/core/edit_dashboard.html:6
+#: aleksis/core/templates/core/edit_dashboard.html:13
+#: aleksis/core/templates/core/index.html:17
 msgid "Edit dashboard"
 msgstr ""
 
-#: templates/core/edit_dashboard.html:24
+#: aleksis/core/templates/core/edit_dashboard.html:24
 msgid ""
 "\n"
 "          On this page you can arrange your personal dashboard. You can drag any items from \"Available widgets\" to \"Your\n"
@@ -1037,7 +1555,7 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/core/edit_dashboard.html:30
+#: aleksis/core/templates/core/edit_dashboard.html:30
 msgid ""
 "\n"
 "          On this page you can arrange the default dashboard which is shown when a user doesn't arrange his own\n"
@@ -1046,19 +1564,19 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/core/edit_dashboard.html:48
+#: aleksis/core/templates/core/edit_dashboard.html:48
 msgid "Available widgets"
 msgstr ""
 
-#: templates/core/edit_dashboard.html:57
+#: aleksis/core/templates/core/edit_dashboard.html:57
 msgid "Your dashboard"
 msgstr ""
 
-#: templates/core/edit_dashboard.html:59
+#: aleksis/core/templates/core/edit_dashboard.html:59
 msgid "Default dashboard"
 msgstr ""
 
-#: templates/core/group/child_groups.html:18
+#: aleksis/core/templates/core/group/child_groups.html:18
 msgid ""
 "\n"
 "          You can use this to assign child groups to groups. Please use the filters below to select groups you want to\n"
@@ -1066,38 +1584,38 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/core/group/child_groups.html:31
+#: aleksis/core/templates/core/group/child_groups.html:31
 msgid "Update selection"
 msgstr ""
 
-#: templates/core/group/child_groups.html:35
+#: aleksis/core/templates/core/group/child_groups.html:35
 msgid "Clear all filters"
 msgstr ""
 
-#: templates/core/group/child_groups.html:39
+#: aleksis/core/templates/core/group/child_groups.html:39
 msgid "Currently selected groups"
 msgstr ""
 
-#: templates/core/group/child_groups.html:52
+#: aleksis/core/templates/core/group/child_groups.html:52
 msgid "Start assigning child groups for this groups"
 msgstr ""
 
-#: templates/core/group/child_groups.html:61
+#: aleksis/core/templates/core/group/child_groups.html:61
 msgid ""
 "\n"
 "            Please select some groups in order to go on with assigning.\n"
 "          "
 msgstr ""
 
-#: templates/core/group/child_groups.html:72
+#: aleksis/core/templates/core/group/child_groups.html:72
 msgid "Current group:"
 msgstr ""
 
-#: templates/core/group/child_groups.html:78
+#: aleksis/core/templates/core/group/child_groups.html:78
 msgid "Please be careful!"
 msgstr ""
 
-#: templates/core/group/child_groups.html:79
+#: aleksis/core/templates/core/group/child_groups.html:79
 msgid ""
 "\n"
 "            If you click \"Back\" or \"Next\" the current group assignments are not saved.\n"
@@ -1106,127 +1624,135 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/core/group/child_groups.html:93
-#: templates/core/group/child_groups.html:128
-#: templates/two_factor/_wizard_actions.html:15
-#: templates/two_factor/_wizard_actions.html:20
+#: aleksis/core/templates/core/group/child_groups.html:93
+#: aleksis/core/templates/core/group/child_groups.html:128
+#: aleksis/core/templates/oauth2_provider/application_detail.html:9
+#: aleksis/core/templates/two_factor/_wizard_actions.html:15
+#: aleksis/core/templates/two_factor/_wizard_actions.html:20
 msgid "Back"
 msgstr ""
 
-#: templates/core/group/child_groups.html:99
-#: templates/core/group/child_groups.html:134
-#: templates/two_factor/_wizard_actions.html:26
+#: aleksis/core/templates/core/group/child_groups.html:99
+#: aleksis/core/templates/core/group/child_groups.html:134
+#: aleksis/core/templates/two_factor/_wizard_actions.html:26
 msgid "Next"
 msgstr ""
 
-#: templates/core/group/child_groups.html:106
-#: templates/core/group/child_groups.html:141
-#: templates/core/partials/save_button.html:3
+#: aleksis/core/templates/core/group/child_groups.html:106
+#: aleksis/core/templates/core/group/child_groups.html:141
+#: aleksis/core/templates/core/partials/save_button.html:3
 msgid "Save"
 msgstr ""
 
-#: templates/core/group/child_groups.html:112
-#: templates/core/group/child_groups.html:147
+#: aleksis/core/templates/core/group/child_groups.html:112
+#: aleksis/core/templates/core/group/child_groups.html:147
 msgid "Save and next"
 msgstr ""
 
-#: templates/core/group/edit.html:11 templates/core/group/edit.html:12
+#: aleksis/core/templates/core/group/edit.html:11
+#: aleksis/core/templates/core/group/edit.html:12
 msgid "Edit group"
 msgstr ""
 
-#: templates/core/group/full.html:38 templates/core/person/full.html:37
+#: aleksis/core/templates/core/group/full.html:38
+#: aleksis/core/templates/core/person/full.html:37
 msgid "Change preferences"
 msgstr ""
 
-#: templates/core/group/full.html:64
+#: aleksis/core/templates/core/group/full.html:64
 msgid "Statistics"
 msgstr ""
 
-#: templates/core/group/full.html:67
+#: aleksis/core/templates/core/group/full.html:67
 msgid "Count of members"
 msgstr ""
 
-#: templates/core/group/full.html:71
+#: aleksis/core/templates/core/group/full.html:71
 msgid "Average age"
 msgstr ""
 
-#: templates/core/group/full.html:76
+#: aleksis/core/templates/core/group/full.html:76
 msgid "Age range"
 msgstr ""
 
-#: templates/core/group/full.html:76
+#: aleksis/core/templates/core/group/full.html:76
 msgid "years to"
 msgstr ""
 
-#: templates/core/group/full.html:76
+#: aleksis/core/templates/core/group/full.html:76
 msgid "years "
 msgstr ""
 
-#: templates/core/group/list.html:14
+#: aleksis/core/templates/core/group/list.html:14
 msgid "Create group"
 msgstr ""
 
-#: templates/core/group/list.html:17
+#: aleksis/core/templates/core/group/list.html:17
 msgid "Filter groups"
 msgstr ""
 
-#: templates/core/group/list.html:24 templates/core/person/list.html:28
+#: aleksis/core/templates/core/group/list.html:24
+#: aleksis/core/templates/core/person/list.html:28
 msgid "Clear"
 msgstr ""
 
-#: templates/core/group/list.html:28
+#: aleksis/core/templates/core/group/list.html:28
 msgid "Selected groups"
 msgstr ""
 
-#: templates/core/group_type/edit.html:6 templates/core/group_type/edit.html:7
+#: aleksis/core/templates/core/group_type/edit.html:6
+#: aleksis/core/templates/core/group_type/edit.html:7
 msgid "Edit group type"
 msgstr ""
 
-#: templates/core/group_type/list.html:14
+#: aleksis/core/templates/core/group_type/list.html:14
 msgid "Create group type"
 msgstr ""
 
-#: templates/core/index.html:4
+#: aleksis/core/templates/core/index.html:4
 msgid "Home"
 msgstr ""
 
-#: templates/core/index.html:50
+#: aleksis/core/templates/core/index.html:49
 msgid ""
 "\n"
-"          You didn't customise your dashboard so that you see the system default. Please click on \"Edit dashboard\" to\n"
-"          customise your personal dashboard.\n"
-"        "
+"        You didn't customise your dashboard so that you see the system default. Please click on \"Edit dashboard\" to\n"
+"        customise your personal dashboard.\n"
+"      "
 msgstr ""
 
-#: templates/core/index.html:59
+#: aleksis/core/templates/core/index.html:59
 msgid "Last activities"
 msgstr ""
 
-#: templates/core/index.html:77
+#: aleksis/core/templates/core/index.html:77
 msgid "No activities available yet."
 msgstr ""
 
-#: templates/core/index.html:82
+#: aleksis/core/templates/core/index.html:82
 msgid "Recent notifications"
 msgstr ""
 
-#: templates/core/index.html:98
+#: aleksis/core/templates/core/index.html:98
+#: aleksis/core/templates/core/notifications.html:23
 msgid "More information →"
 msgstr ""
 
-#: templates/core/index.html:105
+#: aleksis/core/templates/core/index.html:105
+#: aleksis/core/templates/core/notifications.html:30
 msgid "No notifications available yet."
 msgstr ""
 
-#: templates/core/pages/about.html:6 templates/core/pages/about.html:15
+#: aleksis/core/templates/core/pages/about.html:6
+#: aleksis/core/templates/core/pages/about.html:15
 msgid "About AlekSIS"
 msgstr ""
 
-#: templates/core/pages/about.html:7
+#: aleksis/core/templates/core/pages/about.html:7
 msgid "AlekSIS – The Free School Information System"
 msgstr ""
 
-#: templates/core/pages/about.html:17
+#: aleksis/core/templates/core/pages/about.html:17
 msgid ""
 "\n"
 "              This platform is powered by AlekSIS, a web-based school information system (SIS) which can be used\n"
@@ -1235,19 +1761,19 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/pages/about.html:25
+#: aleksis/core/templates/core/pages/about.html:25
 msgid "Website of AlekSIS"
 msgstr ""
 
-#: templates/core/pages/about.html:26
+#: aleksis/core/templates/core/pages/about.html:26
 msgid "Source code"
 msgstr ""
 
-#: templates/core/pages/about.html:35
+#: aleksis/core/templates/core/pages/about.html:35
 msgid "Licence information"
 msgstr ""
 
-#: templates/core/pages/about.html:37
+#: aleksis/core/templates/core/pages/about.html:37
 msgid ""
 "\n"
 "              The core and the official apps of AlekSIS are licenced under the EUPL, version 1.2 or later. For licence\n"
@@ -1256,23 +1782,23 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/pages/about.html:45
+#: aleksis/core/templates/core/pages/about.html:45
 msgid "Free/Open Source Licence"
 msgstr ""
 
-#: templates/core/pages/about.html:46
+#: aleksis/core/templates/core/pages/about.html:46
 msgid "Other Licence"
 msgstr ""
 
-#: templates/core/pages/about.html:50
+#: aleksis/core/templates/core/pages/about.html:50
 msgid "Full licence text"
 msgstr ""
 
-#: templates/core/pages/about.html:51
+#: aleksis/core/templates/core/pages/about.html:51
 msgid "More information about the EUPL"
 msgstr ""
 
-#: templates/core/pages/about.html:90
+#: aleksis/core/templates/core/pages/about.html:90
 #, python-format
 msgid ""
 "\n"
@@ -1280,12 +1806,12 @@ msgid ""
 "                  "
 msgstr ""
 
-#: templates/core/pages/delete.html:6
+#: aleksis/core/templates/core/pages/delete.html:6
 #, python-format
 msgid "Delete %(object_name)s"
 msgstr ""
 
-#: templates/core/pages/delete.html:13
+#: aleksis/core/templates/core/pages/delete.html:13
 #, python-format
 msgid ""
 "\n"
@@ -1293,102 +1819,114 @@ msgid ""
 "    "
 msgstr ""
 
-#: templates/core/pages/progress.html:27
+#: aleksis/core/templates/core/pages/progress.html:27
 msgid ""
 "\n"
 "              Without activated JavaScript the progress status can't be updated.\n"
 "            "
 msgstr ""
 
-#: templates/core/pages/progress.html:47
-#: templates/two_factor/core/otp_required.html:19
+#: aleksis/core/templates/core/pages/progress.html:47
+#: aleksis/core/templates/two_factor/core/otp_required.html:19
 msgid "Go back"
 msgstr ""
 
-#: templates/core/pages/system_status.html:12
+#: aleksis/core/templates/core/pages/system_status.html:12
 msgid "System checks"
 msgstr ""
 
-#: templates/core/pages/system_status.html:21
+#: aleksis/core/templates/core/pages/system_status.html:21
 msgid "Maintenance mode enabled"
 msgstr ""
 
-#: templates/core/pages/system_status.html:23
+#: aleksis/core/templates/core/pages/system_status.html:23
 msgid ""
 "\n"
 "                Only admin and visitors from internal IPs can access thesite.\n"
 "              "
 msgstr ""
 
-#: templates/core/pages/system_status.html:34
+#: aleksis/core/templates/core/pages/system_status.html:34
 msgid "Maintenance mode disabled"
 msgstr ""
 
-#: templates/core/pages/system_status.html:35
+#: aleksis/core/templates/core/pages/system_status.html:35
 msgid "Everyone can access the site."
 msgstr ""
 
-#: templates/core/pages/system_status.html:45
+#: aleksis/core/templates/core/pages/system_status.html:45
 msgid "Debug mode enabled"
 msgstr ""
 
-#: templates/core/pages/system_status.html:47
+#: aleksis/core/templates/core/pages/system_status.html:47
 msgid ""
 "\n"
 "                The web server throws back debug information on errors. Do not use in production!\n"
 "              "
 msgstr ""
 
-#: templates/core/pages/system_status.html:54
+#: aleksis/core/templates/core/pages/system_status.html:54
 msgid "Debug mode disabled"
 msgstr ""
 
-#: templates/core/pages/system_status.html:56
+#: aleksis/core/templates/core/pages/system_status.html:56
 msgid ""
 "\n"
 "                Debug mode is disabled. Default error pages are displayed on errors.\n"
 "              "
 msgstr ""
 
-#: templates/core/pages/system_status.html:69
+#: aleksis/core/templates/core/pages/system_status.html:69
 msgid "System health checks"
 msgstr ""
 
-#: templates/core/pages/system_status.html:75
+#: aleksis/core/templates/core/pages/system_status.html:75
 msgid "Service"
 msgstr ""
 
-#: templates/core/pages/system_status.html:76
-#: templates/core/pages/system_status.html:115
+#: aleksis/core/templates/core/pages/system_status.html:76
+#: aleksis/core/templates/core/pages/system_status.html:115
 msgid "Status"
 msgstr ""
 
-#: templates/core/pages/system_status.html:77
+#: aleksis/core/templates/core/pages/system_status.html:77
 msgid "Time taken"
 msgstr ""
 
-#: templates/core/pages/system_status.html:96
+#: aleksis/core/templates/core/pages/system_status.html:96
 msgid "seconds"
 msgstr ""
 
-#: templates/core/pages/system_status.html:107
+#: aleksis/core/templates/core/pages/system_status.html:107
 msgid "Celery task results"
 msgstr ""
 
-#: templates/core/pages/system_status.html:112
+#: aleksis/core/templates/core/pages/system_status.html:112
 msgid "Task"
 msgstr ""
 
-#: templates/core/pages/system_status.html:113
+#: aleksis/core/templates/core/pages/system_status.html:113
 msgid "ID"
 msgstr ""
 
-#: templates/core/pages/system_status.html:114
+#: aleksis/core/templates/core/pages/system_status.html:114
 msgid "Date done"
 msgstr ""
 
-#: templates/core/partials/announcements.html:9
-#: templates/core/partials/announcements.html:36
+#: aleksis/core/templates/core/pages/test_pdf.html:7
+#: aleksis/core/templates/core/pages/test_pdf.html:8
+msgid "Test PDF generation"
+msgstr ""
+
+#: aleksis/core/templates/core/pages/test_pdf.html:14
+msgid ""
+"\n"
+"        This simple view can be used to ensure the correct function of the built-in PDF generation system.\n"
+"      "
+msgstr ""
+
+#: aleksis/core/templates/core/partials/announcements.html:8
+#: aleksis/core/templates/core/partials/announcements.html:35
 #, python-format
 msgid ""
 "\n"
@@ -1396,7 +1934,7 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/partials/announcements.html:13
+#: aleksis/core/templates/core/partials/announcements.html:12
 #, python-format
 msgid ""
 "\n"
@@ -1404,7 +1942,7 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/partials/announcements.html:40
+#: aleksis/core/templates/core/partials/announcements.html:39
 #, python-format
 msgid ""
 "\n"
@@ -1412,23 +1950,23 @@ msgid ""
 "            "
 msgstr ""
 
-#: templates/core/partials/crud_events.html:15
+#: aleksis/core/templates/core/partials/crud_events.html:15
 msgid "Changed by"
 msgstr ""
 
-#: templates/core/partials/crud_events.html:15
+#: aleksis/core/templates/core/partials/crud_events.html:15
 msgid "Unknown"
 msgstr ""
 
-#: templates/core/partials/language_form.html:15
+#: aleksis/core/templates/core/partials/language_form.html:15
 msgid "Language"
 msgstr ""
 
-#: templates/core/partials/language_form.html:27
+#: aleksis/core/templates/core/partials/language_form.html:27
 msgid "Select language"
 msgstr ""
 
-#: templates/core/partials/no_person.html:12
+#: aleksis/core/templates/core/partials/no_person.html:12
 msgid ""
 "\n"
 "            Your administrator account is not linked to any person. Therefore,\n"
@@ -1436,7 +1974,7 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/core/partials/no_person.html:19
+#: aleksis/core/templates/core/partials/no_person.html:19
 msgid ""
 "\n"
 "            Your user account is not linked to a person. This means you\n"
@@ -1445,105 +1983,178 @@ msgid ""
 "          "
 msgstr ""
 
-#: templates/core/person/accounts.html:12
-#: templates/core/person/accounts.html:14
-msgid "Link persons to accounts"
-msgstr ""
-
-#: templates/core/person/accounts.html:21
-msgid ""
-"\n"
-"        You can use this form to assign user accounts to persons. Use the\n"
-"        dropdowns to select existing accounts; use the text fields to create new\n"
-"        accounts on-the-fly. The latter will create a new account with the\n"
-"        entered username and copy all other details from the person.\n"
-"      "
-msgstr ""
-
-#: templates/core/person/accounts.html:36
-#: templates/core/person/accounts.html:60
-msgid "Update"
-msgstr ""
-
-#: templates/core/person/accounts.html:42
-msgid "Existing account"
-msgstr ""
-
-#: templates/core/person/accounts.html:43
-msgid "New account"
+#: aleksis/core/templates/core/person/create.html:12
+#: aleksis/core/templates/core/person/create.html:13
+#: aleksis/core/templates/core/person/list.html:17
+msgid "Create person"
 msgstr ""
 
-#: templates/core/person/edit.html:12 templates/core/person/edit.html:13
+#: aleksis/core/templates/core/person/edit.html:12
+#: aleksis/core/templates/core/person/edit.html:13
 msgid "Edit person"
 msgstr ""
 
-#: templates/core/person/full.html:44
+#: aleksis/core/templates/core/person/full.html:44
+#: aleksis/core/templates/impersonate/list_users.html:7
+#: aleksis/core/templates/impersonate/list_users.html:8
 msgid "Impersonate"
 msgstr ""
 
-#: templates/core/person/full.html:50
+#: aleksis/core/templates/core/person/full.html:50
 msgid "Contact details"
 msgstr ""
 
-#: templates/core/person/full.html:130
+#: aleksis/core/templates/core/person/full.html:130
 msgid "Children"
 msgstr ""
 
-#: templates/core/person/list.html:17
-msgid "Create person"
-msgstr ""
-
-#: templates/core/person/list.html:21
+#: aleksis/core/templates/core/person/list.html:21
 msgid "Filter persons"
 msgstr ""
 
-#: templates/core/person/list.html:32
+#: aleksis/core/templates/core/person/list.html:32
 msgid "Selected persons"
 msgstr ""
 
-#: templates/core/school_term/create.html:6
-#: templates/core/school_term/create.html:7
-#: templates/core/school_term/list.html:14
+#: aleksis/core/templates/core/school_term/create.html:6
+#: aleksis/core/templates/core/school_term/create.html:7
+#: aleksis/core/templates/core/school_term/list.html:14
 msgid "Create school term"
 msgstr ""
 
-#: templates/core/school_term/edit.html:6
-#: templates/core/school_term/edit.html:7
+#: aleksis/core/templates/core/school_term/edit.html:6
+#: aleksis/core/templates/core/school_term/edit.html:7
 msgid "Edit school term"
 msgstr ""
 
-#: templates/dynamic_preferences/form.html:9
+#: aleksis/core/templates/dynamic_preferences/form.html:9
 msgid "Site preferences"
 msgstr ""
 
-#: templates/dynamic_preferences/form.html:11
+#: aleksis/core/templates/dynamic_preferences/form.html:11
 msgid "My preferences"
 msgstr ""
 
-#: templates/dynamic_preferences/form.html:13
+#: aleksis/core/templates/dynamic_preferences/form.html:13
 #, python-format
 msgid "Preferences for %(instance)s"
 msgstr ""
 
-#: templates/dynamic_preferences/form.html:25
+#: aleksis/core/templates/dynamic_preferences/form.html:25
 msgid "Save preferences"
 msgstr ""
 
-#: templates/impersonate/list_users.html:8
-msgid "Impersonate user"
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:5
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:6
+msgid "Delete application"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:12
+#, python-format
+msgid "Are you sure to delete the application %(application_name)s?"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_confirm_delete.html:24
+#: aleksis/core/templates/oauth2_provider/application_form.html:18
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:24
+#: aleksis/core/templates/two_factor/_wizard_actions.html:6
+msgid "Cancel"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:27
+msgid "Client id"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:35
+msgid "Client secret"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:43
+msgid "Client type"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:51
+msgid "Authorization Grant Type"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_detail.html:59
+msgid "Redirect URIs"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_form.html:5
+msgid "Create OAuth2 Application"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_form.html:10
+msgid "Edit application"
 msgstr ""
 
-#: templates/offline.html:5
+#: aleksis/core/templates/oauth2_provider/application_list.html:8
+msgid "OAuth2 applications"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_list.html:12
+msgid "Register new applications"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/application_list.html:23
+msgid "No applications defined."
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorize.html:5
+#: aleksis/core/templates/oauth2_provider/authorize.html:16
+msgid "Authorize"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorize.html:18
+msgid "The application requests access to the following scopes:"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorize.html:33
+msgid "Allow"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorize.html:36
+msgid "Disallow"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-oob.html:12
+msgid "Success!"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-oob.html:14
+msgid "Please return to your application and enter this code:"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:5
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:6
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:23
+msgid "Revoke access"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:12
+msgid "Are you sure to revoke the access for this application?"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-token-delete.html:20
+msgid "Revoke"
+msgstr ""
+
+#: aleksis/core/templates/oauth2_provider/authorized-tokens.html:33
+msgid "No authorized applications."
+msgstr ""
+
+#: aleksis/core/templates/offline.html:5
 msgid "Network error"
 msgstr ""
 
-#: templates/offline.html:8
+#: aleksis/core/templates/offline.html:8
 msgid ""
 "No internet\n"
 "    connection."
 msgstr ""
 
-#: templates/offline.html:12
+#: aleksis/core/templates/offline.html:12
 msgid ""
 "\n"
 "      There was an error accessing this page. You probably don't have an internet connection. Check to see if your WiFi\n"
@@ -1552,36 +2163,116 @@ msgid ""
 "    "
 msgstr ""
 
-#: templates/search/search.html:8
+#: aleksis/core/templates/search/search.html:8
 msgid "Global Search"
 msgstr ""
 
-#: templates/search/search.html:15
+#: aleksis/core/templates/search/search.html:15
 msgid "Search Term"
 msgstr ""
 
-#: templates/search/search.html:26
+#: aleksis/core/templates/search/search.html:26
 msgid "Results"
 msgstr ""
 
-#: templates/search/search.html:38
+#: aleksis/core/templates/search/search.html:38
 msgid "No search results could be found to your search."
 msgstr ""
 
-#: templates/search/search.html:87
+#: aleksis/core/templates/search/search.html:87
 msgid "Please enter a search term above."
 msgstr ""
 
-#: templates/templated_email/data_checks.email:4
+#: aleksis/core/templates/socialaccount/authentication_error.html:5
+#: aleksis/core/templates/socialaccount/authentication_error.html:6
+msgid "Third-party Account Login Failure"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/authentication_error.html:13
+msgid "Third-party Account Login Failure."
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/authentication_error.html:15
+msgid ""
+"\n"
+"            An error occurred while attempting to login via your third-party account.\n"
+"            Please contact one of your site administrators.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/connections.html:5
+#: aleksis/core/templates/socialaccount/connections.html:6
+msgid "Connections"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/connections.html:24
+msgid "Remove"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/connections.html:34
+msgid "You currently have no third-party accounts connected to this account."
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/connections.html:37
+msgid "Add a Third-party Account"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/login_cancelled.html:5
+#: aleksis/core/templates/socialaccount/login_cancelled.html:6
+#: aleksis/core/templates/socialaccount/login_cancelled.html:13
+msgid "Login cancelled"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/login_cancelled.html:15
+#, python-format
+msgid ""
+"\n"
+"            You decided to cancel logging in to our site using one of your existing accounts. If this was a mistake, please proceed to <a href=\"%(login_url)s\">sign in</a>.\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/signup.html:12
+#, python-format
+msgid ""
+"You are about to use your %(provider_name)s account to login to\n"
+"        %(site_name)s. As a final step, please complete the following form:"
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/snippets/provider_list.html:12
+#, python-format
+msgid ""
+"\n"
+"                Login with %(name)s\n"
+"              "
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/snippets/provider_list.html:21
+#, python-format
+msgid ""
+"\n"
+"            Login with %(name)s\n"
+"          "
+msgstr ""
+
+#: aleksis/core/templates/socialaccount/snippets/provider_list.html:30
+msgid ""
+"\n"
+"          No third-party account providers available.\n"
+"        "
+msgstr ""
+
+#: aleksis/core/templates/templated_email/data_checks.email:4
 msgid "The system detected some new problems with your data."
 msgstr ""
 
-#: templates/templated_email/data_checks.email:8
-#: templates/templated_email/data_checks.email:24
+#: aleksis/core/templates/templated_email/data_checks.email:8
+#: aleksis/core/templates/templated_email/data_checks.email:24
+#: aleksis/core/templates/templated_email/person_changed.email:8
+#: aleksis/core/templates/templated_email/person_changed.email:20
 msgid "Hello,"
 msgstr ""
 
-#: templates/templated_email/data_checks.email:10
+#: aleksis/core/templates/templated_email/data_checks.email:10
 msgid ""
 "\n"
 "  the system detected some new problems with your data.\n"
@@ -1589,7 +2280,7 @@ msgid ""
 " "
 msgstr ""
 
-#: templates/templated_email/data_checks.email:26
+#: aleksis/core/templates/templated_email/data_checks.email:26
 msgid ""
 "\n"
 "   the system detected some new problems with your data.\n"
@@ -1597,35 +2288,35 @@ msgid ""
 "  "
 msgstr ""
 
-#: templates/templated_email/data_checks.email:34
+#: aleksis/core/templates/templated_email/data_checks.email:34
 msgid "Problem description"
 msgstr ""
 
-#: templates/templated_email/data_checks.email:35
+#: aleksis/core/templates/templated_email/data_checks.email:35
 msgid "Count of objects with new problems"
 msgstr ""
 
-#: templates/templated_email/notification.email:3
+#: aleksis/core/templates/templated_email/notification.email:3
 msgid "New notification for"
 msgstr ""
 
-#: templates/templated_email/notification.email:6
-#: templates/templated_email/notification.email:27
+#: aleksis/core/templates/templated_email/notification.email:6
+#: aleksis/core/templates/templated_email/notification.email:27
 #, python-format
 msgid "Dear %(notification_user)s,"
 msgstr ""
 
-#: templates/templated_email/notification.email:8
-#: templates/templated_email/notification.email:29
+#: aleksis/core/templates/templated_email/notification.email:8
+#: aleksis/core/templates/templated_email/notification.email:29
 msgid "we got a new notification for you:"
 msgstr ""
 
-#: templates/templated_email/notification.email:15
-#: templates/templated_email/notification.email:35
+#: aleksis/core/templates/templated_email/notification.email:15
+#: aleksis/core/templates/templated_email/notification.email:35
 msgid "More information"
 msgstr ""
 
-#: templates/templated_email/notification.email:18
+#: aleksis/core/templates/templated_email/notification.email:18
 #, python-format
 msgid ""
 "\n"
@@ -1633,12 +2324,7 @@ msgid ""
 "    "
 msgstr ""
 
-#: templates/templated_email/notification.email:22
-#: templates/templated_email/notification.email:46
-msgid "Your AlekSIS team"
-msgstr ""
-
-#: templates/templated_email/notification.email:40
+#: aleksis/core/templates/templated_email/notification.email:40
 #, python-format
 msgid ""
 "\n"
@@ -1646,24 +2332,41 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/two_factor/_base_focus.html:6
-#: templates/two_factor/core/otp_required.html:22
-#: templates/two_factor/core/setup.html:5
-#: templates/two_factor/profile/profile.html:87
-msgid "Enable Two-Factor Authentication"
+#: aleksis/core/templates/templated_email/person_changed.email:4
+#, python-format
+msgid "%(person)s changed their data!"
 msgstr ""
 
-#: templates/two_factor/_wizard_actions.html:6
-msgid "Cancel"
+#: aleksis/core/templates/templated_email/person_changed.email:10
+#, python-format
+msgid ""
+"\n"
+"   the person %(person)s recently changed the following fields:\n"
+" "
+msgstr ""
+
+#: aleksis/core/templates/templated_email/person_changed.email:22
+#, python-format
+msgid ""
+"\n"
+"    the person %(person)s recently changed the following fields:\n"
+"  "
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:5
-#: templates/two_factor/core/backup_tokens.html:9
-#: templates/two_factor/profile/profile.html:46
+#: aleksis/core/templates/two_factor/_base_focus.html:6
+#: aleksis/core/templates/two_factor/core/otp_required.html:22
+#: aleksis/core/templates/two_factor/core/setup.html:5
+#: aleksis/core/templates/two_factor/profile/profile.html:87
+msgid "Enable Two-Factor Authentication"
+msgstr ""
+
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:5
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:9
+#: aleksis/core/templates/two_factor/profile/profile.html:46
 msgid "Backup Tokens"
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:14
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:14
 msgid ""
 "\n"
 "        Backup tokens can be used when your primary and backup\n"
@@ -1674,115 +2377,129 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:33
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:33
 msgid ""
 "\n"
 "          Print these tokens and keep them somewhere safe.\n"
 "        "
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:39
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:39
 msgid "You don't have any backup codes yet."
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:45
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:45
 msgid "Back to Account Security"
 msgstr ""
 
-#: templates/two_factor/core/backup_tokens.html:49
+#: aleksis/core/templates/two_factor/core/backup_tokens.html:49
 msgid "Generate Tokens"
 msgstr ""
 
-#: templates/two_factor/core/login.html:16
-msgid "You have no permission to view this page. Please login with an other account."
+#: aleksis/core/templates/two_factor/core/login.html:20
+msgid "Login with username and password"
 msgstr ""
 
-#: templates/two_factor/core/login.html:24
-msgid "Please login to see this page."
+#: aleksis/core/templates/two_factor/core/login.html:28
+msgid ""
+"You have no permission to view this page. Please login with an other\n"
+"                    account."
 msgstr ""
 
-#: templates/two_factor/core/login.html:30
-msgid "Login with username and password"
+#: aleksis/core/templates/two_factor/core/login.html:36
+msgid "Please login to see this page."
 msgstr ""
 
-#: templates/two_factor/core/login.html:40
+#: aleksis/core/templates/two_factor/core/login.html:46
 msgid ""
-"We are calling your phone right now, please enter the\n"
-"              digits you hear."
+"\n"
+"                        We are calling your phone right now, please enter the\n"
+"                        digits you hear.\n"
+"                      "
 msgstr ""
 
-#: templates/two_factor/core/login.html:43
+#: aleksis/core/templates/two_factor/core/login.html:51
 msgid ""
-"We sent you a text message, please enter the tokens we\n"
-"              sent."
+"\n"
+"                        We sent you a text message, please enter the tokens we\n"
+"                        sent.\n"
+"                      "
 msgstr ""
 
-#: templates/two_factor/core/login.html:46
+#: aleksis/core/templates/two_factor/core/login.html:56
 msgid ""
-"Please enter the tokens generated by your token\n"
-"              generator."
+"\n"
+"                        Please enter the tokens generated by your token\n"
+"                        generator.\n"
+"                      "
 msgstr ""
 
-#: templates/two_factor/core/login.html:50
+#: aleksis/core/templates/two_factor/core/login.html:62
 msgid ""
-"Use this form for entering backup tokens for logging in.\n"
-"            These tokens have been generated for you to print and keep safe. Please\n"
-"            enter one of these backup tokens to login to your account."
+"\n"
+"                      Use this form for entering backup tokens for logging in.\n"
+"                      These tokens have been generated for you to print and keep safe. Please\n"
+"                      enter one of these backup tokens to login to your account.\n"
+"                    "
 msgstr ""
 
-#: templates/two_factor/core/login.html:68
+#: aleksis/core/templates/two_factor/core/login.html:90
+msgid "Device currently not available?"
+msgstr ""
+
+#: aleksis/core/templates/two_factor/core/login.html:92
 msgid "Or, alternatively, use one of your backup phones:"
 msgstr ""
 
-#: templates/two_factor/core/login.html:78
+#: aleksis/core/templates/two_factor/core/login.html:102
 msgid "As a last resort, you can use a backup token:"
 msgstr ""
 
-#: templates/two_factor/core/login.html:81
+#: aleksis/core/templates/two_factor/core/login.html:105
 msgid "Use Backup Token"
 msgstr ""
 
-#: templates/two_factor/core/login.html:93
+#: aleksis/core/templates/two_factor/core/login.html:116
 msgid "Use alternative login options"
 msgstr ""
 
-#: templates/two_factor/core/otp_required.html:9
+#: aleksis/core/templates/two_factor/core/otp_required.html:9
 msgid "Permission Denied"
 msgstr ""
 
-#: templates/two_factor/core/otp_required.html:10
+#: aleksis/core/templates/two_factor/core/otp_required.html:10
 msgid ""
 "The page you requested, enforces users to verify using\n"
 "          two-factor authentication for security reasons. You need to enable these\n"
 "          security features in order to access this page."
 msgstr ""
 
-#: templates/two_factor/core/otp_required.html:14
+#: aleksis/core/templates/two_factor/core/otp_required.html:14
 msgid ""
 "Two-factor authentication is not enabled for your\n"
 "          account. Enable two-factor authentication for enhanced account\n"
 "          security."
 msgstr ""
 
-#: templates/two_factor/core/phone_register.html:5
-#: templates/two_factor/core/phone_register.html:9
+#: aleksis/core/templates/two_factor/core/phone_register.html:5
+#: aleksis/core/templates/two_factor/core/phone_register.html:9
 msgid "Add Backup Phone"
 msgstr ""
 
-#: templates/two_factor/core/phone_register.html:12
+#: aleksis/core/templates/two_factor/core/phone_register.html:12
 msgid ""
 "You'll be adding a backup phone number to your\n"
 "      account. This number will be used if your primary method of\n"
 "      registration is not available."
 msgstr ""
 
-#: templates/two_factor/core/phone_register.html:16
+#: aleksis/core/templates/two_factor/core/phone_register.html:16
 msgid ""
 "We've sent a token to your phone number. Please\n"
 "      enter the token you've received."
 msgstr ""
 
-#: templates/two_factor/core/setup.html:9
+#: aleksis/core/templates/two_factor/core/setup.html:9
 msgid ""
 "\n"
 "        You are about to take your account security to the\n"
@@ -1791,14 +2508,14 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:17
+#: aleksis/core/templates/two_factor/core/setup.html:17
 msgid ""
 "\n"
 "        Please select which authentication method you would like to use:\n"
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:23
+#: aleksis/core/templates/two_factor/core/setup.html:23
 msgid ""
 "\n"
 "        To start using a token generator, please use your\n"
@@ -1807,7 +2524,7 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:34
+#: aleksis/core/templates/two_factor/core/setup.html:34
 msgid ""
 "\n"
 "        Please enter the phone number you wish to receive the\n"
@@ -1815,7 +2532,7 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:41
+#: aleksis/core/templates/two_factor/core/setup.html:41
 msgid ""
 "\n"
 "        Please enter the phone number you wish to be called on.\n"
@@ -1823,21 +2540,21 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:50
+#: aleksis/core/templates/two_factor/core/setup.html:50
 msgid ""
 "\n"
 "            We are calling your phone right now, please enter the digits you hear.\n"
 "          "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:56
+#: aleksis/core/templates/two_factor/core/setup.html:56
 msgid ""
 "\n"
 "            We sent you a text message, please enter the tokens we sent.\n"
 "          "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:63
+#: aleksis/core/templates/two_factor/core/setup.html:63
 msgid ""
 "\n"
 "          We've encountered an issue with the selected authentication method. Please\n"
@@ -1847,7 +2564,7 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/two_factor/core/setup.html:73
+#: aleksis/core/templates/two_factor/core/setup.html:73
 msgid ""
 "\n"
 "        To identify and verify your YubiKey, please insert a\n"
@@ -1856,29 +2573,29 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:5
-#: templates/two_factor/core/setup_complete.html:9
+#: aleksis/core/templates/two_factor/core/setup_complete.html:5
+#: aleksis/core/templates/two_factor/core/setup_complete.html:9
 msgid "Two-Factor Authentication successfully enabled"
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:14
+#: aleksis/core/templates/two_factor/core/setup_complete.html:14
 msgid ""
 "\n"
 "        Congratulations, you've successfully enabled two-factor authentication.\n"
 "      "
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:24
-#: templates/two_factor/core/setup_complete.html:44
+#: aleksis/core/templates/two_factor/core/setup_complete.html:24
+#: aleksis/core/templates/two_factor/core/setup_complete.html:44
 msgid "Back to Profile"
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:28
-#: templates/two_factor/core/setup_complete.html:48
+#: aleksis/core/templates/two_factor/core/setup_complete.html:28
+#: aleksis/core/templates/two_factor/core/setup_complete.html:48
 msgid "Generate backup codes"
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:34
+#: aleksis/core/templates/two_factor/core/setup_complete.html:34
 msgid ""
 "\n"
 "          However, it might happen that you don't have access to\n"
@@ -1887,65 +2604,65 @@ msgid ""
 "        "
 msgstr ""
 
-#: templates/two_factor/core/setup_complete.html:52
-#: templates/two_factor/profile/profile.html:41
+#: aleksis/core/templates/two_factor/core/setup_complete.html:52
+#: aleksis/core/templates/two_factor/profile/profile.html:41
 msgid "Add Phone Number"
 msgstr ""
 
-#: templates/two_factor/profile/disable.html:5
-#: templates/two_factor/profile/disable.html:9
-#: templates/two_factor/profile/profile.html:63
-#: templates/two_factor/profile/profile.html:73
+#: aleksis/core/templates/two_factor/profile/disable.html:5
+#: aleksis/core/templates/two_factor/profile/disable.html:9
+#: aleksis/core/templates/two_factor/profile/profile.html:63
+#: aleksis/core/templates/two_factor/profile/profile.html:73
 msgid "Disable Two-Factor Authentication"
 msgstr ""
 
-#: templates/two_factor/profile/disable.html:12
+#: aleksis/core/templates/two_factor/profile/disable.html:12
 msgid "You are about to disable two-factor authentication. This weakens your account security, are you sure?"
 msgstr ""
 
-#: templates/two_factor/profile/disable.html:26
+#: aleksis/core/templates/two_factor/profile/disable.html:26
 msgid "Disable"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:5
-#: templates/two_factor/profile/profile.html:10
+#: aleksis/core/templates/two_factor/profile/profile.html:5
+#: aleksis/core/templates/two_factor/profile/profile.html:10
 msgid "Account Security"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:15
+#: aleksis/core/templates/two_factor/profile/profile.html:15
 msgid "Tokens will be generated by your token generator."
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:17
+#: aleksis/core/templates/two_factor/profile/profile.html:17
 #, python-format
 msgid "Primary method: %(primary)s"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:19
+#: aleksis/core/templates/two_factor/profile/profile.html:19
 msgid "Tokens will be generated by your YubiKey."
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:23
+#: aleksis/core/templates/two_factor/profile/profile.html:23
 msgid "Backup Phone Numbers"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:24
+#: aleksis/core/templates/two_factor/profile/profile.html:24
 msgid ""
 "If your primary method is not available, we are able to\n"
 "        send backup tokens to the phone numbers listed below."
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:33
+#: aleksis/core/templates/two_factor/profile/profile.html:33
 msgid "Unregister"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:48
+#: aleksis/core/templates/two_factor/profile/profile.html:48
 msgid ""
 "If you don't have any device with you, you can access\n"
 "        your account using backup tokens."
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:50
+#: aleksis/core/templates/two_factor/profile/profile.html:50
 #, python-format
 msgid ""
 "\n"
@@ -1958,11 +2675,11 @@ msgid_plural ""
 msgstr[0] ""
 msgstr[1] ""
 
-#: templates/two_factor/profile/profile.html:59
+#: aleksis/core/templates/two_factor/profile/profile.html:59
 msgid "Show Codes"
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:65
+#: aleksis/core/templates/two_factor/profile/profile.html:65
 msgid ""
 "\n"
 "        However we strongly discourage you to do so, you can\n"
@@ -1970,7 +2687,7 @@ msgid ""
 "      "
 msgstr ""
 
-#: templates/two_factor/profile/profile.html:78
+#: aleksis/core/templates/two_factor/profile/profile.html:78
 msgid ""
 "\n"
 "        Two-factor authentication is not enabled for your\n"
@@ -1979,99 +2696,135 @@ msgid ""
 "      "
 msgstr ""
 
-#: util/notifications.py:65
+#: aleksis/core/util/notifications.py:65
 msgid "E-Mail"
 msgstr ""
 
-#: util/notifications.py:66
+#: aleksis/core/util/notifications.py:66
 msgid "SMS"
 msgstr ""
 
-#: views.py:141
+#: aleksis/core/util/pdf.py:105
+msgid "Progress: Generate PDF file"
+msgstr ""
+
+#: aleksis/core/util/pdf.py:106
+msgid "Generating PDF file …"
+msgstr ""
+
+#: aleksis/core/util/pdf.py:107
+msgid "The PDF file has been generated successfully."
+msgstr ""
+
+#: aleksis/core/util/pdf.py:108
+msgid "There was a problem while generating the PDF file."
+msgstr ""
+
+#: aleksis/core/util/pdf.py:111
+msgid "Download PDF"
+msgstr ""
+
+#: aleksis/core/views.py:251
 msgid "The school term has been created."
 msgstr ""
 
-#: views.py:153
+#: aleksis/core/views.py:263
 msgid "The school term has been saved."
 msgstr ""
 
-#: views.py:298
+#: aleksis/core/views.py:387
 msgid "The child groups were successfully saved."
 msgstr ""
 
-#: views.py:336
+#: aleksis/core/views.py:406 aleksis/core/views.py:416
 msgid "The person has been saved."
 msgstr ""
 
-#: views.py:375
+#: aleksis/core/views.py:466
 msgid "The group has been saved."
 msgstr ""
 
-#: views.py:467
+#: aleksis/core/views.py:563
 msgid "The announcement has been saved."
 msgstr ""
 
-#: views.py:483
+#: aleksis/core/views.py:579
 msgid "The announcement has been deleted."
 msgstr ""
 
-#: views.py:562
+#: aleksis/core/views.py:663
 msgid "The preferences have been saved successfully."
 msgstr ""
 
-#: views.py:586
+#: aleksis/core/views.py:687
 msgid "The person has been deleted."
 msgstr ""
 
-#: views.py:600
+#: aleksis/core/views.py:701
 msgid "The group has been deleted."
 msgstr ""
 
-#: views.py:632
+#: aleksis/core/views.py:733
 msgid "The additional_field has been saved."
 msgstr ""
 
-#: views.py:666
+#: aleksis/core/views.py:767
 msgid "The additional field has been deleted."
 msgstr ""
 
-#: views.py:691
+#: aleksis/core/views.py:792
 msgid "The group type has been saved."
 msgstr ""
 
-#: views.py:721
+#: aleksis/core/views.py:822
 msgid "The group type has been deleted."
 msgstr ""
 
-#: views.py:749
-msgid "The data check has been started. Please note that it may take a while before you are able to fetch the data on this page."
+#: aleksis/core/views.py:855
+msgid "Progress: Run data checks"
+msgstr ""
+
+#: aleksis/core/views.py:856
+msgid "Run data checks …"
 msgstr ""
 
-#: views.py:754
-msgid "The data check has finished."
+#: aleksis/core/views.py:857
+msgid "The data checks were run successfully."
 msgstr ""
 
-#: views.py:769
+#: aleksis/core/views.py:858
+msgid "There was a problem while running data checks."
+msgstr ""
+
+#: aleksis/core/views.py:874
 #, python-brace-format
 msgid "The solve option '{solve_option_obj.verbose_name}' "
 msgstr ""
 
-#: views.py:811
+#: aleksis/core/views.py:916
 msgid "The dashboard widget has been saved."
 msgstr ""
 
-#: views.py:841
+#: aleksis/core/views.py:946
 msgid "The dashboard widget has been created."
 msgstr ""
 
-#: views.py:851
+#: aleksis/core/views.py:956
 msgid "The dashboard widget has been deleted."
 msgstr ""
 
-#: views.py:914
+#: aleksis/core/views.py:1023
 msgid "Your dashboard configuration has been saved successfully."
 msgstr ""
 
-#: views.py:916
+#: aleksis/core/views.py:1025
 msgid "The configuration of the default dashboard has been saved successfully."
 msgstr ""
+
+#: aleksis/core/views.py:1153
+msgid "The third-party account could not be disconnected because it is the only login method available."
+msgstr ""
+
+#: aleksis/core/views.py:1160
+msgid "The third-party account has been successfully disconnected."
+msgstr ""
diff --git a/aleksis/core/locale/tr_TR/LC_MESSAGES/djangojs.po b/aleksis/core/locale/tr_TR/LC_MESSAGES/djangojs.po
index ea24fc3ff92afc211898fb8b426dccb97fce56ed..dc6d028e5442a3666f35a7cb46e39e5c01cdd3f9 100644
--- a/aleksis/core/locale/tr_TR/LC_MESSAGES/djangojs.po
+++ b/aleksis/core/locale/tr_TR/LC_MESSAGES/djangojs.po
@@ -8,7 +8,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2021-01-11 21:30+0100\n"
+"POT-Creation-Date: 2021-10-28 16:18+0200\n"
 "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
@@ -17,18 +17,18 @@ msgstr ""
 "Content-Type: text/plain; charset=UTF-8\n"
 "Content-Transfer-Encoding: 8bit\n"
 
-#: static/js/main.js:15
+#: aleksis/core/static/js/main.js:15
 msgid "Today"
 msgstr ""
 
-#: static/js/main.js:16
+#: aleksis/core/static/js/main.js:16
 msgid "Cancel"
 msgstr ""
 
-#: static/js/main.js:17
+#: aleksis/core/static/js/main.js:17
 msgid "OK"
 msgstr ""
 
-#: static/js/main.js:118
+#: aleksis/core/static/js/main.js:127
 msgid "This page may contain outdated information since there is no internet connection."
 msgstr ""
diff --git a/aleksis/core/managers.py b/aleksis/core/managers.py
index 02a0cb6fb6504d931b3d848217b154951d8a0932..aed4cab72d4b0cd53d3544b20f0180a7087dfd2c 100644
--- a/aleksis/core/managers.py
+++ b/aleksis/core/managers.py
@@ -1,10 +1,13 @@
 from datetime import date
 from typing import Union
 
+from django.apps import apps
 from django.contrib.sites.managers import CurrentSiteManager as _CurrentSiteManager
 from django.db.models import QuerySet
+from django.db.models.manager import Manager
 
 from calendarweek import CalendarWeek
+from polymorphic.managers import PolymorphicManager
 
 
 class CurrentSiteManagerWithoutMigrations(_CurrentSiteManager):
@@ -91,3 +94,33 @@ class GroupManager(CurrentSiteManagerWithoutMigrations):
 
 class GroupQuerySet(SchoolTermRelatedQuerySet):
     pass
+
+
+class UninstallRenitentPolymorphicManager(PolymorphicManager):
+    """A custom manager for django-polymorphic that filters out submodels of unavailable apps."""
+
+    def get_queryset(self):
+        DashboardWidget = apps.get_model("core", "DashboardWidget")
+        if self.model is DashboardWidget:
+            return super().get_queryset().instance_of(*self.model.__subclasses__())
+        else:
+            # Called on subclasses
+            return super().get_queryset()
+
+
+class InstalledWidgetsDashboardWidgetOrderManager(Manager):
+    """A manager that only returns DashboardWidgetOrder objects with an existing widget."""
+
+    def get_queryset(self):
+        queryset = super().get_queryset()
+
+        # Get the DashboardWidget model class without importing it to avoid a circular import
+        DashboardWidget = queryset.model.widget.field.related_model  # noqa
+        dashboard_widget_pks = DashboardWidget.objects.all().values("id")
+
+        # [obj["id"] for obj in list(Person.objects.all().values("id"))]
+        return super().get_queryset().filter(widget_id__in=dashboard_widget_pks)
+
+
+class PolymorphicCurrentSiteManager(_CurrentSiteManager, PolymorphicManager):
+    """Default manager for extensible, polymorphic models."""
diff --git a/aleksis/core/menus.py b/aleksis/core/menus.py
index 01e65b0aa6d3f0216917f840c3f17579aefda6a3..51f7cac4db999376939b757193489a19c1c1227d 100644
--- a/aleksis/core/menus.py
+++ b/aleksis/core/menus.py
@@ -11,11 +11,22 @@ MENUS = {
             "icon": "lock_open",
             "validators": ["menu_generator.validators.is_anonymous"],
         },
+        {
+            "name": _("Sign up"),
+            "url": "account_signup",
+            "icon": "how_to_reg",
+            "validators": [
+                "menu_generator.validators.is_anonymous",
+                ("aleksis.core.util.predicates.permission_validator", "core.can_register"),
+            ],
+        },
         {
             "name": _("Dashboard"),
             "url": "index",
             "icon": "home",
-            "validators": ["menu_generator.validators.is_authenticated"],
+            "validators": [
+                ("aleksis.core.util.predicates.permission_validator", "core.view_dashboard_rule")
+            ],
         },
         {
             "name": _("Notifications"),
@@ -54,6 +65,18 @@ MENUS = {
                     "icon": "phonelink_lock",
                     "validators": ["menu_generator.validators.is_authenticated",],
                 },
+                {
+                    "name": _("Change password"),
+                    "url": "account_change_password",
+                    "icon": "lock",
+                    "validators": [
+                        "menu_generator.validators.is_authenticated",
+                        (
+                            "aleksis.core.util.predicates.permission_validator",
+                            "core.can_change_password",
+                        ),
+                    ],
+                },
                 {
                     "name": _("Me"),
                     "url": "person",
@@ -72,6 +95,24 @@ MENUS = {
                         "aleksis.core.util.core_helpers.has_person",
                     ],
                 },
+                {
+                    "name": _("Third-party accounts"),
+                    "url": "socialaccount_connections",
+                    "icon": "public",
+                    "validators": [
+                        "menu_generator.validators.is_authenticated",
+                        "aleksis.core.util.core_helpers.has_person",
+                    ],
+                },
+                {
+                    "name": _("Authorized applications"),
+                    "url": "oauth2_provider:authorized-token-list",
+                    "icon": "touch_app",
+                    "validators": [
+                        "menu_generator.validators.is_authenticated",
+                        "aleksis.core.util.core_helpers.has_person",
+                    ],
+                },
             ],
         },
         {
@@ -89,7 +130,7 @@ MENUS = {
                     "validators": [
                         (
                             "aleksis.core.util.predicates.permission_validator",
-                            "core.view_announcements",
+                            "core.view_announcements_rule",
                         ),
                     ],
                 },
@@ -100,7 +141,7 @@ MENUS = {
                     "validators": [
                         (
                             "aleksis.core.util.predicates.permission_validator",
-                            "core.view_schoolterm",
+                            "core.view_schoolterm_rule",
                         ),
                     ],
                 },
@@ -111,7 +152,7 @@ MENUS = {
                     "validators": [
                         (
                             "aleksis.core.util.predicates.permission_validator",
-                            "core.view_dashboardwidget",
+                            "core.view_dashboardwidget_rule",
                         ),
                     ],
                 },
@@ -120,7 +161,10 @@ MENUS = {
                     "url": "data_management",
                     "icon": "view_list",
                     "validators": [
-                        ("aleksis.core.util.predicates.permission_validator", "core.manage_data"),
+                        (
+                            "aleksis.core.util.predicates.permission_validator",
+                            "core.manage_data_rule",
+                        ),
                     ],
                 },
                 {
@@ -130,7 +174,7 @@ MENUS = {
                     "validators": [
                         (
                             "aleksis.core.util.predicates.permission_validator",
-                            "core.view_system_status",
+                            "core.view_system_status_rule",
                         ),
                     ],
                 },
@@ -139,7 +183,10 @@ MENUS = {
                     "url": "impersonate-list",
                     "icon": "people",
                     "validators": [
-                        ("aleksis.core.util.predicates.permission_validator", "core.impersonate"),
+                        (
+                            "aleksis.core.util.predicates.permission_validator",
+                            "core.impersonate_rule",
+                        ),
                     ],
                 },
                 {
@@ -149,7 +196,7 @@ MENUS = {
                     "validators": [
                         (
                             "aleksis.core.util.predicates.permission_validator",
-                            "core.change_site_preferences",
+                            "core.change_site_preferences_rule",
                         ),
                     ],
                 },
@@ -176,6 +223,17 @@ MENUS = {
                     "icon": "settings",
                     "validators": ["menu_generator.validators.is_superuser",],
                 },
+                {
+                    "name": _("OAuth2 Applications"),
+                    "url": "oauth2_applications",
+                    "icon": "touch_app",
+                    "validators": [
+                        (
+                            "aleksis.core.util.predicates.permission_validator",
+                            "core.list_oauth_applications_rule",
+                        ),
+                    ],
+                },
             ],
         },
         {
@@ -184,7 +242,7 @@ MENUS = {
             "icon": "people",
             "root": True,
             "validators": [
-                ("aleksis.core.util.predicates.permission_validator", "core.view_people_menu")
+                ("aleksis.core.util.predicates.permission_validator", "core.view_people_menu_rule")
             ],
             "submenu": [
                 {
@@ -192,36 +250,31 @@ MENUS = {
                     "url": "persons",
                     "icon": "person",
                     "validators": [
-                        ("aleksis.core.util.predicates.permission_validator", "core.view_persons")
+                        (
+                            "aleksis.core.util.predicates.permission_validator",
+                            "core.view_persons_rule",
+                        )
                     ],
                 },
                 {
                     "name": _("Groups"),
                     "url": "groups",
                     "icon": "group",
-                    "validators": [
-                        ("aleksis.core.util.predicates.permission_validator", "core.view_groups")
-                    ],
-                },
-                {
-                    "name": _("Group types"),
-                    "url": "group_types",
-                    "icon": "category",
                     "validators": [
                         (
                             "aleksis.core.util.predicates.permission_validator",
-                            "core.view_group_type",
+                            "core.view_groups_rule",
                         )
                     ],
                 },
                 {
-                    "name": _("Persons and accounts"),
-                    "url": "persons_accounts",
-                    "icon": "person_add",
+                    "name": _("Group types"),
+                    "url": "group_types",
+                    "icon": "category",
                     "validators": [
                         (
                             "aleksis.core.util.predicates.permission_validator",
-                            "core.link_persons_accounts",
+                            "core.view_grouptypes_rule",
                         )
                     ],
                 },
@@ -232,7 +285,7 @@ MENUS = {
                     "validators": [
                         (
                             "aleksis.core.util.predicates.permission_validator",
-                            "core.assign_child_groups_to_groups",
+                            "core.assign_child_groups_to_groups_rule",
                         )
                     ],
                 },
@@ -243,7 +296,7 @@ MENUS = {
                     "validators": [
                         (
                             "aleksis.core.util.predicates.permission_validator",
-                            "core.view_additionalfield",
+                            "core.view_additionalfields_rule",
                         )
                     ],
                 },
@@ -257,7 +310,7 @@ MENUS = {
             "validators": [
                 (
                     "aleksis.core.util.predicates.permission_validator",
-                    "core.assign_child_groups_to_groups",
+                    "core.assign_child_groups_to_groups_rule",
                 )
             ],
         },
diff --git a/aleksis/core/migrations/0001_initial.py b/aleksis/core/migrations/0001_initial.py
index 14041d3eca0f1830dc4da2154f318ad7f7c1d516..aa7398e14a68e208ce75400c04177b752b1be8d4 100644
--- a/aleksis/core/migrations/0001_initial.py
+++ b/aleksis/core/migrations/0001_initial.py
@@ -25,7 +25,7 @@ class Migration(migrations.Migration):
         migrations.CreateModel(
             name='GlobalPermissions',
             fields=[
-                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+                ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
             ],
             options={
                 'default_permissions': (),
@@ -36,7 +36,7 @@ class Migration(migrations.Migration):
         migrations.CreateModel(
             name='AdditionalField',
             fields=[
-                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+                ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
                 ('extended_data', models.JSONField(default=dict, editable=False)),
                 ('title', models.CharField(max_length=255, verbose_name='Title of field')),
                 ('field_type', models.CharField(choices=[('BooleanField', 'Boolean (Yes/No)'), ('CharField', 'Text (one line)'), ('DateField', 'Date'), ('DateTimeField', 'Date and time'), ('DecimalField', 'Decimal number'), ('EmailField', 'E-mail address'), ('IntegerField', 'Integer'), ('GenericIPAddressField', 'IP address'), ('NullBooleanField', 'Boolean or empty (Yes/No/Neither)'), ('TextField', 'Text (multi-line)'), ('TimeField', 'Time'), ('URLField', 'URL / Link')], max_length=50, verbose_name='Type of field')),
@@ -53,7 +53,7 @@ class Migration(migrations.Migration):
         migrations.CreateModel(
             name='Announcement',
             fields=[
-                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+                ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
                 ('extended_data', models.JSONField(default=dict, editable=False)),
                 ('title', models.CharField(max_length=150, verbose_name='Title')),
                 ('description', models.TextField(blank=True, max_length=500, verbose_name='Description')),
@@ -70,7 +70,7 @@ class Migration(migrations.Migration):
         migrations.CreateModel(
             name='CustomMenu',
             fields=[
-                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+                ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
                 ('extended_data', models.JSONField(default=dict, editable=False)),
                 ('name', models.CharField(max_length=100, unique=True, verbose_name='Menu ID')),
                 ('site', models.ForeignKey(default=1, editable=False, on_delete=django.db.models.deletion.CASCADE, to='sites.Site')),
@@ -86,7 +86,7 @@ class Migration(migrations.Migration):
         migrations.CreateModel(
             name='Group',
             fields=[
-                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+                ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
                 ('extended_data', models.JSONField(default=dict, editable=False)),
                 ('name', models.CharField(max_length=255, unique=True, verbose_name='Long name')),
                 ('short_name', models.CharField(blank=True, max_length=255, null=True, unique=True, verbose_name='Short name')),
@@ -105,7 +105,7 @@ class Migration(migrations.Migration):
         migrations.CreateModel(
             name='Person',
             fields=[
-                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+                ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
                 ('extended_data', models.JSONField(default=dict, editable=False)),
                 ('is_active', models.BooleanField(default=True, verbose_name='Is person active?')),
                 ('first_name', models.CharField(max_length=255, verbose_name='First name')),
@@ -155,7 +155,7 @@ class Migration(migrations.Migration):
         migrations.CreateModel(
             name='SitePreferenceModel',
             fields=[
-                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+                ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
                 ('section', models.CharField(blank=True, db_index=True, default=None, max_length=150, null=True, verbose_name='Section Name')),
                 ('name', models.CharField(db_index=True, max_length=150, verbose_name='Name')),
                 ('raw_value', models.TextField(blank=True, null=True, verbose_name='Raw Value')),
@@ -166,7 +166,7 @@ class Migration(migrations.Migration):
         migrations.CreateModel(
             name='PersonPreferenceModel',
             fields=[
-                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+                ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
                 ('section', models.CharField(blank=True, db_index=True, default=None, max_length=150, null=True, verbose_name='Section Name')),
                 ('name', models.CharField(db_index=True, max_length=150, verbose_name='Name')),
                 ('raw_value', models.TextField(blank=True, null=True, verbose_name='Raw Value')),
@@ -177,7 +177,7 @@ class Migration(migrations.Migration):
         migrations.CreateModel(
             name='PersonGroupThrough',
             fields=[
-                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+                ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
                 ('extended_data', models.JSONField(default=dict, editable=False)),
                 ('group', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='core.Group')),
                 ('person', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='core.Person')),
@@ -193,7 +193,7 @@ class Migration(migrations.Migration):
         migrations.CreateModel(
             name='Notification',
             fields=[
-                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+                ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
                 ('extended_data', models.JSONField(default=dict, editable=False)),
                 ('sender', models.CharField(max_length=100, verbose_name='Sender')),
                 ('title', models.CharField(max_length=150, verbose_name='Title')),
@@ -215,7 +215,7 @@ class Migration(migrations.Migration):
         migrations.CreateModel(
             name='GroupType',
             fields=[
-                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+                ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
                 ('extended_data', models.JSONField(default=dict, editable=False)),
                 ('name', models.CharField(max_length=50, verbose_name='Title of type')),
                 ('description', models.CharField(max_length=500, verbose_name='Description')),
@@ -232,7 +232,7 @@ class Migration(migrations.Migration):
         migrations.CreateModel(
             name='GroupPreferenceModel',
             fields=[
-                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+                ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
                 ('section', models.CharField(blank=True, db_index=True, default=None, max_length=150, null=True, verbose_name='Section Name')),
                 ('name', models.CharField(db_index=True, max_length=150, verbose_name='Name')),
                 ('raw_value', models.TextField(blank=True, null=True, verbose_name='Raw Value')),
@@ -268,7 +268,7 @@ class Migration(migrations.Migration):
         migrations.CreateModel(
             name='DashboardWidget',
             fields=[
-                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+                ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
                 ('title', models.CharField(max_length=150, verbose_name='Widget Title')),
                 ('active', models.BooleanField(verbose_name='Activate Widget')),
                 ('polymorphic_ctype', models.ForeignKey(editable=False, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='polymorphic_core.dashboardwidget_set+', to='contenttypes.ContentType')),
@@ -282,7 +282,7 @@ class Migration(migrations.Migration):
         migrations.CreateModel(
             name='CustomMenuItem',
             fields=[
-                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+                ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
                 ('extended_data', models.JSONField(default=dict, editable=False)),
                 ('name', models.CharField(max_length=150, verbose_name='Name')),
                 ('url', models.URLField(verbose_name='Link')),
@@ -301,7 +301,7 @@ class Migration(migrations.Migration):
         migrations.CreateModel(
             name='AnnouncementRecipient',
             fields=[
-                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+                ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
                 ('extended_data', models.JSONField(default=dict, editable=False)),
                 ('recipient_id', models.PositiveIntegerField()),
                 ('announcement', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='recipients', to='core.Announcement')),
@@ -319,7 +319,7 @@ class Migration(migrations.Migration):
         migrations.CreateModel(
             name='Activity',
             fields=[
-                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+                ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
                 ('extended_data', models.JSONField(default=dict, editable=False)),
                 ('title', models.CharField(max_length=150, verbose_name='Title')),
                 ('description', models.TextField(max_length=500, verbose_name='Description')),
diff --git a/aleksis/core/migrations/0002_school_term.py b/aleksis/core/migrations/0002_school_term.py
index fba542ed9c4a7be9b60e94d28414ade4189a6d6b..2799342db2350373a25d2c7572e0da912e3bc8df 100644
--- a/aleksis/core/migrations/0002_school_term.py
+++ b/aleksis/core/migrations/0002_school_term.py
@@ -16,7 +16,7 @@ class Migration(migrations.Migration):
         migrations.CreateModel(
             name='SchoolTerm',
             fields=[
-                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+                ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
                 ('extended_data', models.JSONField(default=dict, editable=False)),
                 ('name', models.CharField(max_length=255, unique=True, verbose_name='Name')),
                 ('date_start', models.DateField(verbose_name='Start date')),
diff --git a/aleksis/core/migrations/0007_dashboard_widget_order.py b/aleksis/core/migrations/0007_dashboard_widget_order.py
index 9387f4ca7c0dbb3307d2a3681f091e4bf631f847..78c1c8bcb372a8fd9abfa6bd578e88c70e61d01f 100644
--- a/aleksis/core/migrations/0007_dashboard_widget_order.py
+++ b/aleksis/core/migrations/0007_dashboard_widget_order.py
@@ -17,7 +17,7 @@ class Migration(migrations.Migration):
         migrations.CreateModel(
             name='DashboardWidgetOrder',
             fields=[
-                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+                ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
                 ('extended_data', models.JSONField(default=dict, editable=False)),
                 ('order', models.PositiveIntegerField(verbose_name='Order')),
                 ('person', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='core.person', verbose_name='Person')),
@@ -29,7 +29,6 @@ class Migration(migrations.Migration):
                 'verbose_name_plural': 'Dashboard widget orders',
             },
             managers=[
-                ('objects', django.contrib.sites.managers.CurrentSiteManager()),
             ],
         ),
     ]
diff --git a/aleksis/core/migrations/0008_data_check_result.py b/aleksis/core/migrations/0008_data_check_result.py
index 3f8285e22a6af4109a67d6baddccd05e69b1bd6e..df2d51119513f8033da981c7cd12c6f9a93425b9 100644
--- a/aleksis/core/migrations/0008_data_check_result.py
+++ b/aleksis/core/migrations/0008_data_check_result.py
@@ -20,7 +20,7 @@ class Migration(migrations.Migration):
             fields=[
                 (
                     "id",
-                    models.AutoField(
+                    models.BigAutoField(
                         auto_created=True, primary_key=True, serialize=False, verbose_name="ID"
                     ),
                 ),
diff --git a/aleksis/core/migrations/0013_pdf_file.py b/aleksis/core/migrations/0013_pdf_file.py
index 63b8a95607ce306f8d14d8b4266e184e12a26838..4ae06cffe58f3d90798c05e67b6ffd8843f14fef 100644
--- a/aleksis/core/migrations/0013_pdf_file.py
+++ b/aleksis/core/migrations/0013_pdf_file.py
@@ -27,7 +27,7 @@ class Migration(migrations.Migration):
         migrations.CreateModel(
             name='PDFFile',
             fields=[
-                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+                ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
                 ('extended_data', models.JSONField(default=dict, editable=False)),
                 ('expires_at', models.DateTimeField(default=aleksis.core.models.PDFFile._get_default_expiration, verbose_name='File expires at')),
                 ('html', models.TextField(verbose_name='Rendered HTML')),
diff --git a/aleksis/core/migrations/0014_alter_pdffile_file.py b/aleksis/core/migrations/0014_alter_pdffile_file.py
new file mode 100644
index 0000000000000000000000000000000000000000..b2cc3beb5e9faf8ab474749c8eec0be031b493db
--- /dev/null
+++ b/aleksis/core/migrations/0014_alter_pdffile_file.py
@@ -0,0 +1,21 @@
+# Generated by Django 3.2 on 2021-04-17 18:47
+
+from django.db import migrations, models
+
+def _get_upload_path(instance, filename):  # noqa
+    return f"pdfs/{instance.secret}.pdf"
+
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        ('core', '0013_pdf_file'),
+    ]
+
+    operations = [
+        migrations.AlterField(
+            model_name='pdffile',
+            name='file',
+            field=models.FileField(blank=True, null=True, upload_to=_get_upload_path, verbose_name='Generated PDF file'),
+        ),
+    ]
diff --git a/aleksis/core/migrations/0015_oauth_permissions.py b/aleksis/core/migrations/0015_oauth_permissions.py
new file mode 100644
index 0000000000000000000000000000000000000000..4ef1fe15883bef465191101396e1be4bafbc1ce1
--- /dev/null
+++ b/aleksis/core/migrations/0015_oauth_permissions.py
@@ -0,0 +1,18 @@
+# Generated by Django 3.1.7 on 2021-03-02 19:15
+
+from django.db import migrations, models
+import django.utils.timezone
+
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        ('core', '0014_alter_pdffile_file'),
+    ]
+
+    operations = [
+        migrations.AlterModelOptions(
+            name='globalpermissions',
+            options={'default_permissions': (), 'managed': False, 'permissions': (('view_system_status', 'Can view system status'), ('link_persons_accounts', 'Can link persons to accounts'), ('manage_data', 'Can manage data'), ('impersonate', 'Can impersonate'), ('search', 'Can use search'), ('change_site_preferences', 'Can change site preferences'), ('change_person_preferences', 'Can change person preferences'), ('change_group_preferences', 'Can change group preferences'), ('add_oauth_applications', 'Can add oauth applications'), ('list_oauth_applications', 'Can list oauth applications'), ('view_oauth_applications', 'Can view oauth applications'), ('update_oauth_applications', 'Can update oauth applications'), ('delete_oauth_applications', 'Can delete oauth applications'), ('test_pdf', 'Can test PDF generation'))},
+        ),
+    ]
diff --git a/aleksis/core/migrations/0016_taskuserassignment.py b/aleksis/core/migrations/0016_taskuserassignment.py
new file mode 100644
index 0000000000000000000000000000000000000000..59328cf6b702ddba0e37d121f5f5ce264c04d657
--- /dev/null
+++ b/aleksis/core/migrations/0016_taskuserassignment.py
@@ -0,0 +1,36 @@
+# Generated by Django 3.2 on 2021-05-09 10:55
+
+from django.conf import settings
+import django.contrib.sites.managers
+from django.db import migrations, models
+import django.db.models.deletion
+
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        ('sites', '0002_alter_domain_unique'),
+        ('django_celery_results', '0008_chordcounter'),
+        migrations.swappable_dependency(settings.AUTH_USER_MODEL),
+        ('core', '0015_oauth_permissions'),
+    ]
+
+    operations = [
+        migrations.CreateModel(
+            name='TaskUserAssignment',
+            fields=[
+                ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+                ('extended_data', models.JSONField(default=dict, editable=False)),
+                ('site', models.ForeignKey(default=1, editable=False, on_delete=django.db.models.deletion.CASCADE, to='sites.site')),
+                ('task_result', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='django_celery_results.taskresult', verbose_name='Task result')),
+                ('user', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL, verbose_name='Task user')),
+            ],
+            options={
+                'verbose_name': 'Task user assignment',
+                'verbose_name_plural': 'Task user assignments',
+            },
+            managers=[
+                ('objects', django.contrib.sites.managers.CurrentSiteManager()),
+            ],
+        ),
+    ]
diff --git a/aleksis/core/migrations/0017_dashboardwidget_broken.py b/aleksis/core/migrations/0017_dashboardwidget_broken.py
new file mode 100644
index 0000000000000000000000000000000000000000..7a9f11d4296ab168b2eed9ed7d6054d163c535f9
--- /dev/null
+++ b/aleksis/core/migrations/0017_dashboardwidget_broken.py
@@ -0,0 +1,18 @@
+# Generated by Django 3.1.4 on 2021-02-19 13:43
+
+from django.db import migrations, models
+
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        ('core', '0016_taskuserassignment'),
+    ]
+
+    operations = [
+        migrations.AddField(
+            model_name='dashboardwidget',
+            name='broken',
+            field=models.BooleanField(default=False, verbose_name='Widget is broken'),
+        ),
+    ]
diff --git a/aleksis/core/migrations/0018_pdffile_html_file.py b/aleksis/core/migrations/0018_pdffile_html_file.py
new file mode 100644
index 0000000000000000000000000000000000000000..d802526def2b33b73362890f0c362d174f53e76c
--- /dev/null
+++ b/aleksis/core/migrations/0018_pdffile_html_file.py
@@ -0,0 +1,31 @@
+# Generated by Django 3.2.3 on 2021-05-20 14:25
+
+from django.db import migrations, models
+from django.core.files.base import ContentFile
+
+def _get_default():
+    return ContentFile("")
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        ('core', '0017_dashboardwidget_broken'),
+    ]
+
+    operations = [
+        migrations.RemoveField(
+            model_name='pdffile',
+            name='html',
+        ),
+        migrations.AddField(
+            model_name='pdffile',
+            name='html_file',
+            field=models.FileField(default=_get_default, upload_to='pdfs/', verbose_name='Generated HTML file'),
+            preserve_default=False,
+        ),
+        migrations.AlterField(
+            model_name='pdffile',
+            name='file',
+            field=models.FileField(blank=True, null=True, upload_to='pdfs/', verbose_name='Generated PDF file'),
+        ),
+    ]
diff --git a/aleksis/core/migrations/0019_fix_uniqueness_per_site.py b/aleksis/core/migrations/0019_fix_uniqueness_per_site.py
new file mode 100644
index 0000000000000000000000000000000000000000..ca1dc482eafe5e2622a0ed1be4f3a0544f599661
--- /dev/null
+++ b/aleksis/core/migrations/0019_fix_uniqueness_per_site.py
@@ -0,0 +1,56 @@
+# Generated by Django 3.2.3 on 2021-05-21 16:23
+
+from django.db import migrations, models
+
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        ('core', '0018_pdffile_html_file'),
+    ]
+
+    operations = [
+        migrations.AlterField(
+            model_name='custommenu',
+            name='name',
+            field=models.CharField(max_length=100, verbose_name='Menu ID'),
+        ),
+        migrations.AlterField(
+            model_name='person',
+            name='short_name',
+            field=models.CharField(blank=True, max_length=255, null=True, verbose_name='Short name'),
+        ),
+        migrations.AlterField(
+            model_name='schoolterm',
+            name='name',
+            field=models.CharField(max_length=255, verbose_name='Name'),
+        ),
+        migrations.AddConstraint(
+            model_name='additionalfield',
+            constraint=models.UniqueConstraint(fields=('site_id', 'title'), name='unique_title_per_site'),
+        ),
+        migrations.AddConstraint(
+            model_name='custommenu',
+            constraint=models.UniqueConstraint(fields=('site_id', 'name'), name='unique_menu_name_per_site'),
+        ),
+        migrations.AddConstraint(
+            model_name='custommenuitem',
+            constraint=models.UniqueConstraint(fields=('menu', 'name'), name='unique_name_per_menu'),
+        ),
+        migrations.AddConstraint(
+            model_name='grouptype',
+            constraint=models.UniqueConstraint(fields=('site_id', 'name'), name='unique_group_type_name_per_site'),
+        ),
+        migrations.AddConstraint(
+            model_name='person',
+            constraint=models.UniqueConstraint(fields=('site_id', 'short_name'), name='unique_short_name_per_site'),
+        ),
+        migrations.AddConstraint(
+            model_name='schoolterm',
+            constraint=models.UniqueConstraint(fields=('site_id', 'name'), name='unique_school_term_name_per_site'),
+        ),
+        migrations.AddConstraint(
+            model_name='schoolterm',
+            constraint=models.UniqueConstraint(fields=('site_id', 'date_start', 'date_end'), name='unique_school_term_dates_per_site'),
+        ),
+    ]
diff --git a/aleksis/core/migrations/0020_pdf_file_person_optional.py b/aleksis/core/migrations/0020_pdf_file_person_optional.py
new file mode 100644
index 0000000000000000000000000000000000000000..c98e5fda647102d06b95ad8b3be36257c1b15456
--- /dev/null
+++ b/aleksis/core/migrations/0020_pdf_file_person_optional.py
@@ -0,0 +1,19 @@
+# Generated by Django 3.2.4 on 2021-07-24 13:14
+
+from django.db import migrations, models
+import django.db.models.deletion
+
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        ('core', '0019_fix_uniqueness_per_site'),
+    ]
+
+    operations = [
+        migrations.AlterField(
+            model_name='pdffile',
+            name='person',
+            field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='pdf_files', to='core.person', verbose_name='Owner'),
+        ),
+    ]
diff --git a/aleksis/core/migrations/0021_drop_persons_accounts_perm.py b/aleksis/core/migrations/0021_drop_persons_accounts_perm.py
new file mode 100644
index 0000000000000000000000000000000000000000..576efcec16e706a27059d80d9c256052c0e74dab
--- /dev/null
+++ b/aleksis/core/migrations/0021_drop_persons_accounts_perm.py
@@ -0,0 +1,17 @@
+# Generated by Django 3.2.8 on 2021-10-24 13:43
+
+from django.db import migrations
+
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        ('core', '0020_pdf_file_person_optional'),
+    ]
+
+    operations = [
+        migrations.AlterModelOptions(
+            name='globalpermissions',
+            options={'default_permissions': (), 'managed': False, 'permissions': (('view_system_status', 'Can view system status'), ('manage_data', 'Can manage data'), ('impersonate', 'Can impersonate'), ('search', 'Can use search'), ('change_site_preferences', 'Can change site preferences'), ('change_person_preferences', 'Can change person preferences'), ('change_group_preferences', 'Can change group preferences'), ('add_oauth_applications', 'Can add oauth applications'), ('list_oauth_applications', 'Can list oauth applications'), ('view_oauth_applications', 'Can view oauth applications'), ('update_oauth_applications', 'Can update oauth applications'), ('delete_oauth_applications', 'Can delete oauth applications'), ('test_pdf', 'Can test PDF generation'))},
+        ),
+    ]
diff --git a/aleksis/core/migrations/0022_public_favicon.py b/aleksis/core/migrations/0022_public_favicon.py
new file mode 100644
index 0000000000000000000000000000000000000000..13864fdaebfb1811ac7629ed891c745595afc5bc
--- /dev/null
+++ b/aleksis/core/migrations/0022_public_favicon.py
@@ -0,0 +1,28 @@
+# Generated by Django 3.2.4 on 2021-07-24 13:14
+import os
+
+from django.db import migrations, models
+import django.db.models.deletion
+
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        ('core', '0021_drop_persons_accounts_perm'),
+        ('favicon', '0004_faviconimg_favicon_size_rel_unique'),
+    ]
+
+    def _migrate_favicons(apps, schema_editor):
+        FaviconImg = apps.get_model('favicon', "FaviconImg")
+        for favicon_img in FaviconImg.objects.all():
+            old_name = favicon_img.faviconImage.name
+            new_name = os.path.join("public", old_name)
+            storage = favicon_img.faviconImage.storage
+            if storage.exists(old_name):
+                storage.save(new_name, favicon_img.faviconImage.file)
+                favicon_img.faviconImage.name = new_name
+                favicon_img.save()
+
+    operations = [
+        migrations.RunPython(_migrate_favicons)
+    ]
diff --git a/aleksis/core/migrations/0023_oauth_application_model.py b/aleksis/core/migrations/0023_oauth_application_model.py
new file mode 100644
index 0000000000000000000000000000000000000000..177c50317ca991ceb86e01563273149386ddb3f2
--- /dev/null
+++ b/aleksis/core/migrations/0023_oauth_application_model.py
@@ -0,0 +1,128 @@
+# Generated by Django 3.2.8 on 2021-11-04 09:52
+
+# Manual migration to replace Django OAuth Toolkit's models with custom models
+# This migration is special, because it needs to work correctly both when applied
+# during the initial database migration of the whole project, and when updating
+# from an older schema version
+#
+# It strictly has to be applied before Django OAuth Toolkit's initial migration,
+# because when the planner introspects existing models, it will stumble upon the swapped
+# foreign key relations that now point to our custom models (as configured in settings).
+#
+# When run during the initial database migration, this simply works by setting run_before.
+#
+# When run during an update later, Django will refuse to migrate, because run_before will
+# lead to an inconsistent migration history. Django OAuth Toolkit's migrations have already
+# been run in the past, and we cannot move a new migration to before that.
+#
+# Therefore, we treat the two cases explicitly:
+#  – If tables from the oauth2_provider label exist, we…
+#     …assume that its migrations have been run in the past, and we are in an udpate
+#     …forcefully drop the migration from the database to convince Django to let our own
+#      migration run first
+#     …rename the tables to the new app label through raw SQL, passing the original
+#      operations that Django OAuth Toolkit would have done to the planner to spoof them
+#      in the state
+#  – If no tables from the oauth2_provider label exist, we…
+#     …assume that we are in the first ever migration of the project
+#     …introspect the operations that Django OAuth Toolkit would do, and re-create them for
+#      our own app
+#  – In both cases, we then let Django OAuth Toolkit do its migrations like normal; they will
+#    will be no-ops as we swapped all models
+
+from django.apps import apps as global_apps
+from django.conf import settings
+from django.db import connection, migrations
+from django.utils.module_loading import import_string
+
+MODEL_NAMES = ("Application", "Grant", "AccessToken", "IDToken", "RefreshToken")
+_MIGRATION_NAMES = ["0001_initial", "0002_auto_20190406_1805", "0003_auto_20201211_1314", "0004_auto_20200902_2022"]
+
+
+class Migration(migrations.Migration):
+    """Empty dummy migration to satisfy loader in utility functions."""
+    # This migration will later be replaced by the real one.
+    # We only need a class called Migration here so the loader we use in
+    # the functions below does not complain that this migration does not
+    # have a Migration class (yet).
+    pass
+
+def _get_oauth_migrations():
+    """Get all original migrations from Django OAuth Toolkit."""
+    loader = migrations.loader.MigrationLoader(connection)
+    return [loader.get_migration("oauth2_provider", name) for name in _MIGRATION_NAMES]
+
+def _get_oauth_migration_operations(for_model=None):
+    """Get all original operations from Django Oauth Toolkit and re-create them for the new models."""
+    operations = []
+    for migration in _get_oauth_migrations():
+        for old_operation in migration.operations:
+            # We need to re-create a new Operation class because Operations
+            # are immutably linked to an app
+            op_name, args, kwargs = old_operation.deconstruct()
+            if not "." in op_name:
+                op_name = f"django.db.migrations.{op_name}"
+            op_cls = import_string(op_name)
+            if "model_name" in kwargs:
+                # If we are not interested for this model, skip it
+                if for_model is not None and for_model != "oauth" + kwargs["model_name"].lower():
+                    continue
+                # We are modifying a field. Replace the model name it belongs to
+                kwargs["model_name"] = "OAuth" + kwargs["model_name"]
+            elif "name" in kwargs:
+                # If we are not interested for this model, skip it
+                if for_model is not None and for_model != "oauth" + kwargs["name"].lower():
+                    continue
+                # We are modifying a model. Replace the full name
+                kwargs["name"] = "OAuth" + kwargs["name"]
+            operations.append(op_cls(*args, **kwargs))
+
+    return operations
+
+def _get_original_models():
+    """Get the original models of Django OAuth Toolkit."""
+    try:
+        return [global_apps.get_model("oauth2_provider", name) for name in MODEL_NAMES]
+    except LookupError:
+        return []
+
+def _get_new_models():
+    """Get the new customised models."""
+    return [global_apps.get_model("core", f"OAuth{name}") for name in MODEL_NAMES]
+
+def _original_tables_exist():
+    """Check if original Django OAuth Toolkit tables exist."""
+    original_tables = set([model._meta.db_table for model in _get_original_models()])
+    return bool(set(connection.introspection.table_names()) & original_tables)
+
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        migrations.swappable_dependency(settings.AUTH_USER_MODEL),
+        ('core', '0022_public_favicon'),
+    ]
+
+    run_before = [("oauth2_provider", _MIGRATION_NAMES[0])]
+
+    operations = []
+    if _original_tables_exist():
+        # If original tables existed, we nned to copy data and remove them afterwards
+        for OldModel, NewModel in zip(_get_original_models(), _get_new_models()):
+            operations.append(migrations.RunSQL(
+                f"ALTER TABLE IF EXISTS {OldModel._meta.db_table} RENAME TO {NewModel._meta.db_table};",
+                reverse_sql=f"ALTER TABLE IF EXISTS {NewModel._meta.db_table} RENAME TO {OldModel._meta.db_table};",
+                # We copy the original migrations here to trick the planner into believing the migrations had run normally
+                state_operations=_get_oauth_migration_operations(NewModel._meta.model_name)
+            ))
+    else:
+        # We need to copy Django OAuth Toolkit's migrations here to re-create
+        # all models, because we cannot auto-generate them due to the big swappable
+        # model circular dependency rabbit hole
+        operations += _get_oauth_migration_operations()
+
+
+if _original_tables_exist():
+    # Fake-unapply migrations so the planner allows run_before
+    for name in _MIGRATION_NAMES:
+        connection.cursor().execute(f"DELETE FROM django_migrations WHERE app='oauth2_provider' AND name='{name}';")
diff --git a/aleksis/core/migrations/0024_oauth_grant_types_optional.py b/aleksis/core/migrations/0024_oauth_grant_types_optional.py
new file mode 100644
index 0000000000000000000000000000000000000000..9586d817e6810f4186a9ad9f1e480dda454cbc14
--- /dev/null
+++ b/aleksis/core/migrations/0024_oauth_grant_types_optional.py
@@ -0,0 +1,19 @@
+# Generated by Django 3.2.8 on 2021-11-04 19:31
+
+from django.db import migrations, models
+
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        ('core', '0023_oauth_application_model'),
+        ('oauth2_provider', '0004_auto_20200902_2022'),
+    ]
+
+    operations = [
+        migrations.AlterField(
+            model_name='oauthapplication',
+            name='authorization_grant_type',
+            field=models.CharField(blank=True, choices=[('authorization-code', 'Authorization code'), ('implicit', 'Implicit'), ('password', 'Resource owner password-based'), ('client-credentials', 'Client credentials'), ('openid-hybrid', 'OpenID connect hybrid')], max_length=32, null=True),
+        ),
+    ]
diff --git a/aleksis/core/migrations/0025_oauth_align_user_fk.py b/aleksis/core/migrations/0025_oauth_align_user_fk.py
new file mode 100644
index 0000000000000000000000000000000000000000..6fab3308cb5c32e443e34964dad0a020efb62d29
--- /dev/null
+++ b/aleksis/core/migrations/0025_oauth_align_user_fk.py
@@ -0,0 +1,41 @@
+# Generated by Django 3.2.8 on 2021-11-04 19:35
+
+from django.conf import settings
+from django.db import migrations, models
+import django.db.models.deletion
+
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        migrations.swappable_dependency(settings.AUTH_USER_MODEL),
+        ('core', '0024_oauth_grant_types_optional'),
+    ]
+
+    operations = [
+        migrations.AlterField(
+            model_name='oauthaccesstoken',
+            name='user',
+            field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='core_oauthaccesstoken', to=settings.AUTH_USER_MODEL),
+        ),
+        migrations.AlterField(
+            model_name='oauthapplication',
+            name='user',
+            field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='core_oauthapplication', to=settings.AUTH_USER_MODEL),
+        ),
+        migrations.AlterField(
+            model_name='oauthgrant',
+            name='user',
+            field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='core_oauthgrant', to=settings.AUTH_USER_MODEL),
+        ),
+        migrations.AlterField(
+            model_name='oauthidtoken',
+            name='user',
+            field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='core_oauthidtoken', to=settings.AUTH_USER_MODEL),
+        ),
+        migrations.AlterField(
+            model_name='oauthrefreshtoken',
+            name='user',
+            field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='core_oauthrefreshtoken', to=settings.AUTH_USER_MODEL),
+        ),
+    ]
diff --git a/aleksis/core/migrations/0026_oauthapplication_allowed_scopes.py b/aleksis/core/migrations/0026_oauthapplication_allowed_scopes.py
new file mode 100644
index 0000000000000000000000000000000000000000..23e1b40b56a2ce2fbe1082b8728adea642f3e82b
--- /dev/null
+++ b/aleksis/core/migrations/0026_oauthapplication_allowed_scopes.py
@@ -0,0 +1,19 @@
+# Generated by Django 3.2.8 on 2021-11-05 10:37
+
+import django.contrib.postgres.fields
+from django.db import migrations, models
+
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        ('core', '0025_oauth_align_user_fk'),
+    ]
+
+    operations = [
+        migrations.AddField(
+            model_name='oauthapplication',
+            name='allowed_scopes',
+            field=django.contrib.postgres.fields.ArrayField(base_field=models.CharField(max_length=32), blank=True, null=True, size=None, verbose_name='Allowed scopes that clients can request'),
+        ),
+    ]
diff --git a/aleksis/core/migrations/0027_person_place_of_birth.py b/aleksis/core/migrations/0027_person_place_of_birth.py
new file mode 100644
index 0000000000000000000000000000000000000000..53c5bf169a3e852f6a83d9681bf6a9dd7666bb1d
--- /dev/null
+++ b/aleksis/core/migrations/0027_person_place_of_birth.py
@@ -0,0 +1,18 @@
+# Generated by Django 3.2.8 on 2021-11-02 20:32
+
+from django.db import migrations, models
+
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        ('core', '0026_oauthapplication_allowed_scopes'),
+    ]
+
+    operations = [
+        migrations.AddField(
+            model_name='person',
+            name='place_of_birth',
+            field=models.CharField(blank=True, max_length=255, null=True, verbose_name='Place of birth'),
+        ),
+    ]
diff --git a/aleksis/core/mixins.py b/aleksis/core/mixins.py
index 26cef894ade3c7645d9ed957a21897ebd3d1071a..2eb8bd1fcdafef42e729b563b7edf152f6f7825d 100644
--- a/aleksis/core/mixins.py
+++ b/aleksis/core/mixins.py
@@ -1,7 +1,7 @@
 # flake8: noqa: DJ12
 
 from datetime import datetime
-from typing import Any, Callable, List, Optional, Tuple, Union
+from typing import Any, Callable, List, Optional, Union
 
 from django.conf import settings
 from django.contrib import messages
@@ -25,9 +25,16 @@ from guardian.admin import GuardedModelAdmin
 from guardian.core import ObjectPermissionChecker
 from jsonstore.fields import IntegerField, JSONFieldMixin
 from material.base import Layout, LayoutNode
+from polymorphic.base import PolymorphicModelBase
+from polymorphic.managers import PolymorphicManager
+from polymorphic.models import PolymorphicModel
 from rules.contrib.admin import ObjectPermissionsModelAdmin
 
-from aleksis.core.managers import CurrentSiteManagerWithoutMigrations, SchoolTermRelatedQuerySet
+from aleksis.core.managers import (
+    CurrentSiteManagerWithoutMigrations,
+    PolymorphicCurrentSiteManager,
+    SchoolTermRelatedQuerySet,
+)
 
 
 class _ExtensibleModelBase(models.base.ModelBase):
@@ -132,7 +139,7 @@ class ExtensibleModel(models.Model, metaclass=_ExtensibleModelBase):
         pass
 
     @property
-    def versions(self) -> List[Tuple[str, Tuple[Any, Any]]]:
+    def versions(self) -> list[tuple[str, tuple[Any, Any]]]:
         """Get all versions of this object from django-reversion.
 
         Includes diffs to previous version.
@@ -281,8 +288,8 @@ class ExtensibleModel(models.Model, metaclass=_ExtensibleModelBase):
 
     @classmethod
     def syncable_fields(
-        cls, recursive: bool = True, exclude_remotes: List = []
-    ) -> List[models.Field]:
+        cls, recursive: bool = True, exclude_remotes: list = []
+    ) -> list[models.Field]:
         """Collect all fields that can be synced on a model.
 
         If recursive is True, it recurses into related models and generates virtual
@@ -327,14 +334,14 @@ class ExtensibleModel(models.Model, metaclass=_ExtensibleModelBase):
         return fields
 
     @classmethod
-    def syncable_fields_choices(cls) -> Tuple[Tuple[str, str]]:
+    def syncable_fields_choices(cls) -> tuple[tuple[str, str]]:
         """Collect all fields that can be synced on a model."""
         return tuple(
             [(field.name, field.verbose_name or field.name) for field in cls.syncable_fields()]
         )
 
     @classmethod
-    def syncable_fields_choices_lazy(cls) -> Callable[[], Tuple[Tuple[str, str]]]:
+    def syncable_fields_choices_lazy(cls) -> Callable[[], tuple[tuple[str, str]]]:
         """Collect all fields that can be synced on a model."""
         return lazy(cls.syncable_fields_choices, tuple)
 
@@ -361,6 +368,22 @@ class ExtensibleModel(models.Model, metaclass=_ExtensibleModelBase):
         abstract = True
 
 
+class _ExtensiblePolymorphicModelBase(_ExtensibleModelBase, PolymorphicModelBase):
+    """Base class for extensible, polymorphic models."""
+
+
+class ExtensiblePolymorphicModel(
+    ExtensibleModel, PolymorphicModel, metaclass=_ExtensiblePolymorphicModelBase
+):
+    """Model class for extensible, polymorphic models."""
+
+    objects = PolymorphicCurrentSiteManager()
+    objects_all_sites = PolymorphicManager()
+
+    class Meta:
+        abstract = True
+
+
 class PureDjangoModel(object):
     """No-op mixin to mark a model as deliberately not using ExtensibleModel."""
 
@@ -400,17 +423,16 @@ class ExtensibleForm(ModelForm, metaclass=_ExtensibleFormMetaclass):
     This mixin adds functionality which allows
     - apps to add layout nodes to the layout used by django-material
 
-    Add layout nodes
-    ================
+    :Add layout nodes:
+
+    .. code-block:: python
 
-    ```
-    from material import Fieldset
+        from material import Fieldset
 
-    from aleksis.core.forms import ExampleForm
+        from aleksis.core.forms import ExampleForm
 
-    node = Fieldset("field_name")
-    ExampleForm.add_node_to_layout(node)
-    ```
+        node = Fieldset("field_name")
+        ExampleForm.add_node_to_layout(node)
 
     """
 
diff --git a/aleksis/core/models.py b/aleksis/core/models.py
index ad975331da48c3def0863b2d56904d0dad2e67a9..8a8ce7a6eea88c88c622c8bbb74850b5c9154f6c 100644
--- a/aleksis/core/models.py
+++ b/aleksis/core/models.py
@@ -9,11 +9,13 @@ from django.contrib.auth import get_user_model
 from django.contrib.auth.models import Group as DjangoGroup
 from django.contrib.contenttypes.fields import GenericForeignKey
 from django.contrib.contenttypes.models import ContentType
+from django.contrib.postgres.fields import ArrayField
 from django.contrib.sites.models import Site
 from django.core.exceptions import ValidationError
 from django.core.validators import MaxValueValidator
 from django.db import models, transaction
 from django.db.models import QuerySet
+from django.db.models.signals import m2m_changed
 from django.dispatch import receiver
 from django.forms.widgets import Media
 from django.urls import reverse
@@ -23,20 +25,32 @@ from django.utils.text import slugify
 from django.utils.translation import gettext_lazy as _
 
 import jsonstore
+from cachalot.api import cachalot_disabled
 from cache_memoize import cache_memoize
+from django_celery_results.models import TaskResult
 from dynamic_preferences.models import PerInstancePreferenceModel
 from model_utils import FieldTracker
 from model_utils.models import TimeStampedModel
+from oauth2_provider.models import (
+    AbstractAccessToken,
+    AbstractApplication,
+    AbstractGrant,
+    AbstractIDToken,
+    AbstractRefreshToken,
+)
 from phonenumber_field.modelfields import PhoneNumberField
 from polymorphic.models import PolymorphicModel
+from templated_email import send_templated_mail
 
-from aleksis.core.data_checks import DataCheck, DataCheckRegistry
+from aleksis.core.data_checks import BrokenDashboardWidgetDataCheck, DataCheck, DataCheckRegistry
 
 from .managers import (
     CurrentSiteManagerWithoutMigrations,
     GroupManager,
     GroupQuerySet,
+    InstalledWidgetsDashboardWidgetOrderManager,
     SchoolTermQuerySet,
+    UninstallRenitentPolymorphicManager,
 )
 from .mixins import (
     ExtensibleModel,
@@ -72,7 +86,7 @@ class SchoolTerm(ExtensibleModel):
 
     objects = CurrentSiteManagerWithoutMigrations.from_queryset(SchoolTermQuerySet)()
 
-    name = models.CharField(verbose_name=_("Name"), max_length=255, unique=True)
+    name = models.CharField(verbose_name=_("Name"), max_length=255)
 
     date_start = models.DateField(verbose_name=_("Start date"))
     date_end = models.DateField(verbose_name=_("End date"))
@@ -110,6 +124,15 @@ class SchoolTerm(ExtensibleModel):
     class Meta:
         verbose_name = _("School term")
         verbose_name_plural = _("School terms")
+        constraints = [
+            models.UniqueConstraint(
+                fields=["site_id", "name"], name="unique_school_term_name_per_site"
+            ),
+            models.UniqueConstraint(
+                fields=["site_id", "date_start", "date_end"],
+                name="unique_school_term_dates_per_site",
+            ),
+        ]
 
 
 class Person(ExtensibleModel):
@@ -130,6 +153,11 @@ class Person(ExtensibleModel):
             ("view_person_groups", _("Can view persons groups")),
             ("view_personal_details", _("Can view personal details")),
         )
+        constraints = [
+            models.UniqueConstraint(
+                fields=["site_id", "short_name"], name="unique_short_name_per_site"
+            ),
+        ]
 
     icon_ = "person"
 
@@ -152,7 +180,7 @@ class Person(ExtensibleModel):
     )
 
     short_name = models.CharField(
-        verbose_name=_("Short name"), max_length=255, blank=True, null=True, unique=True  # noqa
+        verbose_name=_("Short name"), max_length=255, blank=True, null=True  # noqa
     )
 
     street = models.CharField(verbose_name=_("Street"), max_length=255, blank=True)
@@ -166,6 +194,9 @@ class Person(ExtensibleModel):
     email = models.EmailField(verbose_name=_("E-mail address"), blank=True)
 
     date_of_birth = models.DateField(verbose_name=_("Date of birth"), blank=True, null=True)
+    place_of_birth = models.CharField(
+        verbose_name=_("Place of birth"), max_length=255, blank=True, null=True
+    )
     sex = models.CharField(verbose_name=_("Sex"), max_length=1, choices=SEX_CHOICES, blank=True)
 
     photo = models.ImageField(verbose_name=_("Photo"), blank=True, null=True)
@@ -310,6 +341,22 @@ class Person(ExtensibleModel):
             if force or not self.primary_group:
                 self.primary_group = self.member_of.filter(**{f"{field}__regex": pattern}).first()
 
+    def notify_about_changed_data(
+        self, changed_fields: Iterable[str], recipients: Optional[List[str]] = None
+    ):
+        """Notify (configured) recipients about changed data of this person."""
+        context = {"person": self, "changed_fields": changed_fields}
+        recipients = recipients or [
+            get_site_preferences()["account__person_change_notification_contact"]
+        ]
+        send_templated_mail(
+            template_name="person_changed",
+            from_email=self.mail_sender_via,
+            headers={"Reply-To": self.mail_sender, "Sender": self.mail_sender,},
+            recipient_list=recipients,
+            context=context,
+        )
+
 
 class DummyPerson(Person):
     """A dummy person that is not stored into the database.
@@ -342,6 +389,9 @@ class AdditionalField(ExtensibleModel):
     class Meta:
         verbose_name = _("Addtitional field for groups")
         verbose_name_plural = _("Addtitional fields for groups")
+        constraints = [
+            models.UniqueConstraint(fields=["site_id", "title"], name="unique_title_per_site"),
+        ]
 
 
 class Group(SchoolTermRelatedExtensibleModel):
@@ -362,6 +412,7 @@ class Group(SchoolTermRelatedExtensibleModel):
             ("view_group_stats", _("Can view statistics about group.")),
         )
         constraints = [
+            # Heads up: Uniqueness per school term already implies uniqueness per site
             models.UniqueConstraint(fields=["school_term", "name"], name="unique_school_term_name"),
             models.UniqueConstraint(
                 fields=["school_term", "short_name"], name="unique_school_term_short_name"
@@ -436,7 +487,7 @@ class Group(SchoolTermRelatedExtensibleModel):
         else:
             return f"{self.name} ({self.short_name})"
 
-    group_info_tracker = FieldTracker(fields=("name", "short_name", "members", "owners"))
+    group_info_tracker = FieldTracker(fields=("name", "short_name"))
 
     def save(self, force: bool = False, *args, **kwargs):
         # Determine state of object in relation to database
@@ -484,8 +535,9 @@ class PersonGroupThrough(ExtensibleModel):
 
 
 @receiver(models.signals.m2m_changed, sender=PersonGroupThrough)
+@receiver(models.signals.m2m_changed, sender=Group.owners.through)
 def save_group_on_m2m_changed(
-    sender: PersonGroupThrough,
+    sender: Union[PersonGroupThrough, Group.owners.through],
     instance: models.Model,
     action: str,
     reverse: bool,
@@ -743,6 +795,10 @@ class DashboardWidget(PolymorphicModel, PureDjangoModel):
           )
     """
 
+    objects = UninstallRenitentPolymorphicManager()
+
+    data_checks = [BrokenDashboardWidgetDataCheck]
+
     @staticmethod
     def get_media(widgets: Union[QuerySet, Iterable]):
         """Return all media required to render the selected widgets."""
@@ -752,10 +808,12 @@ class DashboardWidget(PolymorphicModel, PureDjangoModel):
         return media
 
     template = None
+    template_broken = "core/dashboard_widget/dashboardwidget_broken.html"
     media = Media()
 
     title = models.CharField(max_length=150, verbose_name=_("Widget Title"))
     active = models.BooleanField(verbose_name=_("Activate Widget"))
+    broken = models.BooleanField(verbose_name=_("Widget is broken"), default=False)
 
     size_s = models.PositiveSmallIntegerField(
         verbose_name=_("Size on mobile devices"),
@@ -782,6 +840,11 @@ class DashboardWidget(PolymorphicModel, PureDjangoModel):
         default=4,
     )
 
+    def _get_context_safe(self, request):
+        if self.broken:
+            return {"title": self.title}
+        return self.get_context(request)
+
     def get_context(self, request):
         """Get the context dictionary to pass to the widget template."""
         raise NotImplementedError("A widget subclass needs to implement the get_context method.")
@@ -790,8 +853,13 @@ class DashboardWidget(PolymorphicModel, PureDjangoModel):
         """Get template.
 
         Get the template to render the widget with. Defaults to the template attribute,
-        but can be overridden to allow more complex template generation scenarios.
+        but can be overridden to allow more complex template generation scenarios. If
+        the widget is marked as broken, the template_broken attribute will be returned.
         """
+        if self.broken:
+            return self.template_broken
+        if not self.template:
+            raise NotImplementedError("A widget subclass needs to define a template.")
         return self.template
 
     def __str__(self):
@@ -827,6 +895,8 @@ class DashboardWidgetOrder(ExtensibleModel):
     order = models.PositiveIntegerField(verbose_name=_("Order"))
     default = models.BooleanField(default=False, verbose_name=_("Part of the default dashboard"))
 
+    objects = InstalledWidgetsDashboardWidgetOrderManager()
+
     @classproperty
     def default_dashboard_widgets(cls):
         """Get default order for dashboard widgets."""
@@ -845,7 +915,7 @@ class DashboardWidgetOrder(ExtensibleModel):
 class CustomMenu(ExtensibleModel):
     """A custom menu to display in the footer."""
 
-    name = models.CharField(max_length=100, verbose_name=_("Menu ID"), unique=True)
+    name = models.CharField(max_length=100, verbose_name=_("Menu ID"))
 
     def __str__(self):
         return self.name if self.name != "" else self.id
@@ -860,6 +930,9 @@ class CustomMenu(ExtensibleModel):
     class Meta:
         verbose_name = _("Custom menu")
         verbose_name_plural = _("Custom menus")
+        constraints = [
+            models.UniqueConstraint(fields=["site_id", "name"], name="unique_menu_name_per_site"),
+        ]
 
 
 class CustomMenuItem(ExtensibleModel):
@@ -878,6 +951,10 @@ class CustomMenuItem(ExtensibleModel):
     class Meta:
         verbose_name = _("Custom menu item")
         verbose_name_plural = _("Custom menu items")
+        constraints = [
+            # Heads up: Uniqueness per menu already implies uniqueness per site
+            models.UniqueConstraint(fields=["menu", "name"], name="unique_name_per_menu"),
+        ]
 
 
 class GroupType(ExtensibleModel):
@@ -896,6 +973,11 @@ class GroupType(ExtensibleModel):
     class Meta:
         verbose_name = _("Group type")
         verbose_name_plural = _("Group types")
+        constraints = [
+            models.UniqueConstraint(
+                fields=["site_id", "name"], name="unique_group_type_name_per_site"
+            ),
+        ]
 
 
 class GlobalPermissions(GlobalPermissionModel):
@@ -904,7 +986,6 @@ class GlobalPermissions(GlobalPermissionModel):
     class Meta(GlobalPermissionModel.Meta):
         permissions = (
             ("view_system_status", _("Can view system status")),
-            ("link_persons_accounts", _("Can link persons to accounts")),
             ("manage_data", _("Can manage data")),
             ("impersonate", _("Can impersonate")),
             ("search", _("Can use search")),
@@ -983,41 +1064,92 @@ class PDFFile(ExtensibleModel):
     def _get_default_expiration():  # noqa
         return timezone.now() + timedelta(minutes=get_site_preferences()["general__pdf_expiration"])
 
-    def _get_upload_path(instance, filename):  # noqa
-        return f"pdfs/{instance.secret}.pdf"
-
     person = models.ForeignKey(
-        to=Person, on_delete=models.CASCADE, verbose_name=_("Owner"), related_name="pdf_files"
+        to=Person,
+        on_delete=models.CASCADE,
+        blank=True,
+        null=True,
+        verbose_name=_("Owner"),
+        related_name="pdf_files",
     )
     expires_at = models.DateTimeField(
         verbose_name=_("File expires at"), default=_get_default_expiration
     )
-    html = models.TextField(verbose_name=_("Rendered HTML"))
+    html_file = models.FileField(upload_to="pdfs/", verbose_name=_("Generated HTML file"))
     file = models.FileField(
-        upload_to=_get_upload_path, blank=True, null=True, verbose_name=_("Generated PDF file")
+        upload_to="pdfs/", blank=True, null=True, verbose_name=_("Generated PDF file")
     )
 
     def __str__(self):
         return f"{self.person} ({self.pk})"
 
-    @property
-    def secret(self) -> str:
-        """Get secret needed for accessing the HTML page."""
-        return hmac.new(
-            bytes(settings.SECRET_KEY, "utf-8"),
-            msg=bytes(self.html + str(self.expires_at), "utf-8"),
-            digestmod="sha256",
-        ).hexdigest()
-
-    @property
-    def html_url(self) -> str:
-        """Get URL for the HTML page."""
-        return (
-            urlparse(reverse("html_for_pdf_file", args=[self.pk]))
-            ._replace(query=f"secret={self.secret}")
-            .geturl()
-        )
-
     class Meta:
         verbose_name = _("PDF file")
         verbose_name_plural = _("PDF files")
+
+
+class TaskUserAssignment(ExtensibleModel):
+    task_result = models.ForeignKey(
+        TaskResult, on_delete=models.CASCADE, verbose_name=_("Task result")
+    )
+    user = models.ForeignKey(
+        get_user_model(), on_delete=models.CASCADE, verbose_name=_("Task user")
+    )
+
+    @classmethod
+    def create_for_task_id(cls, task_id: str, user: "User") -> "TaskUserAssignment":
+        # Use get_or_create to ensure the TaskResult exists
+        # django-celery-results will later add the missing information
+        with cachalot_disabled():
+            result, __ = TaskResult.objects.get_or_create(task_id=task_id)
+        return cls.objects.create(task_result=result, user=user)
+
+    class Meta:
+        verbose_name = _("Task user assignment")
+        verbose_name_plural = _("Task user assignments")
+
+
+class OAuthApplication(AbstractApplication):
+    """Modified OAuth application class that supports Grant Flows configured in preferences."""
+
+    # Override grant types to make field optional
+    authorization_grant_type = models.CharField(
+        max_length=32, choices=AbstractApplication.GRANT_TYPES, blank=True, null=True
+    )
+
+    # Optional list of alloewd scopes
+    allowed_scopes = ArrayField(
+        models.CharField(max_length=32),
+        verbose_name=_("Allowed scopes that clients can request"),
+        null=True,
+        blank=True,
+    )
+
+    def allows_grant_type(self, *grant_types: set[str]) -> bool:
+        allowed_grants = get_site_preferences()["auth__oauth_allowed_grants"]
+
+        return bool(set(allowed_grants) & grant_types)
+
+
+class OAuthGrant(AbstractGrant):
+    """Placeholder for customising the Grant model."""
+
+    pass
+
+
+class OAuthAccessToken(AbstractAccessToken):
+    """Placeholder for customising the AccessToken model."""
+
+    pass
+
+
+class OAuthIDToken(AbstractIDToken):
+    """Placeholder for customising the IDToken model."""
+
+    pass
+
+
+class OAuthRefreshToken(AbstractRefreshToken):
+    """Placeholder for customising the RefreshToken model."""
+
+    pass
diff --git a/aleksis/core/preferences.py b/aleksis/core/preferences.py
index 9824b5d379b5594c8931aa76b28d5577a346b8a1..d0de22fe03cc044fa5722982dfcc7faf86a4503c 100644
--- a/aleksis/core/preferences.py
+++ b/aleksis/core/preferences.py
@@ -14,24 +14,27 @@ from dynamic_preferences.types import (
     MultipleChoicePreference,
     StringPreference,
 )
+from oauth2_provider.models import AbstractApplication
 
 from .models import Group, Person
 from .registries import person_preferences_registry, site_preferences_registry
 from .util.notifications import get_notification_choices_lazy
 
-general = Section("general")
-school = Section("school")
-theme = Section("theme")
-mail = Section("mail")
-notification = Section("notification")
-footer = Section("footer")
-account = Section("account")
+general = Section("general", verbose_name=_("General"))
+school = Section("school", verbose_name=_("School"))
+theme = Section("theme", verbose_name=_("Theme"))
+mail = Section("mail", verbose_name=_("Mail"))
+notification = Section("notification", verbose_name=_("Notifications"))
+footer = Section("footer", verbose_name=_("Footer"))
+account = Section("account", verbose_name=_("Accounts"))
 auth = Section("auth", verbose_name=_("Authentication"))
 internationalisation = Section("internationalisation", verbose_name=_("Internationalisation"))
 
 
 @site_preferences_registry.register
 class SiteTitle(StringPreference):
+    """Title of the AlekSIS instance, e.g. schools display name."""
+
     section = general
     name = "title"
     default = "AlekSIS"
@@ -41,6 +44,8 @@ class SiteTitle(StringPreference):
 
 @site_preferences_registry.register
 class SiteDescription(StringPreference):
+    """Site description, e.g. a slogan."""
+
     section = general
     name = "description"
     default = "The Free School Information System"
@@ -50,6 +55,8 @@ class SiteDescription(StringPreference):
 
 @site_preferences_registry.register
 class ColourPrimary(StringPreference):
+    """Primary colour in AlekSIS frontend."""
+
     section = theme
     name = "primary"
     default = "#0d5eaf"
@@ -60,6 +67,8 @@ class ColourPrimary(StringPreference):
 
 @site_preferences_registry.register
 class ColourSecondary(StringPreference):
+    """Secondary colour in AlekSIS frontend."""
+
     section = theme
     name = "secondary"
     default = "#0d5eaf"
@@ -70,6 +79,8 @@ class ColourSecondary(StringPreference):
 
 @site_preferences_registry.register
 class Logo(FilePreference):
+    """Logo of your AlekSIS instance."""
+
     section = theme
     field_class = ImageField
     name = "logo"
@@ -78,6 +89,8 @@ class Logo(FilePreference):
 
 @site_preferences_registry.register
 class Favicon(FilePreference):
+    """Favicon of your AlekSIS instance."""
+
     section = theme
     field_class = ImageField
     name = "favicon"
@@ -86,6 +99,8 @@ class Favicon(FilePreference):
 
 @site_preferences_registry.register
 class PWAIcon(FilePreference):
+    """PWA-Icon of your AlekSIS instance."""
+
     section = theme
     field_class = ImageField
     name = "pwa_icon"
@@ -94,6 +109,8 @@ class PWAIcon(FilePreference):
 
 @site_preferences_registry.register
 class MailOutName(StringPreference):
+    """Mail out name of your AlekSIS instance."""
+
     section = mail
     name = "name"
     default = "AlekSIS"
@@ -103,6 +120,8 @@ class MailOutName(StringPreference):
 
 @site_preferences_registry.register
 class MailOut(StringPreference):
+    """Mail out address of your AlekSIS instance."""
+
     section = mail
     name = "address"
     default = settings.DEFAULT_FROM_EMAIL
@@ -113,6 +132,8 @@ class MailOut(StringPreference):
 
 @site_preferences_registry.register
 class PrivacyURL(StringPreference):
+    """Link to privacy policy of your AlekSIS instance."""
+
     section = footer
     name = "privacy_url"
     default = ""
@@ -123,6 +144,8 @@ class PrivacyURL(StringPreference):
 
 @site_preferences_registry.register
 class ImprintURL(StringPreference):
+    """Link to imprint of your AlekSIS instance."""
+
     section = footer
     name = "imprint_url"
     default = ""
@@ -133,6 +156,8 @@ class ImprintURL(StringPreference):
 
 @person_preferences_registry.register
 class AdressingNameFormat(ChoicePreference):
+    """User preference for adressing name format."""
+
     section = notification
     name = "addressing_name_format"
     default = "first_last"
@@ -146,6 +171,8 @@ class AdressingNameFormat(ChoicePreference):
 
 @person_preferences_registry.register
 class NotificationChannels(ChoicePreference):
+    """User preference for notification channels."""
+
     # FIXME should be a MultipleChoicePreference
     section = notification
     name = "channels"
@@ -157,6 +184,8 @@ class NotificationChannels(ChoicePreference):
 
 @site_preferences_registry.register
 class PrimaryGroupPattern(StringPreference):
+    """Regular expression to match primary group."""
+
     section = account
     name = "primary_group_pattern"
     default = ""
@@ -166,6 +195,8 @@ class PrimaryGroupPattern(StringPreference):
 
 @site_preferences_registry.register
 class PrimaryGroupField(ChoicePreference):
+    """Field on person to match primary group against."""
+
     section = account
     name = "primary_group_field"
     default = "name"
@@ -196,6 +227,8 @@ class AutoLinkPerson(BooleanPreference):
 
 @site_preferences_registry.register
 class SchoolName(StringPreference):
+    """Display name of the school."""
+
     section = school
     name = "name"
     default = ""
@@ -205,6 +238,8 @@ class SchoolName(StringPreference):
 
 @site_preferences_registry.register
 class SchoolNameOfficial(StringPreference):
+    """Official name of the school, e.g. as given by supervisory authority."""
+
     section = school
     name = "name_official"
     default = ""
@@ -212,8 +247,39 @@ class SchoolNameOfficial(StringPreference):
     verbose_name = _("Official name of the school, e.g. as given by supervisory authority")
 
 
+@site_preferences_registry.register
+class AllowPasswordChange(BooleanPreference):
+    section = auth
+    name = "allow_password_change"
+    default = True
+    verbose_name = _("Allow users to change their passwords")
+
+
+@site_preferences_registry.register
+class SignupEnabled(BooleanPreference):
+    section = auth
+    name = "signup_enabled"
+    default = False
+    verbose_name = _("Enable signup")
+
+
+@site_preferences_registry.register
+class OAuthAllowedGrants(MultipleChoicePreference):
+    """Grant Flows allowed for OAuth applications."""
+
+    section = auth
+    name = "oauth_allowed_grants"
+    default = [grant[0] for grant in AbstractApplication.GRANT_TYPES]
+    widget = SelectMultiple
+    verbose_name = _("Allowed Grant Flows for OAuth applications")
+    field_attribute = {"initial": []}
+    choices = AbstractApplication.GRANT_TYPES
+
+
 @site_preferences_registry.register
 class AvailableLanguages(MultipleChoicePreference):
+    """Available languages  of your AlekSIS instance."""
+
     section = internationalisation
     name = "languages"
     default = [code[0] for code in settings.LANGUAGES]
@@ -320,3 +386,23 @@ class PDFFileExpirationDuration(IntegerPreference):
     default = 3
     verbose_name = _("PDF file expiration duration")
     help_text = _("in minutes")
+
+
+@person_preferences_registry.register
+class AutoUpdatingDashboard(BooleanPreference):
+    """User preference for automatically updating the dashboard."""
+
+    section = general
+    name = "automatically_update_dashboard"
+    default = True
+    verbose_name = _("Automatically update the dashboard and its widgets")
+
+
+@site_preferences_registry.register
+class AutoUpdatingDashboardSite(BooleanPreference):
+    """Automatic updating of dashboard."""
+
+    section = general
+    name = "automatically_update_dashboard_site"
+    default = True
+    verbose_name = _("Automatically update the dashboard and its widgets sitewide")
diff --git a/aleksis/core/rules.py b/aleksis/core/rules.py
index 480d2ba8d365dadb7c13c1307c8950996b3289b9..9fc08709ec48e74f0a3801262d29a527764ac323 100644
--- a/aleksis/core/rules.py
+++ b/aleksis/core/rules.py
@@ -3,7 +3,6 @@ from rules import is_superuser
 
 from .models import AdditionalField, Announcement, Group, GroupType, Person
 from .util.predicates import (
-    contains_site_preference_value,
     has_any_object,
     has_global_perm,
     has_object_perm,
@@ -18,32 +17,32 @@ rules.add_perm("core", rules.always_allow)
 
 # View dashboard
 view_dashboard_predicate = is_site_preference_set("general", "anonymous_dashboard") | has_person
-rules.add_perm("core.view_dashboard", view_dashboard_predicate)
+rules.add_perm("core.view_dashboard_rule", view_dashboard_predicate)
 
 # View notifications
-rules.add_perm("core.view_notifications", has_person)
+rules.add_perm("core.view_notifications_rule", has_person)
 
 # Use search
 search_predicate = has_person & has_global_perm("core.search")
-rules.add_perm("core.search", search_predicate)
+rules.add_perm("core.search_rule", search_predicate)
 
 # View persons
 view_persons_predicate = has_person & (
     has_global_perm("core.view_person") | has_any_object("core.view_person", Person)
 )
-rules.add_perm("core.view_persons", view_persons_predicate)
+rules.add_perm("core.view_persons_rule", view_persons_predicate)
 
 # View person
 view_person_predicate = has_person & (
     has_global_perm("core.view_person") | has_object_perm("core.view_person") | is_current_person
 )
-rules.add_perm("core.view_person", view_person_predicate)
+rules.add_perm("core.view_person_rule", view_person_predicate)
 
 # View person address
 view_address_predicate = has_person & (
     has_global_perm("core.view_address") | has_object_perm("core.view_address") | is_current_person
 )
-rules.add_perm("core.view_address", view_address_predicate)
+rules.add_perm("core.view_address_rule", view_address_predicate)
 
 # View person contact details
 view_contact_details_predicate = has_person & (
@@ -51,13 +50,13 @@ view_contact_details_predicate = has_person & (
     | has_object_perm("core.view_contact_details")
     | is_current_person
 )
-rules.add_perm("core.view_contact_details", view_contact_details_predicate)
+rules.add_perm("core.view_contact_details_rule", view_contact_details_predicate)
 
 # View person photo
 view_photo_predicate = has_person & (
     has_global_perm("core.view_photo") | has_object_perm("core.view_photo") | is_current_person
 )
-rules.add_perm("core.view_photo", view_photo_predicate)
+rules.add_perm("core.view_photo_rule", view_photo_predicate)
 
 # View persons groups
 view_groups_predicate = has_person & (
@@ -65,7 +64,7 @@ view_groups_predicate = has_person & (
     | has_object_perm("core.view_person_groups")
     | is_current_person
 )
-rules.add_perm("core.view_person_groups", view_groups_predicate)
+rules.add_perm("core.view_person_groups_rule", view_groups_predicate)
 
 # Edit person
 edit_person_predicate = has_person & (
@@ -73,98 +72,89 @@ edit_person_predicate = has_person & (
     | has_object_perm("core.change_person")
     | is_current_person & is_site_preference_set("account", "editable_fields_person")
 )
-rules.add_perm("core.edit_person", edit_person_predicate)
+rules.add_perm("core.edit_person_rule", edit_person_predicate)
 
 # Delete person
 delete_person_predicate = has_person & (
     has_global_perm("core.delete_person") | has_object_perm("core.delete_person")
 )
-rules.add_perm("core.delete_person", delete_person_predicate)
-
-# Link persons with accounts
-link_persons_accounts_predicate = has_person & has_global_perm("core.link_persons_accounts")
-rules.add_perm("core.link_persons_accounts", link_persons_accounts_predicate)
+rules.add_perm("core.delete_person_rule", delete_person_predicate)
 
 # View groups
 view_groups_predicate = has_person & (
     has_global_perm("core.view_group") | has_any_object("core.view_group", Group)
 )
-rules.add_perm("core.view_groups", view_groups_predicate)
+rules.add_perm("core.view_groups_rule", view_groups_predicate)
 
 # View group
 view_group_predicate = has_person & (
     has_global_perm("core.view_group") | has_object_perm("core.view_group")
 )
-rules.add_perm("core.view_group", view_group_predicate)
+rules.add_perm("core.view_group_rule", view_group_predicate)
 
 # Edit group
 edit_group_predicate = has_person & (
     has_global_perm("core.change_group") | has_object_perm("core.change_group")
 )
-rules.add_perm("core.edit_group", edit_group_predicate)
+rules.add_perm("core.edit_group_rule", edit_group_predicate)
 
 # Delete group
 delete_group_predicate = has_person & (
     has_global_perm("core.delete_group") | has_object_perm("core.delete_group")
 )
-rules.add_perm("core.delete_group", delete_group_predicate)
+rules.add_perm("core.delete_group_rule", delete_group_predicate)
 
 # Assign child groups to groups
 assign_child_groups_to_groups_predicate = has_person & has_global_perm(
     "core.assign_child_groups_to_groups"
 )
-rules.add_perm("core.assign_child_groups_to_groups", assign_child_groups_to_groups_predicate)
+rules.add_perm("core.assign_child_groups_to_groups_rule", assign_child_groups_to_groups_predicate)
 
 # Edit school information
 edit_school_information_predicate = has_person & has_global_perm("core.change_school")
-rules.add_perm("core.edit_school_information", edit_school_information_predicate)
+rules.add_perm("core.edit_school_information_rule", edit_school_information_predicate)
 
 # Manage data
 manage_data_predicate = has_person & has_global_perm("core.manage_data")
-rules.add_perm("core.manage_data", manage_data_predicate)
+rules.add_perm("core.manage_data_rule", manage_data_predicate)
 
 # Mark notification as read
 mark_notification_as_read_predicate = has_person & is_notification_recipient
-rules.add_perm("core.mark_notification_as_read", mark_notification_as_read_predicate)
+rules.add_perm("core.mark_notification_as_read_rule", mark_notification_as_read_predicate)
 
 # View announcements
 view_announcements_predicate = has_person & (
     has_global_perm("core.view_announcement")
     | has_any_object("core.view_announcement", Announcement)
 )
-rules.add_perm("core.view_announcements", view_announcements_predicate)
+rules.add_perm("core.view_announcements_rule", view_announcements_predicate)
 
 # Create or edit announcement
 create_or_edit_announcement_predicate = has_person & (
     has_global_perm("core.add_announcement")
     & (has_global_perm("core.change_announcement") | has_object_perm("core.change_announcement"))
 )
-rules.add_perm("core.create_or_edit_announcement", create_or_edit_announcement_predicate)
+rules.add_perm("core.create_or_edit_announcement_rule", create_or_edit_announcement_predicate)
 
 # Delete announcement
 delete_announcement_predicate = has_person & (
     has_global_perm("core.delete_announcement") | has_object_perm("core.delete_announcement")
 )
-rules.add_perm("core.delete_announcement", delete_announcement_predicate)
+rules.add_perm("core.delete_announcement_rule", delete_announcement_predicate)
 
 # Use impersonate
 impersonate_predicate = has_person & has_global_perm("core.impersonate")
-rules.add_perm("core.impersonate", impersonate_predicate)
+rules.add_perm("core.impersonate_rule", impersonate_predicate)
 
 # View system status
 view_system_status_predicate = has_person & has_global_perm("core.view_system_status")
-rules.add_perm("core.view_system_status", view_system_status_predicate)
+rules.add_perm("core.view_system_status_rule", view_system_status_predicate)
 
 # View people menu (persons + objects)
 rules.add_perm(
-    "core.view_people_menu",
+    "core.view_people_menu_rule",
     has_person
-    & (
-        view_persons_predicate
-        | view_groups_predicate
-        | link_persons_accounts_predicate
-        | assign_child_groups_to_groups_predicate
-    ),
+    & (view_persons_predicate | view_groups_predicate | assign_child_groups_to_groups_predicate),
 )
 
 # View person personal details
@@ -173,14 +163,14 @@ view_personal_details_predicate = has_person & (
     | has_object_perm("core.view_personal_details")
     | is_current_person
 )
-rules.add_perm("core.view_personal_details", view_personal_details_predicate)
+rules.add_perm("core.view_personal_details_rule", view_personal_details_predicate)
 
 # Change site preferences
 change_site_preferences = has_person & (
     has_global_perm("core.change_site_preferences")
     | has_object_perm("core.change_site_preferences")
 )
-rules.add_perm("core.change_site_preferences", change_site_preferences)
+rules.add_perm("core.change_site_preferences_rule", change_site_preferences)
 
 # Change person preferences
 change_person_preferences = has_person & (
@@ -188,7 +178,7 @@ change_person_preferences = has_person & (
     | has_object_perm("core.change_person_preferences")
     | is_current_person
 )
-rules.add_perm("core.change_person_preferences", change_person_preferences)
+rules.add_perm("core.change_person_preferences_rule", change_person_preferences)
 
 # Change group preferences
 change_group_preferences = has_person & (
@@ -196,81 +186,81 @@ change_group_preferences = has_person & (
     | has_object_perm("core.change_group_preferences")
     | is_group_owner
 )
-rules.add_perm("core.change_group_preferences", change_group_preferences)
+rules.add_perm("core.change_group_preferences_rule", change_group_preferences)
 
 
 # Edit additional field
 change_additional_field_predicate = has_person & (
     has_global_perm("core.change_additionalfield") | has_object_perm("core.change_additionalfield")
 )
-rules.add_perm("core.change_additionalfield", change_additional_field_predicate)
+rules.add_perm("core.change_additionalfield_rule", change_additional_field_predicate)
 
 # Edit additional field
 create_additional_field_predicate = has_person & (
     has_global_perm("core.create_additionalfield") | has_object_perm("core.create_additionalfield")
 )
-rules.add_perm("core.create_additionalfield", create_additional_field_predicate)
+rules.add_perm("core.create_additionalfield_rule", create_additional_field_predicate)
 
 
 # Delete additional field
 delete_additional_field_predicate = has_person & (
     has_global_perm("core.delete_additionalfield") | has_object_perm("core.delete_additionalfield")
 )
-rules.add_perm("core.delete_additionalfield", delete_additional_field_predicate)
+rules.add_perm("core.delete_additionalfield_rule", delete_additional_field_predicate)
 
 # View additional fields
-view_additional_field_predicate = has_person & (
+view_additional_fields_predicate = has_person & (
     has_global_perm("core.view_additionalfield")
     | has_any_object("core.view_additionalfield", AdditionalField)
 )
-rules.add_perm("core.view_additionalfield", view_additional_field_predicate)
+rules.add_perm("core.view_additionalfields_rule", view_additional_fields_predicate)
 
 # Edit group type
 change_group_type_predicate = has_person & (
     has_global_perm("core.change_grouptype") | has_object_perm("core.change_grouptype")
 )
-rules.add_perm("core.edit_grouptype", change_group_type_predicate)
+rules.add_perm("core.edit_grouptype_rule", change_group_type_predicate)
 
 # Create group type
 create_group_type_predicate = has_person & (
     has_global_perm("core.create_grouptype") | has_object_perm("core.change_grouptype")
 )
-rules.add_perm("core.create_grouptype", create_group_type_predicate)
+rules.add_perm("core.create_grouptype_rule", create_group_type_predicate)
 
 
 # Delete group type
 delete_group_type_predicate = has_person & (
     has_global_perm("core.delete_grouptype") | has_object_perm("core.delete_grouptype")
 )
-rules.add_perm("core.delete_grouptype", delete_group_type_predicate)
+rules.add_perm("core.delete_grouptype_rule", delete_group_type_predicate)
 
 # View group types
-view_group_type_predicate = has_person & (
+view_group_types_predicate = has_person & (
     has_global_perm("core.view_grouptype") | has_any_object("core.view_grouptype", GroupType)
 )
-rules.add_perm("core.view_grouptype", view_group_type_predicate)
+rules.add_perm("core.view_grouptypes_rule", view_group_types_predicate)
 
 # Create person
 create_person_predicate = has_person & (
     has_global_perm("core.create_person") | has_object_perm("core.create_person")
 )
-rules.add_perm("core.create_person", create_person_predicate)
+rules.add_perm("core.create_person_rule", create_person_predicate)
 
 # Create group
 create_group_predicate = has_person & (
     has_global_perm("core.create_group") | has_object_perm("core.create_group")
 )
-rules.add_perm("core.create_group", create_group_predicate)
+rules.add_perm("core.create_group_rule", create_group_predicate)
 
 # School years
 view_school_term_predicate = has_person & has_global_perm("core.view_schoolterm")
-rules.add_perm("core.view_schoolterm", view_school_term_predicate)
+rules.add_perm("core.view_schoolterm_rule", view_school_term_predicate)
 
 create_school_term_predicate = has_person & has_global_perm("core.add_schoolterm")
-rules.add_perm("core.create_schoolterm", create_school_term_predicate)
+rules.add_perm("core.create_schoolterm_rule", create_school_term_predicate)
 
 edit_school_term_predicate = has_person & has_global_perm("core.change_schoolterm")
-rules.add_perm("core.edit_schoolterm", edit_school_term_predicate)
+rules.add_perm("core.edit_schoolterm_rule", edit_school_term_predicate)
 
 # View admin menu
 view_admin_menu_predicate = has_person & (
@@ -280,66 +270,77 @@ view_admin_menu_predicate = has_person & (
     | view_system_status_predicate
     | view_announcements_predicate
 )
-rules.add_perm("core.view_admin_menu", view_admin_menu_predicate)
+rules.add_perm("core.view_admin_menu_rule", view_admin_menu_predicate)
 
 # View group stats
 view_group_stats_predicate = has_person & (
     has_global_perm("core.view_group_stats") | has_object_perm("core.view_group_stats")
 )
-rules.add_perm("core.view_group_stats", view_group_stats_predicate)
+rules.add_perm("core.view_group_stats_rule", view_group_stats_predicate)
 
 # View data check results
 view_data_check_results_predicate = has_person & has_global_perm("core.view_datacheckresult")
-rules.add_perm("core.view_datacheckresults", view_data_check_results_predicate)
+rules.add_perm("core.view_datacheckresults_rule", view_data_check_results_predicate)
 
 # Run data checks
 run_data_checks_predicate = (
     has_person & view_data_check_results_predicate & has_global_perm("core.run_data_checks")
 )
-rules.add_perm("core.run_data_checks", run_data_checks_predicate)
+rules.add_perm("core.run_data_checks_rule", run_data_checks_predicate)
 
 # Solve data problems
 solve_data_problem_predicate = (
     has_person & view_data_check_results_predicate & has_global_perm("core.solve_data_problem")
 )
-rules.add_perm("core.solve_data_problem", solve_data_problem_predicate)
+rules.add_perm("core.solve_data_problem_rule", solve_data_problem_predicate)
 
 view_dashboard_widget_predicate = has_person & has_global_perm("core.view_dashboardwidget")
-rules.add_perm("core.view_dashboardwidget", view_dashboard_widget_predicate)
+rules.add_perm("core.view_dashboardwidget_rule", view_dashboard_widget_predicate)
 
 create_dashboard_widget_predicate = has_person & has_global_perm("core.add_dashboardwidget")
-rules.add_perm("core.create_dashboardwidget", create_dashboard_widget_predicate)
+rules.add_perm("core.create_dashboardwidget_rule", create_dashboard_widget_predicate)
 
 edit_dashboard_widget_predicate = has_person & has_global_perm("core.change_dashboardwidget")
-rules.add_perm("core.edit_dashboardwidget", edit_dashboard_widget_predicate)
+rules.add_perm("core.edit_dashboardwidget_rule", edit_dashboard_widget_predicate)
 
 delete_dashboard_widget_predicate = has_person & has_global_perm("core.delete_dashboardwidget")
-rules.add_perm("core.delete_dashboardwidget", delete_dashboard_widget_predicate)
+rules.add_perm("core.delete_dashboardwidget_rule", delete_dashboard_widget_predicate)
 
 edit_dashboard_predicate = is_site_preference_set("general", "dashboard_editing") & has_person
-rules.add_perm("core.edit_dashboard", edit_dashboard_predicate)
+rules.add_perm("core.edit_dashboard_rule", edit_dashboard_predicate)
 
 edit_default_dashboard_predicate = has_person & has_global_perm("core.edit_default_dashboard")
-rules.add_perm("core.edit_default_dashboard", edit_default_dashboard_predicate)
+rules.add_perm("core.edit_default_dashboard_rule", edit_default_dashboard_predicate)
+
+# django-allauth
+can_register_predicate = is_site_preference_set(section="auth", pref="signup_enabled")
+rules.add_perm("core.can_register", can_register_predicate)
+
+can_change_password_predicate = is_site_preference_set(section="auth", pref="allow_password_change")
+rules.add_perm("core.can_change_password", can_change_password_predicate)
+
+# OAuth2 permissions
+create_oauthapplication_predicate = has_person & has_global_perm("core.add_oauthapplication")
+rules.add_perm("core.create_oauthapplication_rule", create_oauthapplication_predicate)
+
+view_oauth_applications_predicate = has_person & has_global_perm("core.view_oauthapplication")
+rules.add_perm("core.view_oauthapplications_rule", view_oauth_applications_predicate)
+
+view_oauth_application_predicate = has_person & has_global_perm("core.view_oauthapplication")
+rules.add_perm("core.view_oauthapplication_rule", view_oauth_application_predicate)
+
+edit_oauth_application_predicate = has_person & has_global_perm("core.change_oauthapplication")
+rules.add_perm("core.edit_oauthapplication_rule", edit_oauth_application_predicate)
+
+delete_oauth_applications_predicate = has_person & has_global_perm("core.delete_oauth_applications")
+rules.add_perm("core.delete_oauth_applications_rule", delete_oauth_applications_predicate)
 
 # Upload and browse files via CKEditor
 upload_files_ckeditor_predicate = has_person & has_global_perm("core.upload_files_ckeditor")
-rules.add_perm("core.upload_files_ckeditor", upload_files_ckeditor_predicate)
+rules.add_perm("core.upload_files_ckeditor_rule", upload_files_ckeditor_predicate)
 
 manage_person_permissions_predicate = has_person & is_superuser
 rules.add_perm("core.manage_permissions", manage_person_permissions_predicate)
 
 test_pdf_generation_predicate = has_person & has_global_perm("core.test_pdf")
-rules.add_perm("core.test_pdf", test_pdf_generation_predicate)
-
-# Generate rules for syncable fields
-for field in Person._meta.fields:
-    perm = (
-        has_global_perm("core.edit_person")
-        | has_object_perm("core.edit_person")
-        | (
-            is_current_person
-            & contains_site_preference_value("account", "editable_fields_person", field.name)
-        )
-    )
-    rules.add_perm(f"core.change_person_field_{field.name}", perm)
+rules.add_perm("core.test_pdf_rule", test_pdf_generation_predicate)
diff --git a/aleksis/core/settings.py b/aleksis/core/settings.py
index 02d74f5a2f01f0a0c65e5d700991c642e875eca0..dd11bc32912daa37f7e8e1421fbe4f194acc69b1 100644
--- a/aleksis/core/settings.py
+++ b/aleksis/core/settings.py
@@ -1,16 +1,14 @@
 import os
 from glob import glob
+from socket import getfqdn
 
 from django.utils.translation import gettext_lazy as _
 
 from dynaconf import LazySettings
 
-from .util.core_helpers import (
-    get_app_packages,
-    lazy_get_favicon_url,
-    lazy_preference,
-    merge_app_settings,
-)
+from .util.core_helpers import get_app_packages, merge_app_settings, monkey_patch
+
+monkey_patch()
 
 IN_PYTEST = "PYTEST_CURRENT_TEST" in os.environ or "TOX_ENV_DIR" in os.environ
 
@@ -19,6 +17,7 @@ DIRS_FOR_DYNACONF = ["/etc/aleksis"]
 
 SETTINGS_FILE_FOR_DYNACONF = []
 for directory in DIRS_FOR_DYNACONF:
+    SETTINGS_FILE_FOR_DYNACONF += glob(os.path.join(directory, "*.json"))
     SETTINGS_FILE_FOR_DYNACONF += glob(os.path.join(directory, "*.ini"))
     SETTINGS_FILE_FOR_DYNACONF += glob(os.path.join(directory, "*.yaml"))
     SETTINGS_FILE_FOR_DYNACONF += glob(os.path.join(directory, "*.toml"))
@@ -67,9 +66,12 @@ UWSGI = {
     "module": "aleksis.core.wsgi",
 }
 UWSGI_SERVE_STATIC = True
-UWSGI_SERVE_MEDIA = True
+UWSGI_SERVE_MEDIA = False
 
-ALLOWED_HOSTS = _settings.get("http.allowed_hosts", [])
+ALLOWED_HOSTS = _settings.get("http.allowed_hosts", [getfqdn(), "localhost", "127.0.0.1", "[::1]"])
+BASE_URL = _settings.get(
+    "http.base_url", "http://localhost:8000" if DEBUG else f"https://{ALLOWED_HOSTS[0]}"
+)
 
 # Application definition
 INSTALLED_APPS = [
@@ -87,7 +89,6 @@ INSTALLED_APPS = [
     "rules.apps.AutodiscoverRulesConfig",
     "haystack",
     "polymorphic",
-    "django_global_request",
     "dbbackup",
     "django_celery_beat",
     "django_celery_results",
@@ -95,7 +96,6 @@ INSTALLED_APPS = [
     "health_check.contrib.celery",
     "djcelery_email",
     "celery_haystack",
-    "settings_context_processor",
     "sass_processor",
     "django_any_js",
     "django_yarnpkg",
@@ -107,7 +107,6 @@ INSTALLED_APPS = [
     "debug_toolbar",
     "django_prometheus",
     "django_select2",
-    "hattori",
     "templated_email",
     "html2text",
     "django_otp.plugins.otp_totp",
@@ -115,6 +114,9 @@ INSTALLED_APPS = [
     "django_otp",
     "otp_yubikey",
     "aleksis.core",
+    "allauth",
+    "allauth.account",
+    "allauth.socialaccount",
     "health_check",
     "health_check.db",
     "health_check.cache",
@@ -126,7 +128,6 @@ INSTALLED_APPS = [
     "impersonate",
     "two_factor",
     "material",
-    "pwa",
     "ckeditor",
     "ckeditor_uploader",
     "django_js_reverse",
@@ -134,6 +135,8 @@ INSTALLED_APPS = [
     "django_bleach",
     "favicon",
     "django_filters",
+    "oauth2_provider",
+    "rest_framework",
 ]
 
 merge_app_settings("INSTALLED_APPS", INSTALLED_APPS, True)
@@ -155,7 +158,6 @@ MIDDLEWARE = [
     "debug_toolbar.middleware.DebugToolbarMiddleware",
     "django.middleware.locale.LocaleMiddleware",
     "django.middleware.http.ConditionalGetMiddleware",
-    "django_global_request.middleware.GlobalRequestMiddleware",
     "django.contrib.sites.middleware.CurrentSiteMiddleware",
     "django.middleware.common.CommonMiddleware",
     "django.middleware.csrf.CsrfViewMiddleware",
@@ -183,7 +185,6 @@ TEMPLATES = [
                 "django.contrib.auth.context_processors.auth",
                 "django.contrib.messages.context_processors.messages",
                 "maintenance_mode.context_processors.maintenance_mode",
-                "settings_context_processor.context_processors.settings",
                 "dynamic_preferences.processors.global_preferences",
                 "aleksis.core.util.core_helpers.custom_information_processor",
             ],
@@ -231,8 +232,6 @@ if _settings.get("caching.redis.enabled", not IN_PYTEST):
     }
     if REDIS_PASSWORD:
         CACHES["default"]["OPTIONS"]["PASSWORD"] = REDIS_PASSWORD
-    DJANGO_REDIS_IGNORE_EXCEPTIONS = True
-    DJANGO_REDIS_LOG_IGNORED_EXCEPTIONS = True
 else:
     CACHES = {
         "default": {
@@ -245,7 +244,7 @@ INSTALLED_APPS.append("cachalot")
 DEBUG_TOOLBAR_PANELS.append("cachalot.panels.CachalotPanel")
 CACHALOT_TIMEOUT = _settings.get("caching.cachalot.timeout", None)
 CACHALOT_DATABASES = set(["default"])
-SILENCED_SYSTEM_CHECKS += ["cachalot.W001", "cachalot.E003"]
+SILENCED_SYSTEM_CHECKS += ["cachalot.W001"]
 
 SESSION_ENGINE = "django.contrib.sessions.backends.cache"
 SESSION_CACHE_ALIAS = "default"
@@ -269,6 +268,89 @@ AUTH_INITIAL_SUPERUSER = {
 # Authentication backends are dynamically populated
 AUTHENTICATION_BACKENDS = []
 
+# Configuration for django-allauth.
+
+# Use custom adapter to override some behaviour, i.e. honour the LDAP backend
+SOCIALACCOUNT_ADAPTER = "aleksis.core.util.auth_helpers.OurSocialAccountAdapter"
+
+# Get django-allauth providers from config
+_SOCIALACCOUNT_PROVIDERS = _settings.get("auth.providers", None)
+if _SOCIALACCOUNT_PROVIDERS:
+    SOCIALACCOUNT_PROVIDERS = _SOCIALACCOUNT_PROVIDERS.to_dict()
+
+    # Add configured social auth providers to INSTALLED_APPS
+    for provider, config in SOCIALACCOUNT_PROVIDERS.items():
+        INSTALLED_APPS.append(f"allauth.socialaccount.providers.{provider}")
+        SOCIALACCOUNT_PROVIDERS[provider] = {k.upper(): v for k, v in config.items()}
+
+
+# Configure custom forms
+
+ACCOUNT_FORMS = {
+    "signup": "aleksis.core.forms.AccountRegisterForm",
+}
+
+# Use custom adapter
+ACCOUNT_ADAPTER = "aleksis.core.util.auth_helpers.OurAccountAdapter"
+
+# Require password confirmation
+SIGNUP_PASSWORD_ENTER_TWICE = True
+
+# Allow login by either username or email
+ACCOUNT_AUTHENTICATION_METHOD = _settings.get("auth.registration.method", "username_email")
+
+# Require email address to sign up
+ACCOUNT_EMAIL_REQUIRED = _settings.get("auth.registration.email_required", True)
+SOCIALACCOUNT_EMAIL_REQUIRED = False
+
+# Require email verification after sigm up
+ACCOUNT_EMAIL_VERIFICATION = _settings.get("auth.registration.email_verification", "mandatory")
+SOCIALACCOUNT_EMAIL_VERIFICATION = False
+
+# Email subject prefix for verification mails
+ACCOUNT_EMAIL_SUBJECT_PREFIX = _settings.get("auth.registration.subject", "[AlekSIS] ")
+
+# Max attempts before login timeout
+ACCOUNT_LOGIN_ATTEMPTS_LIMIT = _settings.get("auth.login.login_limit", 5)
+
+# Login timeout after max attempts in seconds
+ACCOUNT_LOGIN_ATTEMPTS_TIMEOUT = _settings.get("auth.login.login_timeout", 300)
+
+# Email confirmation field in form
+ACCOUNT_SIGNUP_EMAIL_ENTER_TWICE = True
+
+# Enforce uniqueness of email addresses
+ACCOUNT_UNIQUE_EMAIL = _settings.get("auth.login.registration.unique_email", True)
+
+# Configuration for OAuth2 provider
+OAUTH2_PROVIDER = {"SCOPES_BACKEND_CLASS": "aleksis.core.util.auth_helpers.AppScopes"}
+OAUTH2_PROVIDER_APPLICATION_MODEL = "core.OAuthApplication"
+OAUTH2_PROVIDER_GRANT_MODEL = "core.OAuthGrant"
+OAUTH2_PROVIDER_ACCESS_TOKEN_MODEL = "core.OAuthAccessToken"  # noqa: S105
+OAUTH2_PROVIDER_ID_TOKEN_MODEL = "core.OAuthIDToken"  # noqa: S105
+OAUTH2_PROVIDER_REFRESH_TOKEN_MODEL = "core.OAuthRefreshToken"  # noqa: S105
+
+if _settings.get("oauth2.oidc.enabled", False):
+    with open(_settings.get("oauth2.oidc.rsa_key", "/etc/aleksis/oidc.pem"), "r") as f:
+        oid_rsa_key = f.read()
+
+    OAUTH2_PROVIDER.update(
+        {
+            "OAUTH2_VALIDATOR_CLASS": "aleksis.core.util.auth_helpers.CustomOAuth2Validator",
+            "OIDC_ENABLED": True,
+            "OIDC_RSA_PRIVATE_KEY": oid_rsa_key,
+            #        "OIDC_ISS_ENDPOINT": _settings.get("oauth2.oidc.issuer_name", "example.com"),
+        }
+    )
+
+# Configuration for REST framework
+REST_FRAMEWORK = {
+    "DEFAULT_AUTHENTICATION_CLASSES": [
+        "oauth2_provider.contrib.rest_framework.OAuth2Authentication",
+    ]
+}
+
+# LDAP config
 if _settings.get("ldap.uri", None):
     # LDAP dependencies are not necessarily installed, so import them here
     import ldap  # noqa
@@ -291,7 +373,7 @@ if _settings.get("ldap.uri", None):
         AUTH_LDAP_BIND_PASSWORD = _settings.get("ldap.bind.password")
 
     # Keep local password for users to be required to proveide their old password on change
-    AUTH_LDAP_SET_USABLE_PASSWORD = True
+    AUTH_LDAP_SET_USABLE_PASSWORD = _settings.get("ldap.handle_passwords", True)
 
     # Keep bound as the authenticating user
     # Ensures proper read permissions, and ability to change password without admin
@@ -360,16 +442,12 @@ if _settings.get("ldap.uri", None):
                 "is_superuser"
             ]
 
-CUSTOM_AUTHENTICATION_BACKENDS = []
-merge_app_settings("AUTHENTICATION_BACKENDS", CUSTOM_AUTHENTICATION_BACKENDS)
-
 # Add ModelBckend last so all other backends get a chance
 # to verify passwords first
 AUTHENTICATION_BACKENDS.append("django.contrib.auth.backends.ModelBackend")
 
-# Structure of items: backend, URL name, icon name, button title
-ALTERNATIVE_LOGIN_VIEWS = []
-merge_app_settings("ALTERNATIVE_LOGIN_VIEWS", ALTERNATIVE_LOGIN_VIEWS, True)
+# Authentication backend for django-allauth.
+AUTHENTICATION_BACKENDS.append("allauth.account.auth_backends.AuthenticationBackend")
 
 # Internationalization
 # https://docs.djangoproject.com/en/2.1/topics/i18n/
@@ -377,8 +455,6 @@ merge_app_settings("ALTERNATIVE_LOGIN_VIEWS", ALTERNATIVE_LOGIN_VIEWS, True)
 LANGUAGES = [
     ("en", _("English")),
     ("de", _("German")),
-    ("fr", _("French")),
-    ("nb", _("Norwegian (bokmål)")),
 ]
 LANGUAGE_CODE = _settings.get("l10n.lang", "en")
 TIME_ZONE = _settings.get("l10n.tz", "UTC")
@@ -402,7 +478,6 @@ NODE_MODULES_ROOT = _settings.get("node_modules.root", os.path.join(BASE_DIR, "n
 
 YARN_INSTALLED_APPS = [
     "@fontsource/roboto",
-    "datatables",
     "jquery",
     "materialize-css",
     "material-design-icons-iconfont",
@@ -411,6 +486,7 @@ YARN_INSTALLED_APPS = [
     "paper-css",
     "jquery-sortablejs",
     "sortablejs",
+    "@sentry/tracing",
 ]
 
 merge_app_settings("YARN_INSTALLED_APPS", YARN_INSTALLED_APPS, True)
@@ -423,7 +499,6 @@ SELECT2_JS = JS_URL + "/select2/dist/js/select2.min.js"
 SELECT2_I18N_PATH = JS_URL + "/select2/dist/js/i18n"
 
 ANY_JS = {
-    "DataTables": {"js_url": JS_URL + "/datatables/media/js/jquery.dataTables.min.js"},
     "materialize": {"js_url": JS_URL + "/materialize-css/dist/js/materialize.min.js"},
     "jQuery": {"js_url": JS_URL + "/jquery/dist/jquery.min.js"},
     "material-design-icons": {
@@ -436,7 +511,13 @@ ANY_JS = {
     },
     "sortablejs": {"js_url": JS_URL + "/sortablejs/Sortable.min.js"},
     "jquery-sortablejs": {"js_url": JS_URL + "/jquery-sortablejs/jquery-sortable.js"},
-    "Roboto": {"css_url": JS_URL + "/@fontsource/roboto/index.css"},
+    "Roboto100": {"css_url": JS_URL + "/@fontsource/roboto/100.css"},
+    "Roboto300": {"css_url": JS_URL + "/@fontsource/roboto/300.css"},
+    "Roboto400": {"css_url": JS_URL + "/@fontsource/roboto/400.css"},
+    "Roboto500": {"css_url": JS_URL + "/@fontsource/roboto/500.css"},
+    "Roboto700": {"css_url": JS_URL + "/@fontsource/roboto/700.css"},
+    "Roboto900": {"css_url": JS_URL + "/@fontsource/roboto/900.css"},
+    "Sentry": {"js_url": JS_URL + "/@sentry/tracing/build/bundle.tracing.js"},
 }
 
 merge_app_settings("ANY_JS", ANY_JS, True)
@@ -448,15 +529,16 @@ SASS_PROCESSOR_CUSTOM_FUNCTIONS = {
     "get-preference": "aleksis.core.util.sass_helpers.get_preference",
 }
 SASS_PROCESSOR_INCLUDE_DIRS = [
-    _settings.get("materialize.sass_path", JS_ROOT + "/materialize-css/sass/"),
-    STATIC_ROOT + "/materialize-css/sass/",
-    STATIC_ROOT,
+    _settings.get("materialize.sass_path", os.path.join(JS_ROOT, "materialize-css", "sass")),
+    os.path.join(STATIC_ROOT, "public"),
 ]
 
-ADMINS = _settings.get("contact.admins", [])
-SERVER_EMAIL = _settings.get("contact.from", "root@localhost")
-DEFAULT_FROM_EMAIL = _settings.get("contact.from", "root@localhost")
-MANAGERS = _settings.get("contact.admins", [])
+ADMINS = _settings.get(
+    "contact.admins", [(AUTH_INITIAL_SUPERUSER["username"], AUTH_INITIAL_SUPERUSER["email"])]
+)
+SERVER_EMAIL = _settings.get("contact.from", ADMINS[0][1])
+DEFAULT_FROM_EMAIL = _settings.get("contact.from", ADMINS[0][1])
+MANAGERS = _settings.get("contact.admins", ADMINS)
 
 if _settings.get("mail.server.host", None):
     EMAIL_HOST = _settings.get("mail.server.host")
@@ -471,9 +553,6 @@ if _settings.get("mail.server.host", None):
 TEMPLATED_EMAIL_BACKEND = "templated_email.backends.vanilla_django"
 TEMPLATED_EMAIL_AUTO_PLAIN = True
 
-
-TEMPLATE_VISIBLE_SETTINGS = ["ADMINS", "DEBUG"]
-
 DYNAMIC_PREFERENCES = {
     "REGISTRY_MODULE": "preferences",
 }
@@ -554,65 +633,19 @@ if _settings.get("dev.uwsgi.celery", DEBUG):
     UWSGI["attach-daemon"].append(f"celery -A aleksis.core worker --concurrency={concurrency}")
     UWSGI["attach-daemon"].append("celery -A aleksis.core beat")
 
-PWA_APP_NAME = lazy_preference("general", "title")
-PWA_APP_DESCRIPTION = lazy_preference("general", "description")
-PWA_APP_THEME_COLOR = lazy_preference("theme", "primary")
-PWA_APP_BACKGROUND_COLOR = "#ffffff"
-PWA_APP_DISPLAY = "standalone"
-PWA_APP_ORIENTATION = "any"
-PWA_APP_ICONS = [
-    {
-        "src": lazy_get_favicon_url(
-            title="pwa_icon", size=192, rel="android", default=STATIC_URL + "icons/android_192.png"
-        ),
-        "sizes": "192x192",
-    },
-    {
-        "src": lazy_get_favicon_url(
-            title="pwa_icon", size=512, rel="android", default=STATIC_URL + "icons/android_512.png"
-        ),
-        "sizes": "512x512",
-    },
-]
-PWA_APP_ICONS_APPLE = [
-    {
-        "src": lazy_get_favicon_url(
-            title="pwa_icon", size=192, rel="apple", default=STATIC_URL + "icons/apple_76.png"
-        ),
-        "sizes": "76x76",
-    },
-    {
-        "src": lazy_get_favicon_url(
-            title="pwa_icon", size=192, rel="apple", default=STATIC_URL + "icons/apple_114.png"
-        ),
-        "sizes": "114x114",
-    },
-    {
-        "src": lazy_get_favicon_url(
-            title="pwa_icon", size=192, rel="apple", default=STATIC_URL + "icons/apple_152.png"
-        ),
-        "sizes": "152x152",
-    },
-    {
-        "src": lazy_get_favicon_url(
-            title="pwa_icon", size=192, rel="apple", default=STATIC_URL + "icons/apple_180.png"
-        ),
-        "sizes": "180x180",
-    },
-]
-PWA_APP_SPLASH_SCREEN = [
-    {
-        "src": lazy_get_favicon_url(
-            title="pwa_icon", size=192, rel="apple", default=STATIC_URL + "icons/apple_180.png"
-        ),
-        "media": (
-            "(device-width: 320px) and (device-height: 568px) and" "(-webkit-device-pixel-ratio: 2)"
-        ),
-    }
-]
-
+DEFAULT_FAVICON_PATHS = {
+    "pwa_icon": os.path.join(STATIC_ROOT, "img/aleksis-icon.png"),
+    "favicon": os.path.join(STATIC_ROOT, "img/aleksis-icon.png"),
+}
+PWA_ICONS_CONFIG = {
+    "android": [192, 512],
+    "apple": [76, 114, 152, 180],
+    "apple_splash": [192],
+    "microsoft": [144],
+}
+FAVICON_PATH = os.path.join("public", "favicon")
 
-PWA_SERVICE_WORKER_PATH = os.path.join(STATIC_ROOT, "js", "serviceworker.js")
+SERVICE_WORKER_PATH = os.path.join(STATIC_ROOT, "js", "serviceworker.js")
 
 SITE_ID = 1
 
@@ -739,11 +772,21 @@ BLEACH_STRIP_COMMENTS = True
 LOGGING = {
     "version": 1,
     "disable_existing_loggers": False,
-    "handlers": {"console": {"class": "logging.StreamHandler", "formatter": "verbose"},},
+    "handlers": {
+        "console": {"class": "logging.StreamHandler", "formatter": "verbose"},
+        "null": {"class": "logging.NullHandler"},
+    },
     "formatters": {"verbose": {"format": "%(levelname)s %(asctime)s %(module)s: %(message)s"}},
     "root": {"handlers": ["console"], "level": _settings.get("logging.level", "WARNING"),},
+    "loggers": {},
 }
 
+if not _settings.get("logging.disallowed_host", False):
+    LOGGING["loggers"]["django.security.DisallowedHost"] = {
+        "handlers": ["null"],
+        "propagate": False,
+    }
+
 # Rules and permissions
 
 GUARDIAN_RAISE_403 = True
@@ -754,26 +797,9 @@ SILENCED_SYSTEM_CHECKS.append("guardian.W001")
 # Append authentication backends
 AUTHENTICATION_BACKENDS.append("rules.permissions.ObjectPermissionBackend")
 
-HAYSTACK_BACKEND_SHORT = _settings.get("search.backend", "simple")
-
-if HAYSTACK_BACKEND_SHORT == "simple":
-    HAYSTACK_CONNECTIONS = {
-        "default": {"ENGINE": "haystack.backends.simple_backend.SimpleEngine",},
-    }
-elif HAYSTACK_BACKEND_SHORT == "xapian":
-    HAYSTACK_CONNECTIONS = {
-        "default": {
-            "ENGINE": "xapian_backend.XapianEngine",
-            "PATH": _settings.get("search.index", os.path.join(BASE_DIR, "xapian_index")),
-        },
-    }
-elif HAYSTACK_BACKEND_SHORT == "whoosh":
-    HAYSTACK_CONNECTIONS = {
-        "default": {
-            "ENGINE": "haystack.backends.whoosh_backend.WhooshEngine",
-            "PATH": _settings.get("search.index", os.path.join(BASE_DIR, "whoosh_index")),
-        },
-    }
+HAYSTACK_CONNECTIONS = {
+    "default": {"ENGINE": "haystack_redis.RedisEngine", "PATH": REDIS_URL,},
+}
 
 HAYSTACK_SIGNAL_PROCESSOR = "celery_haystack.signals.CelerySignalProcessor"
 CELERY_HAYSTACK_IGNORE_RESULT = True
@@ -791,6 +817,10 @@ DBBACKUP_CHECK_SECONDS = _settings.get("backup.database.check_seconds", 7200)
 MEDIABACKUP_CHECK_SECONDS = _settings.get("backup.media.check_seconds", 7200)
 
 PROMETHEUS_EXPORT_MIGRATIONS = False
+PROMETHEUS_METRICS_EXPORT_PORT = _settings.get("prometheus.metrics.port", None)
+PROMETHEUS_METRICS_EXPORT_ADDRESS = _settings.get("prometheus.metrucs.address", None)
+
+SECURE_PROXY_SSL_HEADER = ("REQUEST_SCHEME", "https")
 
 if _settings.get("storage.type", "").lower() == "s3":
     INSTALLED_APPS.append("storages")
@@ -825,10 +855,39 @@ if _settings.get("storage.type", "").lower() == "s3":
     AWS_S3_GZIP = _settings.get("storage.s3.gzip", True)
     AWS_S3_SIGNATURE_VERSION = _settings.get("storage.s3.signature_version", None)
     AWS_S3_FILE_OVERWRITE = _settings.get("storage.s3.file_overwrite", False)
+    AWS_S3_VERIFY = _settings.get("storage.s3.verify", True)
+    AWS_S3_USE_SSL = _settings.get("storage.s3.use_ssl", True)
 else:
-    DEFAULT_FILE_STORAGE = "django.core.files.storage.FileSystemStorage"
+    DEFAULT_FILE_STORAGE = "titofisto.TitofistoStorage"
+    TITOFISTO_TIMEOUT = 10 * 60
 
 SASS_PROCESSOR_STORAGE = DEFAULT_FILE_STORAGE
 
+SENTRY_ENABLED = _settings.get("health.sentry.enabled", False)
+if SENTRY_ENABLED:
+    import sentry_sdk
+    from sentry_sdk.integrations.celery import CeleryIntegration
+    from sentry_sdk.integrations.django import DjangoIntegration
+    from sentry_sdk.integrations.redis import RedisIntegration
+
+    from aleksis.core import __version__
+
+    SENTRY_SETTINGS = {
+        "dsn": _settings.get("health.sentry.dsn"),
+        "environment": _settings.get("health.sentry.environment"),
+        "traces_sample_rate": _settings.get("health.sentry.traces_sample_rate", 1.0),
+        "send_default_pii": _settings.get("health.sentry.send_default_pii", False),
+        "release": f"aleksis-core@{__version__}",
+        "in_app_include": "aleksis",
+    }
+    sentry_sdk.init(
+        integrations=[
+            DjangoIntegration(transaction_style="function_name"),
+            RedisIntegration(),
+            CeleryIntegration(),
+        ],
+        **SENTRY_SETTINGS,
+    )
+
 # Add django-cleanup after all apps to ensure that it gets all signals as last app
 INSTALLED_APPS.append("django_cleanup.apps.CleanupConfig")
diff --git a/aleksis/core/static/js/main.js b/aleksis/core/static/js/main.js
index df44aaf966b1f446a5e7463a7dd207dc4109e861..1d2a2fe28e3c4304e72d8abfc398f0fda3908586 100644
--- a/aleksis/core/static/js/main.js
+++ b/aleksis/core/static/js/main.js
@@ -91,12 +91,6 @@ $(document).ready(function () {
         $("#" + $(this).data("preview")).css("color", $(this).val());
     });
 
-    $('table.datatable').each(function (index) {
-        $(this).DataTable({
-            "paging": false
-        });
-    });
-
     // Initialise auto-completion for search bar
     window.autocomplete = new Autocomplete({minimum_length: 2});
     window.autocomplete.setup();
@@ -112,6 +106,18 @@ $(document).ready(function () {
         var el = $(e.target).parent();
         el.addClass("closed").removeClass("opened");
     });
+
+    // Initialize the service worker
+    if ('serviceWorker' in navigator) {
+        console.debug("Start registration of service worker.");
+        navigator.serviceWorker.register('/serviceworker.js', {
+            scope: '/'
+        }).then(function() {
+            console.debug("Service worker has been registered.");
+        }).catch(function() {
+            console.debug("Service worker registration has failed.")
+        });
+    }
 });
 
 // Show notice if serviceworker broadcasts that the current page comes from its cache
diff --git a/aleksis/core/static/js/progress.js b/aleksis/core/static/js/progress.js
index 8b897096f18c221bde8fd34c88bd93f9800ff06d..9be718b7e67f30e8cc77a436755e50f08b540fd3 100644
--- a/aleksis/core/static/js/progress.js
+++ b/aleksis/core/static/js/progress.js
@@ -47,7 +47,7 @@ function customSuccess(progressBarElement, progressBarMessageElement) {
     $("#result-icon").text("check_circle");
     $("#result-text").text(OPTIONS.success);
     $("#result-box").show();
-    const redirect = "redirect_on_success" in OPTIONS;
+    const redirect = "redirect_on_success" in OPTIONS && OPTIONS.redirect_on_success;
     if (redirect) {
         window.location.replace(OPTIONS.redirect_on_success);
     }
@@ -64,7 +64,7 @@ function customError(progressBarElement, progressBarMessageElement) {
 $(document).ready(function () {
     $("#progress-bar").removeClass("indeterminate").addClass("determinate");
 
-    var progressUrl = Urls["celeryProgress:taskStatus"](OPTIONS.task_id);
+    var progressUrl = Urls["taskStatus"](OPTIONS.task_id);
     CeleryProgressBar.initProgressBar(progressUrl, {
         onProgress: customProgress,
         onSuccess: customSuccess,
diff --git a/aleksis/core/static/js/search.js b/aleksis/core/static/js/search.js
index ea1dfb25b061f73a8a74c40edad3e3c8d073278e..fd121dbdfafe20c1c60ef2c793cd1f13b2a7573e 100644
--- a/aleksis/core/static/js/search.js
+++ b/aleksis/core/static/js/search.js
@@ -32,10 +32,11 @@ Autocomplete.prototype.setup = function () {
     // Trigger the "keyup" event if input gets focused
 
     this.query_box.focus(function () {
-        self.query_box.trigger("keydown");
+        self.query_box.trigger("input");
     });
 
-    this.query_box.keyup(function () {
+    this.query_box.on("input", () => {
+        console.log("Input changed, fetching again...")
         var query = self.query_box.val();
 
         if (query.length < self.minimum_length) {
@@ -97,11 +98,16 @@ Autocomplete.prototype.fetch = function (query) {
     var self = this;
 
     $.ajax({
-        url: this.url
-        , data: {
+        url: this.url,
+        data: {
             'q': query
-        }
-        , success: function (data) {
+        },
+        beforeSend: (request, settings) => {
+            $('#search-results').remove();
+            self.setLoader(true);
+        },
+        success: function (data) {
+            self.setLoader(false);
             self.show_results(data);
         }
     })
@@ -122,3 +128,7 @@ Autocomplete.prototype.setSelectedResult = function (element) {
     this.selected_element = element;
     console.log("New element: ", element);
 };
+
+Autocomplete.prototype.setLoader = function (value) {
+        $("#search-loader").css("display", (value === true ? "block" : "none"))
+}
diff --git a/aleksis/core/static/materialize.scss b/aleksis/core/static/public/materialize-custom.scss
similarity index 100%
rename from aleksis/core/static/materialize.scss
rename to aleksis/core/static/public/materialize-custom.scss
diff --git a/aleksis/core/static/style.scss b/aleksis/core/static/public/style.scss
similarity index 86%
rename from aleksis/core/static/style.scss
rename to aleksis/core/static/public/style.scss
index 1b6a8e7cb40d855320b15e1b8711d810183902e9..c2aa516e8cfcf88190aa13f43cc8f6aad5d747de 100644
--- a/aleksis/core/static/style.scss
+++ b/aleksis/core/static/public/style.scss
@@ -1,4 +1,4 @@
-@import "materialize";
+@import "materialize-custom";
 
 .primary-color {
   background-color: $primary-color !important;
@@ -87,12 +87,6 @@ header, main, footer {
   width: auto;
 }
 
-@media only screen and (max-width: 993px) {
-  header div.nav-wrapper {
-    z-index: -5;
-  }
-}
-
 
 /********/
 /* MAIN */
@@ -179,17 +173,25 @@ ul.sidenav li.logo > a:hover {
   box-shadow: none;
 }
 
-.sidenav li.search .search-wrapper > i.material-icons {
+.sidenav li.search .search-wrapper > button.search-button {
   position: absolute;
-  top: 21px;
+  top: calc(50% - 18px);
   right: 10px;
-  cursor: pointer;
+}
+
+button.btn-flat.search-button:hover {
+  background-color: $button-disabled-background;
 }
 
 a.collection-item.search-item {
   padding: 20px 10px;
 }
 
+div#search-loader {
+  margin: 0.5rem 0 0 0;
+  display: none;
+}
+
 div#search-results {
   position: absolute;
   margin-top: -10px;
@@ -271,6 +273,23 @@ ul.footer-ul {
   fill: white;
 }
 
+//////////////
+// HEADINGS //
+//////////////
+
+h1 {
+  font-weight: 300;
+  font-style: normal;
+  font-size: 3.6rem;
+  line-height: 110%;
+  margin: 1.4rem 0 1.68rem 0;
+}
+
+h2 {
+  font-weight: 300;
+  font-size: 3.0rem;
+}
+
 /* Collections */
 
 ul.collection .collection-item .title {
@@ -326,6 +345,11 @@ input[type="checkbox"].chips-checkbox + span {
   }
 }
 
+// Move label of ckeditor up
+.margin-top-35 {
+  margin-top: 35px;
+}
+
 // Badges
 
 span.badge.new::after {
@@ -508,33 +532,62 @@ th.orderable.desc > a {
   margin: 0;
 }
 
-.alert > p, .alert > div {
+.alert strong {
+  font-weight: 700;
+}
+
+figure.alert > :not(.material-icons.left){
+  margin-left: 39px;
+}
+
+.alert figcaption  {
+  font-weight: 700;
+  font-size: 17px;
+  margin: auto;
+}
+
+div.alert > p,
+div.alert > div,
+figure.alert{
   margin: 10px;
   padding: 10px;
   border-left: 5px solid;
 }
 
-.alert.success > p, .alert.success > div {
+div.alert.success > p,
+div.alert.success > div,
+figure.alert.success{
   @extend .success;
   border-color: color("green", "base");
 }
 
-.alert.error > p, .alert.error > div {
+div.alert.error > p,
+div.alert.error > div,
+figure.alert.error {
   @extend .error;
   border-color: color("red", "darken-4");
 }
 
-.alert.primary > p, .alert.primary > div, .alert.info > p, .alert.info > div {
+div.alert.primary > p,
+div.alert.primary > div,
+figure.alert.primary,
+div.alert.info > p,
+div.alert.info > div,
+figure.alert.info {
   background-color: #ececec;
   border-color: $primary-color;
 }
 
-.alert.warning p, .alert.warning div {
+div.alert.warning p,
+div.alert.warning div,
+figure.alert.warning {
   @extend .warning;
   border-color: color("orange", "darken-4");
 }
 
-main .alert p:first-child, main .alert div:first-child {
+main div.alert p:first-child,
+main div.alert div:first-child,
+main figure.alert {
   margin-left: -10px;
   margin-right: -10px;
 }
@@ -618,6 +671,11 @@ main .alert p:first-child, main .alert div:first-child {
   font-weight: 500;
 }
 
+.stacked-card-icon {
+  position: relative;
+  top: 50%;
+  transform: translateY(-50%);
+}
 
 .flex-row {
   display: flex;
@@ -691,3 +749,22 @@ main .alert p:first-child, main .alert div:first-child {
   background-color: inherit !important;
   color: inherit !important;
 }
+
+.break-word {
+  overflow-wrap: anywhere;
+}
+
+.card .card-action-light {
+	background-color: inherit;
+	border-top: 1px solid rgba(160, 160, 160, 0.2);
+	position: relative;
+	padding: 16px 24px;
+}
+
+.margin-top {
+  margin-top: 0.7rem !important;
+}
+
+.margin-bottom {
+  margin-bottom: 0.7rem !important;
+}
diff --git a/aleksis/core/static/theme.scss b/aleksis/core/static/public/theme.scss
similarity index 100%
rename from aleksis/core/static/theme.scss
rename to aleksis/core/static/public/theme.scss
diff --git a/aleksis/core/templates/500.html b/aleksis/core/templates/500.html
index a084d76f804f9fa1089c225ac4a16db7bf360f69..2abf1445494da40885904252ecbd0aa5bffed077 100644
--- a/aleksis/core/templates/500.html
+++ b/aleksis/core/templates/500.html
@@ -6,7 +6,7 @@
   <div class="container">
     <div class="card red">
       <div class="card-content white-text">
-        <div class="material-icons small left">error_outline</div>
+        <i class="material-icons small left">error_outline</i>
         <span class="card-title">{% trans "Error" %} (500): {% blocktrans %}An unexpected error has
           occured.{% endblocktrans %}</span>
         <p>
diff --git a/aleksis/core/templates/503.html b/aleksis/core/templates/503.html
index dd65828745eb846ee1f6b934043996ffa06759e1..97d76caa4fab9e641bf522779f7c282a9ab2d2dd 100644
--- a/aleksis/core/templates/503.html
+++ b/aleksis/core/templates/503.html
@@ -6,7 +6,7 @@
   <div class="container">
     <div class="card red">
       <div class="card-content white-text">
-        <div class="material-icons small left">error_outline</div>
+        <i class="material-icons small left">error_outline</i>
         <span class="card-title">{% blocktrans %}The maintenance mode is currently enabled. Please try again
           later.{% endblocktrans %}</span>
         <p>
diff --git a/aleksis/core/templates/account/account_inactive.html b/aleksis/core/templates/account/account_inactive.html
new file mode 100644
index 0000000000000000000000000000000000000000..62b03a3db859699aabb816ef1d05455777965bd9
--- /dev/null
+++ b/aleksis/core/templates/account/account_inactive.html
@@ -0,0 +1,24 @@
+{% extends "core/base.html" %}
+
+{% load i18n material_form %}
+
+{% block browser_title %}{% trans "Account inactive" %}{% endblock %}
+{% block page_title %}{% trans "Account inactive" %}{% endblock %}
+
+{% block content %}
+  <div class="container">
+    <div class="card red">
+      <div class="card-content white-text">
+        <div class="material-icons small left">error_outline</div>
+        <span class="card-title">{% blocktrans %}Account inactive.{% endblocktrans %}</span>
+        <p>
+          {% blocktrans %}
+            This account is currently inactive. If you think this is an
+            error, please contact one of your site administrators.
+          {% endblocktrans %}
+        </p>
+        {% include "core/partials/admins_list.html" %}
+      </div>
+    </div>
+  </div>
+{% endblock %}
diff --git a/aleksis/core/templates/account/email/base_message.txt b/aleksis/core/templates/account/email/base_message.txt
new file mode 100644
index 0000000000000000000000000000000000000000..307fdf2b8ed73b48906b13bb01728ccc91bb7856
--- /dev/null
+++ b/aleksis/core/templates/account/email/base_message.txt
@@ -0,0 +1,10 @@
+{% load i18n %}
+
+{% autoescape off %}
+
+{% blocktrans %}Hello!{% endblocktrans %}
+
+{% block content %}{% endblock %}
+
+{% trans "Your AlekSIS team" %}
+{% endautoescape %}
diff --git a/aleksis/core/templates/account/email_confirm.html b/aleksis/core/templates/account/email_confirm.html
new file mode 100644
index 0000000000000000000000000000000000000000..0f560556b6567f0a2133d0efb5c324730761243c
--- /dev/null
+++ b/aleksis/core/templates/account/email_confirm.html
@@ -0,0 +1,30 @@
+{% extends "core/base.html" %}
+
+{% load i18n material_form account %}
+
+{% block browser_title %}{% trans "Confirm" %}{% endblock %}
+{% block page_title %}{% trans "Confirm" %}{% endblock %}
+
+{% block content %}
+  {% if confirmation %}
+    {% user_display confirmation.email_address.user as user_display %}
+    <p class="flow-text">
+      {% blocktrans with confirmation.email_address.email as email %}Please confirm that <a href="mailto:{{ email }}">{{ email }}</a> is an e-mail address for user {{ user_display }}.{% endblocktrans %}
+    </p>
+    <form method="post" action="{% url 'account_confirm_email' confirmation.key %}">
+      {% csrf_token %}
+      {% form form=form %}{% endform %}
+      {% trans "Confirm" as caption %}
+      {% include "core/partials/save_button.html" with caption=caption icon="how_to_reg" %}
+    </form>
+  {% else %}
+    {% url "account_email" as email_url %}
+    <div class="alert warning">
+      <p>
+        <i class="material-icons left">warning</i>
+        {% blocktrans %}This e-mail confirmation link expired or is invalid. Please <a href="{{ email_url }}">issue a new e-mail confirmation request</a>.{% endblocktrans %}
+      </p>
+    </div>
+  {% endif %}
+
+{% endblock %}
diff --git a/aleksis/core/templates/account/password_change.html b/aleksis/core/templates/account/password_change.html
new file mode 100644
index 0000000000000000000000000000000000000000..be331751b31187a792b72cf97a178ddddf4de42b
--- /dev/null
+++ b/aleksis/core/templates/account/password_change.html
@@ -0,0 +1,23 @@
+{% extends "core/base.html" %}
+
+{% load i18n material_form %}
+
+{% block browser_title %}{% trans "Change password" %}{% endblock %}
+{% block page_title %}{% trans "Change password" %}{% endblock %}
+
+{% block content %}
+  <div class="alert warning">
+    <p>
+      <i class="material-icons left">warning</i>
+      {% trans "Forgot your current password? Click here to reset it:" %} <a href="{% url 'account_reset_password' %}">{% trans "Forgot Password?" %}</a>.
+    </p>
+  </div>
+
+  <form method="post" action="{% url 'account_change_password' %}">
+    {% csrf_token %}
+    {% form form=form %}{% endform %}
+    {% trans "Change password" as caption %}
+    {% include "core/partials/save_button.html" with caption=caption icon="priority_high" %}
+  </form>
+
+{% endblock %}
diff --git a/aleksis/core/templates/account/password_change_disabled.html b/aleksis/core/templates/account/password_change_disabled.html
new file mode 100644
index 0000000000000000000000000000000000000000..131df489102ec8d257f584892ceb94f04d8e93b1
--- /dev/null
+++ b/aleksis/core/templates/account/password_change_disabled.html
@@ -0,0 +1,24 @@
+{% extends "core/base.html" %}
+
+{% load i18n material_form %}
+
+{% block browser_title %}{% trans "Changing of password disabled" %}{% endblock %}
+{% block page_title %}{% trans "Changing of password disabled" %}{% endblock %}
+
+{% block content %}
+  <div class="container">
+    <div class="card red">
+      <div class="card-content white-text">
+        <div class="material-icons small left">error_outline</div>
+        <span class="card-title">{% blocktrans %}Changing of password disabled.{% endblocktrans %}</span>
+        <p>
+          {% blocktrans %}
+            Users are not allowed to edit their own passwords. If you think
+            this is an error please contact one of your site administrators.
+          {% endblocktrans %}
+        </p>
+        {% include "core/partials/admins_list.html" %}
+      </div>
+    </div>
+  </div>
+{% endblock %}
diff --git a/aleksis/core/templates/account/password_reset.html b/aleksis/core/templates/account/password_reset.html
new file mode 100644
index 0000000000000000000000000000000000000000..4e93262643e0e28919e405c9b2f0465bf2a2bcf8
--- /dev/null
+++ b/aleksis/core/templates/account/password_reset.html
@@ -0,0 +1,36 @@
+{% extends "core/base.html" %}
+
+{% load i18n material_form %}
+
+{% block browser_title %}{% trans "Reset password" %}{% endblock %}
+{% block no_page_title %}{% endblock %}
+
+{% block content %}
+  <div class="row">
+    <div class="col m1 l2 xl3"></div>
+    <div class="col s12 m10 l8 xl6">
+      <div class="card">
+        <form method="post" action="{% url 'account_reset_password' %}" class="password_reset">
+          <div class="card-content">
+            <div class="card-title">{% trans "Reset password" %}</div>
+              <p class="margin-bottom">
+                {% blocktrans %}Forgotten your password? Enter your e-mail address below, and we'll send you an e-mail allowing you to reset it.{% endblocktrans %}
+              </p> 
+              {% csrf_token %}
+              {% form form=form %}{% endform %}
+            </div>  
+          <div class="card-action-light">
+            {% trans "Reset password" as caption %}
+            {% include "core/partials/save_button.html" with caption=caption icon="priority_high" %}
+          </div>
+        </form>
+      </div>
+
+      <div class="grey-text text-darken-2">
+        {% blocktrans %}Please contact one of your site administrators, if you
+        have any trouble resetting your password:{% endblocktrans %}
+        {% include "core/partials/admins_list.html" %}
+      </div>
+    </div>
+  </div>
+{% endblock %}
diff --git a/aleksis/core/templates/account/password_reset_done.html b/aleksis/core/templates/account/password_reset_done.html
new file mode 100644
index 0000000000000000000000000000000000000000..ddd8b3871c242e8ee2eb9b22cc472778a3b096b9
--- /dev/null
+++ b/aleksis/core/templates/account/password_reset_done.html
@@ -0,0 +1,28 @@
+{% extends "core/base.html" %}
+
+{% load i18n %}
+
+{% block browser_title %}{% trans "Reset password" %}{% endblock %}
+{% block no_page_title %}{% endblock %}
+
+{% block content %}
+
+  <div class="container">
+    <div class="card green">
+      <div class="card-content white-text">
+        <div class="card-title">
+          <i class="material-icons small left">check_circle</i>
+          {% blocktrans %}Password reset mail sent{% endblocktrans %}
+        </div>
+        <p>
+          {% blocktrans %}
+            We have sent you an e-mail. Please contact one of your site
+            administrators if you do not receive it within a few minutes.
+          {% endblocktrans %}
+        </p>
+        {% include "core/partials/admins_list.html" %}
+      </div>
+    </div>
+  </div>
+
+{% endblock %}
diff --git a/aleksis/core/templates/account/password_reset_from_key.html b/aleksis/core/templates/account/password_reset_from_key.html
new file mode 100644
index 0000000000000000000000000000000000000000..4b2b91a28c875e7c1337fd0543646eae881da356
--- /dev/null
+++ b/aleksis/core/templates/account/password_reset_from_key.html
@@ -0,0 +1,67 @@
+{% extends "core/base.html" %}
+
+{% load i18n material_form %}
+
+{% block browser_title %}{% trans "Change password" %}{% endblock %}
+{% block no_page_title %}{% endblock %}
+
+{% block content %}
+  {% if token_fail %}
+    <div class="container">
+      <div class="card red">
+        <div class="card-content white-text">
+          <div class="card-title">
+            <i class="material-icons small left">error_outline</i>
+            {% blocktrans %}Bad token{% endblocktrans %}
+          </div>
+          <p>
+            {% url 'account_reset_password' as passwd_reset_url %}
+            {% blocktrans %}
+              The password reset link was invalid, possibly because it has already been used. Please request a <a href="{{ passwd_reset_url }}"
+              class="blue-text text-lighten-2">new password reset</a>.
+            {% endblocktrans %}
+          </p>
+          <p>
+            {% blocktrans %}
+              If this issue persists, please contact one of your site
+              administrators
+            {% endblocktrans %}
+          </p>
+          {% include "core/partials/admins_list.html" %}
+        </div>
+      </div>
+    </div>
+  {% else  %}
+  <div class="row">
+    <div class="col m1 l2 xl3"></div>
+    <div class="col s12 m10 l8 xl6">
+      <div class="card">
+        {% if form %}
+          <form method="post" action="{{ action_url }}">
+            <div class="card-content">
+              <div class="card-title">{% trans "Change password" %}</div>
+                {% csrf_token %}
+                {% form form=form %}{% endform %}
+              <div class="card-action-light">
+                {% trans "Change password" as caption %}
+                {% include "core/partials/save_button.html" with caption=caption icon="priority_high" %}
+              </div>
+            </div>
+          </form>
+        {% else %}
+      </div>
+      <div class="alert success">
+        <p>
+          <i class="material-icons left">success</i>
+          {% blocktrans %}
+            Your password is now changed!
+          {% endblocktrans %}
+        </p>
+        <div class="card-action">
+          <a href="{% url "login" %}">{% trans "Back to login" %}</a>
+        </div>
+      </div>
+    {% endif %}
+  {% endif %}
+
+{% endblock %}
diff --git a/aleksis/core/templates/account/password_reset_from_key_done.html b/aleksis/core/templates/account/password_reset_from_key_done.html
new file mode 100644
index 0000000000000000000000000000000000000000..b32538e53ee5b1b0f2115e645d0f955d235ba958
--- /dev/null
+++ b/aleksis/core/templates/account/password_reset_from_key_done.html
@@ -0,0 +1,22 @@
+{% extends "core/base.html" %}
+
+{% load i18n %}
+
+{% block browser_title %}{% trans "Change password" %}{% endblock %}
+{% block page_title %}{% trans "Change password" %}{% endblock %}
+
+{% block content %}
+  <div class="container">
+    <div class="card green">
+      <div class="card-content white-text">
+        <div class="material-icons small left">success</div>
+        <span class="card-title">{% blocktrans %}Password changed!{% endblocktrans %}</span>
+        <p>
+          {% blocktrans %}
+            Your password is now changed!
+          {% endblocktrans %}
+        </p>
+      </div>
+    </div>
+  </div>
+{% endblock %}
diff --git a/aleksis/core/templates/account/password_set.html b/aleksis/core/templates/account/password_set.html
new file mode 100644
index 0000000000000000000000000000000000000000..08819c1e0904f53e62f11d86bd2e82ccef7b2469
--- /dev/null
+++ b/aleksis/core/templates/account/password_set.html
@@ -0,0 +1,16 @@
+{% extends "core/base.html" %}
+
+{% load i18n material_form %}
+
+{% block browser_title %}{% trans "Set password" %}{% endblock %}
+{% block page_title %}{% trans "Set password" %}{% endblock %}
+
+{% block content %}
+  <form method="post" action="{% url 'account_set_password' %}">
+    {% csrf_token %}
+    {% form form=form %}{% endform %}
+    {% trans "Set password" as caption %}
+    {% include "core/partials/save_button.html" with caption=caption icon="priority_high" %}
+  </form>
+
+{% endblock %}
diff --git a/aleksis/core/templates/account/signup.html b/aleksis/core/templates/account/signup.html
new file mode 100644
index 0000000000000000000000000000000000000000..b4a74179908dafc208d34c5b2c472f3bd73c3de4
--- /dev/null
+++ b/aleksis/core/templates/account/signup.html
@@ -0,0 +1,24 @@
+{% extends "core/base.html" %}
+
+{% load i18n material_form %}
+
+{% block browser_title %}{% trans "Signup" %}{% endblock %}
+{% block page_title %}{% trans "Signup" %}{% endblock %}
+
+{% block content %}
+  <div class="alert warning">
+    <p>
+      <i class="material-icons left">warning</i>
+      {% blocktrans %}Already have an account? Then please <a href="{{ login_url }}">sign in</a>.{% endblocktrans %}
+    </p>
+  </div>
+
+  <form method="post" action="{% url 'account_signup' %}">
+    {% csrf_token %}
+    {% form form=form %}{% endform %}
+    <input type="hidden" name="{{ redirect_field_name }}" value="{{ redirect_field_value }}" />
+    {% trans "Sign up" as caption %}
+    {% include "core/partials/save_button.html" with caption=caption icon="how_to_reg" %}
+  </form>
+
+{% endblock %}
diff --git a/aleksis/core/templates/account/signup_closed.html b/aleksis/core/templates/account/signup_closed.html
new file mode 100644
index 0000000000000000000000000000000000000000..33ef749f5598fbd209e3f72dc2b4a79620f17b1a
--- /dev/null
+++ b/aleksis/core/templates/account/signup_closed.html
@@ -0,0 +1,24 @@
+{% extends "core/base.html" %}
+
+{% load i18n material_form %}
+
+{% block browser_title %}{% trans "Signup closed" %}{% endblock %}
+{% block page_title %}{% trans "Signup closed" %}{% endblock %}
+
+{% block content %}
+  <div class="container">
+    <div class="card red">
+      <div class="card-content white-text">
+        <div class="material-icons small left">error_outline</div>
+        <span class="card-title">{% blocktrans %}Signup closed.{% endblocktrans %}</span>
+        <p>
+          {% blocktrans %}
+            This sign up is currently closed. If you think this is an
+            error, please contact one of your site administrators.
+          {% endblocktrans %}
+        </p>
+        {% include "core/partials/admins_list.html" %}
+      </div>
+    </div>
+  </div>
+{% endblock %}
diff --git a/aleksis/core/templates/account/verification_email_required.html b/aleksis/core/templates/account/verification_email_required.html
new file mode 100644
index 0000000000000000000000000000000000000000..b1a4a482a3ee64b0f2c6339e76ed44c66b29ef50
--- /dev/null
+++ b/aleksis/core/templates/account/verification_email_required.html
@@ -0,0 +1,26 @@
+{% extends "core/base.html" %}
+
+{% load i18n %}
+
+{% block browser_title %}{% trans "Reset password" %}{% endblock %}
+{% block page_title %}{% trans "Reset password" %}{% endblock %}
+
+{% block content %}
+
+  <div class="container">
+    <div class="card green">
+      <div class="card-content white-text">
+        <div class="material-icons small left">success</div>
+        <span class="card-title">{% blocktrans %}Password reset mail sent!{% endblocktrans %}</span>
+        <p>
+          {% blocktrans %}
+            We have sent you an e-mail. Please contact one of your site
+            administrators if you do not receive it within a few minutes.
+          {% endblocktrans %}
+        </p>
+        {% include "core/partials/admins_list.html" %}
+      </div>
+    </div>
+  </div>
+
+{% endblock %}
diff --git a/aleksis/core/templates/account/verification_sent.html b/aleksis/core/templates/account/verification_sent.html
new file mode 100644
index 0000000000000000000000000000000000000000..dd486aeb2bf212838a014fe868db4cfe6716cf8d
--- /dev/null
+++ b/aleksis/core/templates/account/verification_sent.html
@@ -0,0 +1,37 @@
+{% extends "core/base.html" %}
+
+{% load i18n %}
+
+{% block browser_title %}{% trans "Verify your email address" %}{% endblock %}
+{% block page_title %}{% trans "Verify your email address" %}{% endblock %}
+
+{% block content %}
+
+  <div class="container">
+    <div class="card red">
+      <div class="card-content white-text">
+        <div class="material-icons small left">error_outline</div>
+        <span class="card-title">{% blocktrans %}Verify your email!{% endblocktrans %}</span>
+        <p>
+          {% blocktrans %}
+            This part of the site requires us to verify that you are who you claim to be.
+            For this purpose, we require that you verify ownership of your e-mail address.
+          {% endblocktrans %}
+        </p>
+        <p>
+          {% blocktrans %}
+            We have sent an e-mail to you for verification.
+            Please click on the link inside this e-mail. Please
+            contact us if you do not receive it within a few minutes.
+          {% endblocktrans %}
+        </p>
+        <p>
+          {% url 'account_email' as email_url %}
+          {% blocktrans with email_url=email_url %}<strong>Note:</strong> you can still <a href="{{ email_url }}">change your e-mail address</a>{% endblocktrans %}
+        </p>
+        {% include "core/partials/admins_list.html" %}
+      </div>
+    </div>
+  </div>
+
+{% endblock %}
diff --git a/aleksis/core/templates/core/base.html b/aleksis/core/templates/core/base.html
index d34db4f65d05021f1d68e61190bf323c249e5ada..0dd75f2c6f237dea1ecac47b010abfac9453000f 100644
--- a/aleksis/core/templates/core/base.html
+++ b/aleksis/core/templates/core/base.html
@@ -1,6 +1,6 @@
 {# -*- engine:django -*- #}
 
-{% load i18n menu_generator static sass_tags any_js pwa rules %}
+{% load i18n menu_generator static sass_tags any_js rules %}
 {% get_current_language as LANGUAGE_CODE %}
 
 
@@ -18,8 +18,13 @@
 
   {# CSS #}
   {% include_css "material-design-icons" %}
-  {% include_css "Roboto" %}
-  <link rel="stylesheet" href="{% sass_src 'style.scss' %}">
+  {% include_css "Roboto100" %}
+  {% include_css "Roboto300" %}
+  {% include_css "Roboto400" %}
+  {% include_css "Roboto500" %}
+  {% include_css "Roboto700" %}
+  {% include_css "Roboto900" %}
+  <link rel="stylesheet" href="{% sass_src 'public/style.scss' %}">
 
   {# Add JS URL resolver #}
   <script src="{% url "js_reverse" %}" type="text/javascript"></script>
@@ -30,6 +35,24 @@
   <script src="{% url "calendarweek_i18n_js" %}?first_day=6&amp;locale={{ LANGUAGE_CODE }}"
           type="text/javascript"></script>
 
+  {% if SENTRY_ENABLED %}
+    {% if SENTRY_TRACE_ID %}
+      <meta name="sentry-trace" content="{{ SENTRY_TRACE_ID }}" />
+    {% endif %}
+    {% include_js "Sentry" %}
+    {{ SENTRY_SETTINGS|json_script:"sentry_settings" }}
+    <script type="text/javascript">
+      const sentry_settings = JSON.parse(document.getElementById('sentry_settings').textContent);
+
+      Sentry.init({
+        dsn: sentry_settings.dsn,
+        environment: sentry_settings.environment,
+        tracesSampleRate: sentry_settings.traces_sample_rate,
+        integrations: [new Sentry.Integrations.BrowserTracing()]
+      });
+    </script>
+  {% endif %}
+
   {# Include jQuery early to provide $(document).ready #}
   {% include_js "jQuery" %}
 
@@ -47,9 +70,9 @@
 
   <!-- Nav bar (logged in as, logout) -->
   <nav>
-    <a class="brand-logo" href="/">{{ request.site.preferences.general__title }}</a>
-
     <div class="nav-wrapper">
+      <a class="brand-logo" href="/">{{ request.site.preferences.general__title }}</a>
+
       <ul id="nav-mobile" class="right hide-on-med-and-down">
         {% if user.is_authenticated %}
           <li>{% trans "Logged in as" %} {{ user.get_username }}</li>
@@ -70,13 +93,16 @@
              alt="{{ request.site.preferences.general__title }} – Logo">
       </a>
     </li>
-    {% has_perm 'core.search' user as search %}
+    {% has_perm 'core.search_rule' user as search %}
     {% if search %}
       <li class="search">
         <form method="get" action="{% url "haystack_search" %}" id="search-form" class="autocomplete">
           <div class="search-wrapper">
-            <input id="search" name="q" placeholder="{% trans "Search" %}">
-            <i class="material-icons">search</i>
+            <input id="search" name="q" type="search" enterkeyhint="search" placeholder="{% trans "Search" %}">
+            <button class="btn btn-flat search-button" type="submit" aria-label="{% trans "Search" %}">
+              <i class="material-icons">search</i>
+            </button>
+            <div class="progress" id="search-loader"><div class="indeterminate"></div></div>
           </div>
         </form>
       </li>
@@ -93,8 +119,7 @@
 
   {% if messages %}
     {% for message in messages %}
-      <div class="alert {% if message.tags %}{{ message.tags }}{% else %}info{% endif %}">
-        <p>
+      <figure class="alert {% if message.tags %}{{ message.tags }}{% else %}info{% endif %}">
           {% if message.tags == "success" %}
             <i class="material-icons left">check_circle</i>
           {% elif  message.tags == "info" %}
@@ -105,13 +130,12 @@
             <i class="material-icons left">error</i>
           {% endif %}
           {{ message }}
-        </p>
-      </div>
+      </figure>
     {% endfor %}
   {% endif %}
 
   {% block no_page_title %}
-    <h4>{% block page_title %}{% endblock %}</h4>
+    <h1>{% block page_title %}{% endblock %}</h1>
   {% endblock %}
 
   {% block content %}{% endblock %}
@@ -151,12 +175,12 @@
       </div>
       <div class="right">
         <span id="doit"></span>
-        {% if request.site.preferences.footer__impress_url %}
-          <a class="blue-text text-lighten-4" href="{{ request.site.preferences.footer__impress_url }}">
+        {% if request.site.preferences.footer__imprint_url %}
+          <a class="blue-text text-lighten-4" href="{{ request.site.preferences.footer__imprint_url }}">
             {% trans "Impress" %}
           </a>
         {% endif %}
-        {% if request.site.preferences.footer__privacy_url and request.site.preferences.footer__impress_url %}
+        {% if request.site.preferences.footer__privacy_url and request.site.preferences.footer__imprint_url %}
           ·
         {% endif %}
         {% if request.site.preferences.footer__privacy_url %}
diff --git a/aleksis/core/templates/core/base_print.html b/aleksis/core/templates/core/base_print.html
index 9539808f963d541811d9581ca377006bd1ef74a8..428d28f2858406c771b635c20ff874bf5c9f9d5a 100644
--- a/aleksis/core/templates/core/base_print.html
+++ b/aleksis/core/templates/core/base_print.html
@@ -15,9 +15,14 @@
   </title>
 
   {% include_css "material-design-icons" %}
-  {% include_css "Roboto" %}
+  {% include_css "Roboto100" %}
+  {% include_css "Roboto300" %}
+  {% include_css "Roboto400" %}
+  {% include_css "Roboto500" %}
+  {% include_css "Roboto700" %}
+  {% include_css "Roboto900" %}
   {% include_css "paper-css" %}
-  <link rel="stylesheet" href="{% sass_src 'style.scss' %}"/>
+  <link rel="stylesheet" href="{% sass_src 'public/style.scss' %}"/>
   <link rel="stylesheet" href="{% static "print.css" %}"/>
   {% if landscape %}
     <link rel="stylesheet" href="{% static 'print_landscape.css' %}"/>
diff --git a/aleksis/core/templates/core/dashboard_widget/dashboardwidget_broken.html b/aleksis/core/templates/core/dashboard_widget/dashboardwidget_broken.html
new file mode 100644
index 0000000000000000000000000000000000000000..23a18714637bf90d9610547210b412eec6fae729
--- /dev/null
+++ b/aleksis/core/templates/core/dashboard_widget/dashboardwidget_broken.html
@@ -0,0 +1,22 @@
+{% load i18n %}
+
+<div class="card horizontal">
+  <div class="card-image">
+    <i class="material-icons large red-text stacked-card-icon">assignment_late</i>
+  </div>
+  <div class="card-stacked">
+    <div class="card-content">
+      <div class="card-title">
+        {% trans "This widget is currently not available." %}
+      </div>
+      {% if title %}
+        <p>
+          {% blocktrans %}
+            There is a problem getting the widget "{{ title }}".
+            There is no need for you to take any action.
+          {% endblocktrans %}
+        </p>
+      {% endif %}
+    </div>
+  </div>
+</div>
diff --git a/aleksis/core/templates/core/dashboard_widget/list.html b/aleksis/core/templates/core/dashboard_widget/list.html
index 6715e3df96e2faed4a1afc78333529039000de32..ec5fac70d20c3997d2a02128691b947719eb4309 100644
--- a/aleksis/core/templates/core/dashboard_widget/list.html
+++ b/aleksis/core/templates/core/dashboard_widget/list.html
@@ -25,7 +25,7 @@
     {% endfor %}
   </ul>
 
-  {% has_perm "core.edit_default_dashboard" user as can_edit_default_dashboard %}
+  {% has_perm "core.edit_default_dashboard_rule" user as can_edit_default_dashboard %}
   {% if can_edit_default_dashboard %}
     <a class="btn orange waves-effect waves-light" href="{% url "edit_default_dashboard" %}">
       <i class="material-icons left">edit</i>
diff --git a/aleksis/core/templates/core/edit_dashboard.html b/aleksis/core/templates/core/edit_dashboard.html
index 09d9afd97ee8b983ed4577267641eae77fd2f14e..da50adeea4b2380075113753a9626ef3cbb30fbe 100644
--- a/aleksis/core/templates/core/edit_dashboard.html
+++ b/aleksis/core/templates/core/edit_dashboard.html
@@ -45,20 +45,20 @@
     {% include "core/partials/save_button.html" %}
   </form>
 
-  <h5>{% trans "Available widgets" %}</h5>
+  <h2>{% trans "Available widgets" %}</h2>
   <div class="row card-panel grey lighten-3" id="not-used-widgets">
     {% for widget in not_used_widgets %}
       {% include "core/partials/edit_dashboard_widget.html" %}
     {% endfor %}
   </div>
 
-  <h5>
+  <h2>
     {% if not default_dashboard %}
       {% trans "Your dashboard" %}
     {% else %}
       {% trans "Default dashboard" %}
     {% endif %}
-  </h5>
+  </h2>
 
   <div class="row card-panel grey lighten-3" id="widgets">
     {% for widget in widgets %}
diff --git a/aleksis/core/templates/core/group/child_groups.html b/aleksis/core/templates/core/group/child_groups.html
index bb5a429f6ec09effe094e14663f86187258919d3..900df71a59f24817bbec128b5ec15748b5a9d752 100644
--- a/aleksis/core/templates/core/group/child_groups.html
+++ b/aleksis/core/templates/core/group/child_groups.html
@@ -36,7 +36,7 @@
       </a>
     </form>
 
-    <h5>{% trans "Currently selected groups" %}</h5>
+    <h2>{% trans "Currently selected groups" %}</h2>
 
     {% for group in filter.qs %}
       <div class="chip">
diff --git a/aleksis/core/templates/core/group/full.html b/aleksis/core/templates/core/group/full.html
index 1a15daf5c490c4cd1af12e7c43c02a3872c8b883..65d94ad13a7e563302e9ae283fe4df24c6135353 100644
--- a/aleksis/core/templates/core/group/full.html
+++ b/aleksis/core/templates/core/group/full.html
@@ -9,12 +9,12 @@
 {% block browser_title %}{{ group.name }}{% endblock %}
 
 {% block content %}
-  <h4>{{ group.name }} <small class="grey-text">{{ group.short_name }}</small></h4>
+  <h1>{{ group.name }} <small class="grey-text">{{ group.short_name }}</small></h1>
 
-  {% has_perm 'core.edit_group' user group as can_change_group %}
-  {% has_perm 'core.change_group_preferences' user group as can_change_group_preferences %}
-  {% has_perm 'core.delete_group' user group as can_delete_group %}
-  {% has_perm 'core.view_group_stats' user group as can_view_group_stats %}
+  {% has_perm 'core.edit_group_rule' user group as can_change_group %}
+  {% has_perm 'core.change_group_preferences_rule' user group as can_change_group_preferences %}
+  {% has_perm 'core.delete_group_rule' user group as can_delete_group %}
+  {% has_perm 'core.view_group_stats_rule' user group as can_view_group_stats %}
 
   {% if can_change_group or can_change_group_preferences or can_delete_group %}
     <p>
@@ -61,7 +61,7 @@
   </table>
 
   {% if can_view_group_stats %}
-    <h5>{% blocktrans %}Statistics{% endblocktrans %}</h5>
+    <h2>{% blocktrans %}Statistics{% endblocktrans %}</h2>
     <ul>
       <li>
         {% trans "Count of members" %}: {{ stats.members }}
@@ -79,10 +79,10 @@
     </ul>
   {% endif %}
 
-  <h5>{% blocktrans %}Owners{% endblocktrans %}</h5>
+  <h2>{% blocktrans %}Owners{% endblocktrans %}</h2>
   {% render_table owners_table %}
 
-  <h5>{% blocktrans %}Members{% endblocktrans %}</h5>
+  <h2>{% blocktrans %}Members{% endblocktrans %}</h2>
   {% render_table members_table %}
 
 {% endblock %}
diff --git a/aleksis/core/templates/core/group/list.html b/aleksis/core/templates/core/group/list.html
index 436b193e74f4c7e7e718fe5ba69a0b9fa0812a2b..9fe2c925f79f423cba0e8d195fb0299cca1c906b 100644
--- a/aleksis/core/templates/core/group/list.html
+++ b/aleksis/core/templates/core/group/list.html
@@ -14,7 +14,7 @@
     {% trans "Create group" %}
   </a>
 
-  <h5>{% trans "Filter groups" %}</h5>
+  <h2>{% trans "Filter groups" %}</h2>
   <form method="get">
     {% form form=groups_filter.form %}{% endform %}
     {% trans "Search" as caption %}
@@ -25,6 +25,6 @@
     </button>
   </form>
 
-  <h5>{% trans "Selected groups" %}</h5>
+  <h2>{% trans "Selected groups" %}</h2>
   {% render_table groups_table %}
 {% endblock %}
diff --git a/aleksis/core/templates/core/index.html b/aleksis/core/templates/core/index.html
index 70103f517963f3a9e306e43c9212ac856594b229..4de8ddcf35ef9153d094798c5cc019aa1bb0f863 100644
--- a/aleksis/core/templates/core/index.html
+++ b/aleksis/core/templates/core/index.html
@@ -9,20 +9,18 @@
 {% endblock %}
 
 {% block content %}
-  {% has_perm "core.edit_dashboard" user as can_edit_dashboard %}
+  {% has_perm "core.edit_dashboard_rule" user as can_edit_dashboard %}
   {% if can_edit_dashboard %}
-    <a class="btn-flat waves-effect waves-light right" href="{% url "edit_dashboard" %}">
-      <i class="material-icons left">edit</i>
-      {% trans "Edit dashboard" %}
-    </a>
+    <div class="row no-margin">
+      <a class="btn-flat waves-effect waves-light right" href="{% url "edit_dashboard" %}">
+        <i class="material-icons left">edit</i>
+        {% trans "Edit dashboard" %}
+      </a>
+    </div>
   {% endif %}
-  <h4>
-    {{ request.site.preferences.general__title }}
-  </h4>
 
   {% for notification in unread_notifications %}
-    <div class="alert primary scale-transition">
-      <div>
+    <figure class="alert primary scale-transition">
         <i class="material-icons left">info</i>
 
         <div class="right">
@@ -31,10 +29,9 @@
           </a>
         </div>
 
-        <strong>{{ notification.title }}</strong>
+        <figcaption>{{ notification.title }}</figcaption>
         <p>{{ notification.description|linebreaks }}</p>
-      </div>
-    </div>
+    </figure>
   {% endfor %}
 
   {% include "core/partials/announcements.html" with announcements=announcements %}
@@ -59,7 +56,7 @@
   {% if activities or notifications %}
     <div class="row">
       <div class="col s12 m6">
-        <h5>{% blocktrans %}Last activities{% endblocktrans %}</h5>
+        <h2>{% blocktrans %}Last activities{% endblocktrans %}</h2>
 
         {% if activities %}
           <ul class="collection">
@@ -82,7 +79,7 @@
       </div>
 
       <div class="col s12 m6">
-        <h5>{% blocktrans %}Recent notifications{% endblocktrans %}</h5>
+        <h2>{% blocktrans %}Recent notifications{% endblocktrans %}</h2>
 
         {% if notifications %}
           <ul class="collection">
@@ -111,5 +108,7 @@
     </div>
   {% endif %}
 
-  <script type="text/javascript" src="{% static "js/include_ajax_live.js" %}"></script>
+  {% if user.person.preferences.general__automatically_update_dashboard and request.site.preferences.general__automatically_update_dashboard_site %}
+    <script type="text/javascript" src="{% static "js/include_ajax_live.js" %}"></script>
+  {% endif %}
 {% endblock %}
diff --git a/aleksis/core/templates/core/partials/alternative_login_options.html b/aleksis/core/templates/core/partials/alternative_login_options.html
deleted file mode 100644
index 24d36e9ef76f3da11554169106fc1eafd3c7ac46..0000000000000000000000000000000000000000
--- a/aleksis/core/templates/core/partials/alternative_login_options.html
+++ /dev/null
@@ -1,10 +0,0 @@
-{% if ALTERNATIVE_LOGIN_VIEWS %}
-  <p>
-    {% for backend, url, icon, text in ALTERNATIVE_LOGIN_VIEWS %}
-      <a class="btn-large waves-effect waves-light primary" href="{% url url %}">
-        <i class="material-icons left">{{ icon }}</i>
-        {{ text }}
-      </a>
-    {% endfor %}
-  </p>
-{% endif %}
diff --git a/aleksis/core/templates/core/partials/announcements.html b/aleksis/core/templates/core/partials/announcements.html
index 2c74bead64e6c0f0f9da1753114d76357c2994e1..e3dcd5165f4149e3d5fa2016487ef771ad84f12d 100644
--- a/aleksis/core/templates/core/partials/announcements.html
+++ b/aleksis/core/templates/core/partials/announcements.html
@@ -1,8 +1,7 @@
 {% load i18n humanize %}
 
 {% for announcement in announcements %}
-  <div class="alert primary">
-    <div>
+  <figure class="alert primary">
       {% if show_interval %}
         <em class="right hide-on-small-and-down">
           {% if announcement.valid_from.date == announcement.valid_until.date %}
@@ -25,8 +24,8 @@
         </p>
       {% endif %}
 
+      <figcaption>{{ announcement.title }}</figcaption>
       <p>
-        <strong>{{ announcement.title }}</strong> <br/>
         {{ announcement.description }}
       </p>
 
@@ -43,6 +42,5 @@
           {% endif %}
         </em>
       {% endif %}
-    </div>
-  </div>
+  </figure>
 {% endfor %}
diff --git a/aleksis/core/templates/core/partials/meta.html b/aleksis/core/templates/core/partials/meta.html
index 04f917cf102276a4e6c8d8ab06b9cad0d69033d8..8fdbea9e15dd2c9520d8d6643ab251337da68609 100644
--- a/aleksis/core/templates/core/partials/meta.html
+++ b/aleksis/core/templates/core/partials/meta.html
@@ -1,4 +1,4 @@
-{% load pwa favtags %}
+{% load favtags %}
 
 <meta charset="utf-8"/>
 <meta http-equiv="X-UA-Compatible" content="IE=edge"/>
@@ -6,5 +6,31 @@
 <meta name="description" content="{{ request.site.preferences.general__description }}"/>
 <meta name="generator" content="AlekSIS School Information System"/>
 
+<meta name="theme-color" content="red">
+<meta name="mobile-web-app-capable" content="yes">
+
+<meta name="apple-mobile-web-app-title" content="{{ request.site.preferences.general__title }}">
+<meta name="apple-mobile-web-app-capable" content="yes">
+<meta name="apple-mobile-web-app-status-bar-style" content="default">
+
+<meta name="msapplication-navbutton-color" content="{{ request.site.preferences.theme__primary }}">
+<meta name="msapplication-TileColor" content="{{ request.site.preferences.theme__primary }}">
+<meta name="msapplication-TileImage" content="{{ PWA_ICONS.microsoft.144.faviconImage.url }}">
+<meta name="application-name" content="{{ request.site.preferences.general__title }}">
+<meta name="msapplication-starturl" content="/">
+<meta name="msapplication-tap-highlight" content="no">
+<meta name="browsermode" content="application">
+
+<link href="/manifest.json" rel="manifest">
+
 {% place_favicon %}
-{% progressive_web_app_meta %}
+
+{% for icon in PWA_ICONS.apple.values %}
+  <link rel="apple-touch-icon" href="{{ icon.faviconImage.url }}" sizes="{{ icon.size }}x{{ icon.size }}">
+{% endfor %}
+
+{% with icon=PWA_ICONS.apple_splash.192 %}
+  <link href="{{ icon.faviconImage.url }}" media="(device-width: 320px) and (device-height: 568px) and (-webkit-device-pixel-ratio: 2)" rel="apple-touch-startup-image"/>
+{% endwith %}
+
+
diff --git a/aleksis/core/templates/core/person/accounts.html b/aleksis/core/templates/core/person/accounts.html
deleted file mode 100644
index 03725518dfcbf8552a53199790daa44d541bac32..0000000000000000000000000000000000000000
--- a/aleksis/core/templates/core/person/accounts.html
+++ /dev/null
@@ -1,65 +0,0 @@
-{# -*- engine:django -*- #}
-
-{% extends "core/base.html" %}
-
-{% load i18n any_js %}
-
-{% block extra_head %}
-  {{ persons_accounts_formset.media.css }}
-  {% include_css "select2-materialize" %}
-{% endblock %}
-
-{% block browser_title %}{% blocktrans %}Link persons to accounts{% endblocktrans %}{% endblock %}
-{% block page_title %}
-  {% blocktrans %}Link persons to accounts{% endblocktrans %}
-{% endblock %}
-
-{% block content %}
-  <div class="alert info">
-    <p>
-      <i class="material-icons left">info</i>
-      {% blocktrans %}
-        You can use this form to assign user accounts to persons. Use the
-        dropdowns to select existing accounts; use the text fields to create new
-        accounts on-the-fly. The latter will create a new account with the
-        entered username and copy all other details from the person.
-      {% endblocktrans %}
-    </p>
-  </div>
-
-  <form method="post">
-    {% csrf_token %}
-    {{ persons_accounts_formset.management_form }}
-
-    <button type="submit" class="btn green waves-effect waves-light">
-      <i class="material-icons left">save</i>
-      {% blocktrans %}Update{% endblocktrans %}
-    </button>
-
-    <table>
-      <tr>
-        <th>{% blocktrans %}Person{% endblocktrans %}</th>
-        <th>{% blocktrans %}Existing account{% endblocktrans %}</th>
-        <th>{% blocktrans %}New account{% endblocktrans %}</th>
-      </tr>
-      {% for form in persons_accounts_formset %}
-        {{ form.id }}
-        <tr>
-          <td>
-            {{ form.last_name }}
-            {{ form.first_name }}
-          </td>
-          <td>{{ form.user }}</td>
-          <td>{{ form.new_user }}</td>
-        </tr>
-      {% endfor %}
-    </table>
-
-    <button type="submit" class="btn green waves-effect waves-light">
-      <i class="material-icons left">save</i>
-      {% blocktrans %}Update{% endblocktrans %}
-    </button>
-  </form>
-  {% include_js "select2-materialize" %}
-  {{ persons_accounts_formset.media.js }}
-{% endblock %}
diff --git a/aleksis/core/templates/core/person/create.html b/aleksis/core/templates/core/person/create.html
new file mode 100644
index 0000000000000000000000000000000000000000..08a6639e05e45f82912fd3e4adb25021264010c9
--- /dev/null
+++ b/aleksis/core/templates/core/person/create.html
@@ -0,0 +1,24 @@
+{# -*- engine:django -*- #}
+
+{% extends "core/base.html" %}
+
+{% load material_form i18n any_js %}
+
+{% block extra_head %}
+  {{ form.media.css }}
+  {% include_css "select2-materialize" %}
+{% endblock %}
+
+{% block browser_title %}{% blocktrans %}Create person{% endblocktrans %}{% endblock %}
+{% block page_title %}{% blocktrans %}Create person{% endblocktrans %}{% endblock %}
+
+
+{% block content %}
+  <form method="post" enctype="multipart/form-data">
+    {% csrf_token %}
+    {% form form=form %}{% endform %}
+    {% include "core/partials/save_button.html" %}
+  </form>
+  {% include_js "select2-materialize" %}
+  {{ form.media.js }}
+{% endblock %}
diff --git a/aleksis/core/templates/core/person/edit.html b/aleksis/core/templates/core/person/edit.html
index 3bf16ca3521ea744acfe48c6829dfae21db06eb5..788721c3541cba056c1ef05214eb0c39eb1b6f69 100644
--- a/aleksis/core/templates/core/person/edit.html
+++ b/aleksis/core/templates/core/person/edit.html
@@ -5,7 +5,7 @@
 {% load material_form i18n any_js %}
 
 {% block extra_head %}
-  {{ edit_person_form.media }}
+  {{ form.media.css }}
   {% include_css "select2-materialize" %}
 {% endblock %}
 
@@ -14,13 +14,11 @@
 
 
 {% block content %}
-
   <form method="post" enctype="multipart/form-data">
     {% csrf_token %}
-    {% form form=edit_person_form %}{% endform %}
+    {% form form=form %}{% endform %}
     {% include "core/partials/save_button.html" %}
   </form>
   {% include_js "select2-materialize" %}
-  {{ edit_group_form.media.js }}
-
+  {{ form.media.js }}
 {% endblock %}
diff --git a/aleksis/core/templates/core/person/full.html b/aleksis/core/templates/core/person/full.html
index 80aff9b9fe0b2d3c4146929758430b3401bf5270..d16354f00ebcd2162ffc7bab16715707b41adb81 100644
--- a/aleksis/core/templates/core/person/full.html
+++ b/aleksis/core/templates/core/person/full.html
@@ -8,12 +8,12 @@
 {% block browser_title %}{{ person.first_name }} {{ person.last_name }}{% endblock %}
 
 {% block content %}
-  <h4>{{ person.first_name }} {{ person.last_name }}</h4>
+  <h1>{{ person.first_name }} {{ person.last_name }}</h1>
 
-  {% has_perm 'core.edit_person' user person as can_change_person %}
-  {% has_perm 'core.change_person_preferences' user person as can_change_person_preferences %}
-  {% has_perm 'core.delete_person' user person as can_delete_person %}
-  {% has_perm "core.impersonate" user person as can_impersonate %}
+  {% has_perm 'core.edit_person_rule' user person as can_change_person %}
+  {% has_perm 'core.change_person_preferences_rule' user person as can_change_person_preferences %}
+  {% has_perm 'core.delete_person_rule' user person as can_delete_person %}
+  {% has_perm "core.impersonate_rule" user person as can_impersonate %}
 
   {% if can_change_person or can_change_person_preferences or can_delete_person or can_impersonate %}
     <p>
@@ -47,10 +47,10 @@
     </p>
   {% endif %}
 
-  <h5>{% blocktrans %}Contact details{% endblocktrans %}</h5>
+  <h2>{% blocktrans %}Contact details{% endblocktrans %}</h2>
   <div class="row">
     <div class="col s12 m4">
-      {% has_perm 'core.view_photo' user person as can_view_photo %}
+      {% has_perm 'core.view_photo_rule' user person as can_view_photo %}
       {% if person.photo and can_view_photo %}
         <img class="person-img" src="{{ person.photo.url }}"
              alt="{{ person.first_name }} {{ person.last_name }}"/>
@@ -78,7 +78,7 @@
           </td>
           <td colspan="3">{{ person.get_sex_display }}</td>
         </tr>
-        {% has_perm 'core.view_address' user person as can_view_address %}
+        {% has_perm 'core.view_address_rule' user person as can_view_address %}
         {% if can_view_address %}
           <tr>
             <td>
@@ -88,7 +88,7 @@
             <td colspan="2">{{ person.postal_code }} {{ person.place }}</td>
           </tr>
         {% endif %}
-        {% has_perm 'core.view_contact_details' user person as can_view_contact_details %}
+        {% has_perm 'core.view_contact_details_rule' user person as can_view_contact_details %}
         {% if can_view_contact_details %}
           <tr>
             <td>
@@ -104,20 +104,21 @@
             <td colspan="3">{{ person.email }}</td>
           </tr>
         {% endif %}
-        {% has_perm 'core.view_personal_details' user person as can_view_personal_details %}
+        {% has_perm 'core.view_personal_details_rule' user person as can_view_personal_details %}
         {% if can_view_personal_details %}
           <tr>
             <td>
               <i class="material-icons small">cake</i>
             </td>
-            <td colspan="3">{{ person.date_of_birth|date }}</td>
+            <td colspan="2">{{ person.date_of_birth|date }}</td>
+            <td colspan="2">{{ person.place_of_birth }}</td>
           </tr>
         {% endif %}
       </table>
     </div>
     {% if person.description %}
       <div class="col s12 m12">
-        <h5>{% trans "Description" %}</h5>
+        <h2>{% trans "Description" %}</h2>
         <p>
           {{ person.description }}
         </p>
@@ -127,21 +128,21 @@
 
   {% if person.children.all and can_view_personal_details %}
     <div class="col s12 m12">
-      <h5>{% trans "Children" %}</h5>
+      <h2>{% trans "Children" %}</h2>
       {% include "core/person/collection.html" with persons=person.children.all %}
     </div>
   {% endif %}
 
   {% if person.guardians.all and can_view_personal_details %}
     <div class="col s12 m12">
-      <h5>{% trans "Guardians / Parents" %}</h5>
+      <h2>{% trans "Guardians / Parents" %}</h2>
       {% include "core/person/collection.html" with persons=person.guardians.all %}
     </div>
   {% endif %}
 
-  {% has_perm 'core.view_person_groups' user person as can_view_groups %}
+  {% has_perm 'core.view_person_groups_rule' user person as can_view_groups %}
   {% if can_view_groups %}
-    <h5>{% blocktrans %}Groups{% endblocktrans %}</h5>
+    <h2>{% blocktrans %}Groups{% endblocktrans %}</h2>
     {% render_table groups_table %}
   {% endif %}
 {% endblock %}
diff --git a/aleksis/core/templates/core/person/list.html b/aleksis/core/templates/core/person/list.html
index b48baf3e5ff8b0f229ab13dfb7c6b5a5885d105f..ca441cac485847cf6a4975071d73787473804694 100644
--- a/aleksis/core/templates/core/person/list.html
+++ b/aleksis/core/templates/core/person/list.html
@@ -9,7 +9,7 @@
 {% block page_title %}{% blocktrans %}Persons{% endblocktrans %}{% endblock %}
 
 {% block content %}
-  {% has_perm 'core.create_person' user person as can_create_person %}
+  {% has_perm 'core.create_person_rule' user person as can_create_person %}
 
   {% if can_create_person %}
     <a class="btn green waves-effect waves-light" href="{% url 'create_person' %}">
@@ -18,7 +18,7 @@
     </a>
   {% endif %}
 
-  <h5>{% trans "Filter persons" %}</h5>
+  <h2>{% trans "Filter persons" %}</h2>
   <form method="get">
     {% form form=persons_filter.form %}{% endform %}
     {% trans "Search" as caption %}
@@ -29,6 +29,6 @@
     </button>
   </form>
 
-  <h5>{% trans "Selected persons" %}</h5>
+  <h2>{% trans "Selected persons" %}</h2>
   {% render_table persons_table %}
 {% endblock %}
diff --git a/aleksis/core/templates/impersonate/list_users.html b/aleksis/core/templates/impersonate/list_users.html
index aff2f8660ac374cebb1bbeff646bc038aee9e5f1..fd273ca60902e5defbc41e567a9205cfe99fb349 100644
--- a/aleksis/core/templates/impersonate/list_users.html
+++ b/aleksis/core/templates/impersonate/list_users.html
@@ -4,9 +4,8 @@
 {% extends "core/base.html" %}
 {% load i18n %}
 
-{% block page_title %}
-  {% blocktrans %}Impersonate user{% endblocktrans %}
-{% endblock %}
+{% block browser_title %}{% trans "Impersonate" %}{% endblock %}
+{% block page_title %}{% trans "Impersonate" %}{% endblock %}
 
 {% block content %}
   {% if page.object_list %}
diff --git a/aleksis/core/templates/material/fields/ckeditor_ckeditorwidget.html b/aleksis/core/templates/material/fields/ckeditor_ckeditorwidget.html
new file mode 100644
index 0000000000000000000000000000000000000000..73f57170017ed5e21a4c495f6f683dfb5b957ef4
--- /dev/null
+++ b/aleksis/core/templates/material/fields/ckeditor_ckeditorwidget.html
@@ -0,0 +1,28 @@
+{% load material_form material_form_internal %}
+{% part bound_field.field %}<{{ field.widget.component|default:'dmc-textarea' }}>
+  <div class="row">
+    <div{% attrs bound_field 'group' %}
+      id="id_{{ bound_field.html_name }}_container"
+      class="input-field col s12{% if field.required %} required{% endif %}{% if bound_field.errors %} has-error{% endif %}"
+    {% endattrs %}>
+      {% part field label %}
+        <label{% attrs bound_field 'label' %}
+          for="{{ bound_field.id_for_label }}"
+          {% if bound_field.value %}class="active"{% endif %}
+        {% endattrs %}>{{ bound_field.label }}</label>
+      {% endpart %}
+      <div class="margin-top-35">
+      {% part field prefix %}{% endpart %}{% part field control %}
+        {{ bound_field }}
+      {% endpart %}
+      </div>
+      {% part field help_text %}{% if field.help_text %}
+        <div class="help-block">{{ bound_field.help_text|safe }}</div>
+      {% endif %}
+      {% endpart %}{% part field errors %}
+        {% if bound_field.errors %}
+          {% include  'material/field_errors.html' %}
+        {% endif %}
+      {% endpart %}{{ hidden_initial }}
+    </div>
+  </div></{{ field.widget.component|default:'dmc-textarea' }}>{% endpart %}
diff --git a/aleksis/core/templates/oauth2_provider/application/create.html b/aleksis/core/templates/oauth2_provider/application/create.html
new file mode 100644
index 0000000000000000000000000000000000000000..d81489e922a76de8d7a8e92a7f48686714a3b3a7
--- /dev/null
+++ b/aleksis/core/templates/oauth2_provider/application/create.html
@@ -0,0 +1,17 @@
+{% extends "core/base.html" %}
+
+{% load i18n material_form %}
+
+{% block browser_title %}{% blocktrans %}Register OAuth2 Application{% endblocktrans %}{% endblock %}
+{% block page_title %}{% blocktrans %}Register OAuth2 Application{% endblocktrans %}{% endblock %}
+
+{% block content %}
+  <form method="post">
+    {% csrf_token %}
+    {% form form=form %}{% endform %}
+    {% include "core/partials/save_button.html" %}
+    <a class="btn waves-effect red waves-light" href="{% url "oauth2_applications" %}">
+      <i class="material-icons left">clear</i> {% trans "Cancel" %}
+    </a>
+  </form>
+{% endblock %}
diff --git a/aleksis/core/templates/oauth2_provider/application/detail.html b/aleksis/core/templates/oauth2_provider/application/detail.html
new file mode 100644
index 0000000000000000000000000000000000000000..da6b8abf81ddc1916de7b0c7feb5a7262f453037
--- /dev/null
+++ b/aleksis/core/templates/oauth2_provider/application/detail.html
@@ -0,0 +1,75 @@
+{% extends "core/base.html" %}
+
+{% load i18n %}
+
+{% block browser_title %}{% blocktrans %}OAuth2 Application{% endblocktrans %}{% endblock %}
+{% block page_title %}
+  <a href="{% url "oauth2_applications" %}"
+     class="btn-flat primary-color-text waves-light waves-effect">
+    <i class="material-icons left">chevron_left</i> {% trans "Back" %}
+  </a>
+  {{ application.name }}
+{% endblock %}
+
+{% block content %}
+  <a class="btn orange waves-effect waves-light btn-margin" href="{% url "edit_oauth2_application" application.id %}">
+    <i class="material-icons left">edit</i>
+    {% trans "Edit" %}
+  </a>
+  <a class="btn red waves-effect waves-light btn-margin" href="{% url "delete_oauth2_application" application.id %}">
+    <i class="material-icons left">delete</i>
+    {% trans "Delete" %}
+  </a>
+  <table class="responsive-table">
+    <tbody>
+    <tr>
+      <th>
+        {% trans "Client id" %}
+      </th>
+      <td>
+        <code class="break-word">{{ application.client_id }}</code>
+      </td>
+    </tr>
+    <tr>
+      <th>
+        {% trans "Client secret" %}
+      </th>
+      <td>
+        <code class="break-word">{{ application.client_secret }}</code>
+      </td>
+    </tr>
+    <tr>
+      <th>
+        {% trans "Client type" %}
+      </th>
+      <td>
+        {{ application.client_type }}
+      </td>
+    </tr>
+    <tr>
+      <th>
+        {% trans "Allowed scopes" %}
+      </th>
+      <td>
+        {{ application.allowed_scopes|join:", " }}
+      </td>
+    </tr>
+    <tr>
+      <th>
+        {% trans "Redirect URIs" %}
+      </th>
+      <td>
+        {{ application.redirect_uris }}
+      </td>
+    </tr>
+    <tr>
+      <th>
+        {% trans "Skip Authorisation" %}
+      </th>
+      <td>
+        <i class="material-icons">{{ application.skip_authorization|yesno:"check,close" }}</i>
+      </td>
+    </tr>
+    </tbody>
+  </table>
+{% endblock %}
diff --git a/aleksis/core/templates/oauth2_provider/application/edit.html b/aleksis/core/templates/oauth2_provider/application/edit.html
new file mode 100644
index 0000000000000000000000000000000000000000..6755d2420fb6a181f671b825475afb4ec9581521
--- /dev/null
+++ b/aleksis/core/templates/oauth2_provider/application/edit.html
@@ -0,0 +1,17 @@
+{% extends "core/base.html" %}
+
+{% load i18n material_form %}
+
+{% block browser_title %}{% blocktrans %}Edit OAuth2 Application{% endblocktrans %}{% endblock %}
+{% block page_title %}{% blocktrans %}Edit OAuth2 Application{% endblocktrans %}{% endblock %}
+
+{% block content %}
+  <form method="post">
+    {% csrf_token %}
+    {% form form=form %}{% endform %}
+    {% include "core/partials/save_button.html" %}
+    <a class="btn waves-effect red waves-light" href="{% url "oauth2_application" application.id %}">
+      <i class="material-icons left">clear</i> {% trans "Cancel" %}
+    </a>
+  </form>
+{% endblock %}
diff --git a/aleksis/core/templates/oauth2_provider/application/list.html b/aleksis/core/templates/oauth2_provider/application/list.html
new file mode 100644
index 0000000000000000000000000000000000000000..06f1a95c4e05c14dcb5efe69f2416040d184f0bb
--- /dev/null
+++ b/aleksis/core/templates/oauth2_provider/application/list.html
@@ -0,0 +1,24 @@
+{% extends "core/base.html" %}
+
+{% load i18n %}
+
+{% block browser_title %}{% blocktrans %}OAuth2 Applications{% endblocktrans %}{% endblock %}
+{% block page_title %}{% blocktrans %}OAuth2 Applications{% endblocktrans %}{% endblock %}
+
+{% block content %}
+  <a href="{% url "register_oauth_application" %}" class="btn green waves-effect waves-light">
+    <i class="material-icons left">add</i>
+    {% blocktrans %}Register new application{% endblocktrans %}
+  </a>
+  <div class="collection">
+    {% for application in applications %}
+      <a class="collection-item" href="{% url "oauth2_application" application.id %}">
+        {{ application.name }}
+      </a>
+      {% empty %}
+      <div class="collection-item flow-text">
+        {% blocktrans %}No applications defined.{% endblocktrans %}
+      </div>
+    {% endfor %}
+  </div>
+{% endblock %}
diff --git a/aleksis/core/templates/oauth2_provider/authorize.html b/aleksis/core/templates/oauth2_provider/authorize.html
new file mode 100644
index 0000000000000000000000000000000000000000..48b996837b8721ef6ba74337d286236e91729f5b
--- /dev/null
+++ b/aleksis/core/templates/oauth2_provider/authorize.html
@@ -0,0 +1,59 @@
+{% extends "core/base.html" %}
+
+{% load i18n material_form %}
+
+{% block browser_title %}{% blocktrans %}Authorize{% endblocktrans %}{% endblock %}
+
+{% block content %}
+
+  <div class="row">
+  {% if not error %}
+    <div class="col m1 l2 xl3"></div>
+    <div class="col s12 m10 l8 xl6">
+      <div class="card">
+        <div class="card-content">
+          <div class="card-title">
+            {% trans "Authorize" %} {{ application.name }}
+          </div>
+          <p class="margin-bottom">{% trans "The application requests access to the following scopes:" %}</p>
+          {% for scope in scopes_descriptions %}
+            <p class="margin-bottom">
+              <i class="material-icons left">check</i>
+              {{ scope }}
+            </p>
+          {% endfor %}
+        </div>
+        <div class="card-action-light">
+          <form method="post">
+            {% csrf_token %}
+            {% form form=form %}
+              {% part form.allow %}<input type="hidden" value="true" name="allow">{% endpart %}
+            {% endform %}
+            <button type="submit" class="btn green waves-effect waves-light btn-margin">
+              <i class="material-icons left">done_all</i> {% trans "Allow" %}
+            </button>
+            <a class="btn red waves-effect waves-light btn-margin" href="{% block app-form-back-url %}{% url "oauth2_application" application.id %}{% endblock app-form-back-url %}">
+              <i class="material-icons left">cancel</i> {% trans "Disallow" %}
+            </a>
+          </form>
+        </div>
+      </div>
+    </div>
+  {% else %}
+    <div class="container">
+      <div class="card red">
+        <div class="card-content white-text">
+          <div class="material-icons small left">error_outline</div>
+          <span class="card-title">{% trans "Error" %}: {{ error.error }}</span>
+          <p>
+            {{ error.description }}
+          </p>
+          <p>
+            Please verify if the application is configured correctly or contact one of your site administrators:
+          </p>
+          {% include "core/partials/admins_list.html" %}
+        </div>
+      </div>
+    </div>
+  {% endif %}
+{% endblock %}
diff --git a/aleksis/core/templates/oauth2_provider/authorized-oob.html b/aleksis/core/templates/oauth2_provider/authorized-oob.html
new file mode 100644
index 0000000000000000000000000000000000000000..3e95b0bf3a25691e0fc84e36dbcf71559cf86dc1
--- /dev/null
+++ b/aleksis/core/templates/oauth2_provider/authorized-oob.html
@@ -0,0 +1,36 @@
+{% extends "core/base.html" %}
+
+{% load i18n %}
+
+{% block content %}
+
+  {% if not error %}
+    <div class="container">
+      <div class="card green">
+        <div class="card-content white-text">
+          <div class="material-icons small left">check</div>
+          <span class="card-title">{% blocktrans %}Success!{% endblocktrans %}</span>
+          <p>
+            {% trans "Please return to your application and enter this code:" %} {{ code }}
+          </p>
+        </div>
+      </div>
+    </div>
+  {% else %}
+    <div class="container">
+      <div class="card red">
+        <div class="card-content white-text">
+          <div class="material-icons small left">error_outline</div>
+          <span class="card-title">{% trans "Error" %}: {{ error.error }}</span>
+          <p>
+            {{ error.description }}
+          </p>
+          <p>
+            Please verify if the application is configured correctly or contact one of your site administrators:
+          </p>
+          {% include "core/partials/admins_list.html" %}
+        </div>
+      </div>
+    </div>
+  {% endif %}
+{% endblock %}
diff --git a/aleksis/core/templates/oauth2_provider/authorized-token-delete.html b/aleksis/core/templates/oauth2_provider/authorized-token-delete.html
new file mode 100644
index 0000000000000000000000000000000000000000..7f6f25920bebcbbaad7bb93fd486db617f25d662
--- /dev/null
+++ b/aleksis/core/templates/oauth2_provider/authorized-token-delete.html
@@ -0,0 +1,27 @@
+{% extends "core/base.html" %}
+
+{% load i18n %}
+
+{% block browser_title %}{% trans "Revoke access" %}{% endblock %}
+{% block page_title %}{% trans "Revoke access" %}{% endblock %}
+
+{% block content %}
+  <div class="alert info">
+    <p>
+      <i class="material-icons left">warning</i>
+      {% trans "Are you sure to revoke the access for this application?" %}
+    </p>
+  </div>
+
+  <form method="post">
+    {% csrf_token %}
+    <a class="btn waves-effect waves-light red" href="{% url "oauth2_applications" %}">
+      <i class="material-icons left">delete</i>
+      {% trans "Revoke" %}
+    </a>
+    <a class="btn waves-effect waves-light" href="{% url "oauth2_applications" %}">
+      <i class="material-icons left">cancel</i>
+      {% trans "Cancel" %}
+    </a>
+  </form>
+{% endblock %}
diff --git a/aleksis/core/templates/oauth2_provider/authorized-tokens.html b/aleksis/core/templates/oauth2_provider/authorized-tokens.html
new file mode 100644
index 0000000000000000000000000000000000000000..46b0ce570a5ec1077825b283a244b6652045c437
--- /dev/null
+++ b/aleksis/core/templates/oauth2_provider/authorized-tokens.html
@@ -0,0 +1,37 @@
+{% extends "core/base.html" %}
+
+{% load i18n %}
+
+{% block browser_title %}{% blocktrans %}Authorized applications{% endblocktrans %}{% endblock %}
+{% block page_title %}{% trans "Authorized applications" %}{% endblock %}
+
+{% block content %}
+  {% if authorized_tokens %}
+    <div class="row">
+      {% for authorized_token in authorized_tokens %}
+        <div class="col s12 m6 l4 xl3">
+          <div class="card">
+            <div class="card-content">
+              <div class="card-title">{{ authorized_token.application }}</div>
+              {% for scope_name, scope_description in authorized_token.scopes.items %}
+                <p>
+                  {{ scope_name }}: {{ scope_description }}
+                </p>
+              {% endfor %}
+            </div>
+            <div class="card-action">
+              <a href="{% url 'oauth2_provider:authorized-token-delete' authorized_token.pk %}">{% trans "Revoke access" %}</a>
+            </div>
+          </div>
+        </div>
+      {% endfor %}
+    </div>
+  {% else %}
+    <div class="alert info">
+      <p>
+        <i class="material-icons left">info</i>
+        {% trans "No authorized applications." %}
+      </p>
+    </div>
+  {% endif %}
+{% endblock %}
diff --git a/aleksis/core/templates/search/search.html b/aleksis/core/templates/search/search.html
index caff1cc6f62f30c67dcb0e2fe082dfa08db389a8..5af1e55cc1076e404a921596da1cbc0774f7dbc3 100644
--- a/aleksis/core/templates/search/search.html
+++ b/aleksis/core/templates/search/search.html
@@ -23,11 +23,11 @@
       </button>
     </p>
 
-    <h5>{% trans "Results" %}</h5>
+    <h2>{% trans "Results" %}</h2>
 
     {% if query %}
       <div class="collection">
-        {% for result in page.object_list %}
+        {% for result in page_obj.object_list %}
           <a href="{{ result.object.get_absolute_url|default:"#" }}" class="collection-item">
             <i class="material-icons left">{{ result.object.icon_ }}</i>
             {{ result.object }}
@@ -41,11 +41,11 @@
         {% endfor %}
       </div>
 
-      {% if page.has_other_pages %}
+      {% if page_obj.has_other_pages %}
         <ul class="pagination">
-          {% if page.has_previous %}
+          {% if page_obj.has_previous %}
             <li class="waves-effect">
-              <a href="?q={{ query }}&amp;page={{ page.previous_page_number }}">
+              <a href="?q={{ query }}&amp;page={{ page_obj.previous_page_number }}">
                 <i class="material-icons">chevron_left</i>
               </a>
             </li>
@@ -55,8 +55,8 @@
             </li>
           {% endif %}
 
-          {% for page_num in page.paginator.page_range %}
-            {% if page.number == page_num %}
+          {% for page_num in page_obj.paginator.page_range %}
+            {% if page_obj.number == page_num %}
               <li class="active">
                 <a href="#">{{ page_num }}</a>
               </li>
@@ -67,9 +67,9 @@
             {% endif %}
           {% endfor %}
 
-          {% if page.has_next %}
+          {% if page_obj.has_next %}
             <li class="waves-effect">
-              <a href="?q={{ query }}&amp;page={{ page.next_page_number }}">
+              <a href="?q={{ query }}&amp;page={{ page_obj.next_page_number }}">
                 <i class="material-icons">chevron_right</i>
               </a>
             </li>
diff --git a/aleksis/core/templates/socialaccount/authentication_error.html b/aleksis/core/templates/socialaccount/authentication_error.html
new file mode 100644
index 0000000000000000000000000000000000000000..00901daf7fc3542ba07490381f7903e8f6d14048
--- /dev/null
+++ b/aleksis/core/templates/socialaccount/authentication_error.html
@@ -0,0 +1,24 @@
+{% extends "core/base.html" %}
+
+{% load i18n material_form %}
+
+{% block browser_title %}{% trans "Third-party Account Login Failure" %}{% endblock %}
+{% block page_title %}{% trans "Third-party Account Login Failure" %}{% endblock %}
+
+{% block content %}
+  <div class="container">
+    <div class="card red">
+      <div class="card-content white-text">
+        <div class="material-icons small left">error_outline</div>
+        <span class="card-title">{% blocktrans %}Third-party Account Login Failure.{% endblocktrans %}</span>
+        <p>
+          {% blocktrans %}
+            An error occurred while attempting to login via your third-party account.
+            Please contact one of your site administrators.
+          {% endblocktrans %}
+        </p>
+        {% include "core/partials/admins_list.html" %}
+      </div>
+    </div>
+  </div>
+{% endblock %}
diff --git a/aleksis/core/templates/socialaccount/connections.html b/aleksis/core/templates/socialaccount/connections.html
new file mode 100644
index 0000000000000000000000000000000000000000..2339be7e041609f85311ce152c4083a05ec4e651
--- /dev/null
+++ b/aleksis/core/templates/socialaccount/connections.html
@@ -0,0 +1,43 @@
+{% extends "core/base.html" %}
+
+{% load i18n material_form %}
+
+{% block browser_title %}{% trans "Connections" %}{% endblock %}
+{% block page_title %}{% trans "Connections" %}{% endblock %}
+
+{% block content %}
+  {% if form.accounts %}
+    {% csrf_token %}
+    <div class="row">
+      {% for base_account in form.accounts %}
+      {% with base_account.get_provider_account as account %}
+        <div class="col s12 m6 l4 xl3">
+          <div class="card">
+            <div class="card-content">
+              <div class="card-title">{{ account.get_brand.name }}</div>
+                <p>
+                  {{ account }}
+                </p>
+            </div>
+            <div class="card-action">
+              <a href="{% url 'delete_social_account_by_pk' base_account.pk %}">
+                {% trans "Remove" %}
+              </a>
+            </div>
+          </div>
+        </div>
+      {% endwith %}
+      {% endfor %}
+    </div>
+
+  {% else %}
+    <p>{% trans 'You currently have no third-party accounts connected to this account.' %}</p>
+  {% endif %}
+
+  <h2>{% trans 'Add a Third-party Account' %}</h2>
+
+  {% include "socialaccount/snippets/provider_list.html" with process="connect" %}
+
+{% include "socialaccount/snippets/login_extra.html" %}
+
+{% endblock %}
diff --git a/aleksis/core/templates/socialaccount/login_cancelled.html b/aleksis/core/templates/socialaccount/login_cancelled.html
new file mode 100644
index 0000000000000000000000000000000000000000..bc1ebb6cb31a6b15a9d448cb93fa40a7cccb7d26
--- /dev/null
+++ b/aleksis/core/templates/socialaccount/login_cancelled.html
@@ -0,0 +1,22 @@
+{% extends "core/base.html" %}
+
+{% load i18n material_form %}
+
+{% block browser_title %}{% trans "Login cancelled" %}{% endblock %}
+{% block page_title %}{% trans "Login cancelled" %}{% endblock %}
+
+{% block content %}
+  <div class="container">
+    <div class="card red">
+      <div class="card-content white-text">
+        <div class="material-icons small left">error_outline</div>
+        <span class="card-title">{% blocktrans %}Login cancelled{% endblocktrans %}</span>
+        <p>
+          {% blocktrans %}
+            You decided to cancel logging in to our site using one of your existing accounts. If this was a mistake, please proceed to <a href="{{login_url}}">sign in</a>.
+          {% endblocktrans %}
+        </p>
+      </div>
+    </div>
+  </div>
+{% endblock %}
diff --git a/aleksis/core/templates/socialaccount/signup.html b/aleksis/core/templates/socialaccount/signup.html
new file mode 100644
index 0000000000000000000000000000000000000000..df79ecce459fa0ea7bcd8fe83ea21f530944c49b
--- /dev/null
+++ b/aleksis/core/templates/socialaccount/signup.html
@@ -0,0 +1,27 @@
+{% extends "core/base.html" %}
+
+{% load i18n material_form %}
+
+{% block browser_title %}{% trans "Signup" %}{% endblock %}
+{% block page_title %}{% trans "Signup" %}{% endblock %}
+
+{% block content %}
+  <div class="alert success">
+    <p>
+      <i class="material-icons left">check_circle_outline</i>
+        {% blocktrans with provider_name=account.get_provider.name site_name=site.name %}You are about to use your {{provider_name}} account to login to
+        {{site_name}}. As a final step, please complete the following form:{% endblocktrans %}
+    </p>
+  </div>
+
+  <form method="post" action="{% url 'account_signup' %}">
+    {% csrf_token %}
+    {% form form=form %}{% endform %}
+    {% if redirect_field_value %}
+      <input type="hidden" name="{{ redirect_field_name }}" value="{{ redirect_field_value }}" />
+    {% endif %}
+    {% trans "Sign up" as caption %}
+    {% include "core/partials/save_button.html" with caption=caption icon="how_to_reg" %}
+  </form>
+
+{% endblock %}
diff --git a/aleksis/core/templates/socialaccount/snippets/provider_list.html b/aleksis/core/templates/socialaccount/snippets/provider_list.html
new file mode 100644
index 0000000000000000000000000000000000000000..9d9a9c87712076d8954ebce192ea2ac38383cc48
--- /dev/null
+++ b/aleksis/core/templates/socialaccount/snippets/provider_list.html
@@ -0,0 +1,36 @@
+{% load i18n socialaccount %}
+<p>
+  {% get_providers as socialaccount_providers %}
+  {% if socialaccount_providers %}
+    {% for provider in socialaccount_providers %}
+      {% if provider.id == "openid" %}
+        {% for brand in provider.get_brands %}
+            <a title="{{brand.name}}" 
+              class="socialaccount_provider {{provider.id}} {{brand.id}}
+              btn-large waves-effect waves-light primary-color" 
+              href="{% provider_login_url provider.id openid=brand.openid_url process=process %}">
+              {% blocktrans with name=brand.name %}
+                Login with {{ name }}
+              {% endblocktrans %}
+            </a>
+        {% endfor %}
+      {% endif %}
+        <a title="{{provider.name}}" class="socialaccount_provider {{provider.id}}
+        btn hundred-percent waves-effect waves-light primary-color" 
+          href="{% provider_login_url provider.id process=process scope=scope auth_params=auth_params %}">
+          {% blocktrans with name=provider.name %}
+            Login with {{ name }}
+          {% endblocktrans %}
+        </a>
+    {% endfor %}
+  {% else %}
+    <div class="alert primary">
+      <div>
+        <i class="material-icons left">info</i>
+        {% blocktrans %}
+          No third-party account providers available.
+        {% endblocktrans %}
+      </div>
+    </div>
+  {% endif %}
+</p>
diff --git a/aleksis/core/templates/templated_email/person_changed.email b/aleksis/core/templates/templated_email/person_changed.email
index 2e1db653256873d3f72afcd34a1e19914d116480..1e0d8be25684486bfdceac0b979b461253c21df7 100644
--- a/aleksis/core/templates/templated_email/person_changed.email
+++ b/aleksis/core/templates/templated_email/person_changed.email
@@ -11,7 +11,7 @@
    the person {{ person }} recently changed the following fields:
  {% endblocktrans %}
 
- {% for field in send_notification_fields %}
+ {% for field in changed_fields %}
   * {{ field }}
  {% endfor %}
 {% endblock %}
@@ -25,7 +25,7 @@
  </p>
 
  <ul>
-  {% for field in send_notification_fields %}
+  {% for field in changed_fields %}
    <li>{{ field }}</li>
   {% endfor %}
  </ul>
diff --git a/aleksis/core/templates/two_factor/core/backup_tokens.html b/aleksis/core/templates/two_factor/core/backup_tokens.html
index 84d6ef6802d5d6e85459eb662fe1d99f65a0dcf9..b85450747fbe00623900af8d2832c0060446a042 100644
--- a/aleksis/core/templates/two_factor/core/backup_tokens.html
+++ b/aleksis/core/templates/two_factor/core/backup_tokens.html
@@ -6,7 +6,7 @@
 {% endblock %}
 
 {% block content %}
-  <h4>{% block title %}{% trans "Backup Tokens" %}{% endblock %}</h4>
+  <h1>{% block title %}{% trans "Backup Tokens" %}{% endblock %}</h1>
 
   <div class="alert info">
     <p>
diff --git a/aleksis/core/templates/two_factor/core/login.html b/aleksis/core/templates/two_factor/core/login.html
index 0e62194597024dbb613d2c13669513c33dcb9d05..9ea4bc6c24b7d9e764742bbddf75ceb86b785868 100644
--- a/aleksis/core/templates/two_factor/core/login.html
+++ b/aleksis/core/templates/two_factor/core/login.html
@@ -1,97 +1,126 @@
 {# -*- engine:django -*- #}
 {% extends "two_factor/_base_focus.html" %}
-{% load i18n two_factor %}
+{% load i18n two_factor account socialaccount %}
 
 {% block browser_title %}
   {% trans "Login" %}
 {% endblock %}
 
 {% block content %}
-  <h4>{% trans "Login" %}</h4>
-
-  {% if wizard.steps.current == "auth" and user.is_authenticated %}
-    <div class="alert warning">
-      <p>
-        <i class="material-icons left">warning</i>
-        {% blocktrans %}You have no permission to view this page. Please login with an other account.{% endblocktrans %}
-      </p>
-    </div>
-  {% elif wizard.steps.current == 'auth' %}
-    <div class="alert primary">
-      <p>
-        <i class="material-icons left">info</i>
-
-        {% blocktrans %}Please login to see this page.{% endblocktrans %}
-      </p>
-    </div>
-  {% endif %}
-
-  {% if wizard.steps.current == 'auth' and ALTERNATIVE_LOGIN_VIEWS %}
-    <h5>{% trans "Login with username and password" %}</h5>
-  {% endif %}
-
-  {% if not  wizard.steps.current == "auth" %}
-    <div class="alert primary">
-      <p>
-        <i class="material-icons left">info</i>
-
-        {% if wizard.steps.current == 'token' %}
-          {% if device.method == 'call' %}
-            {% blocktrans %}We are calling your phone right now, please enter the
-              digits you hear.{% endblocktrans %}
-          {% elif device.method == 'sms' %}
-            {% blocktrans %}We sent you a text message, please enter the tokens we
-              sent.{% endblocktrans %}
-          {% else %}
-            {% blocktrans %}Please enter the tokens generated by your token
-              generator.{% endblocktrans %}
-          {% endif %}
-        {% elif wizard.steps.current == 'backup' %}
-          {% blocktrans %}Use this form for entering backup tokens for logging in.
-            These tokens have been generated for you to print and keep safe. Please
-            enter one of these backup tokens to login to your account.{% endblocktrans %}
-        {% endif %}
-    </div>
-  {% endif %}
-
+  {% get_providers as socialaccount_providers %}
 
   <form action="" method="post">
     {% csrf_token %}
-    {% include "two_factor/_wizard_forms.html" %}
-
-    {# hidden submit button to enable [enter] key #}
-    <div style="margin-left: -9999px">
-      <input type="submit" value=""/>
-    </div>
-
-    {% if other_devices %}
-      <p>{% trans "Or, alternatively, use one of your backup phones:" %}</p>
-      <p>
-        {% for other in other_devices %}
-          <button name="challenge_device" value="{{ other.persistent_id }}"
-                  class="btn" type="submit">
-            {{ other|device_action }}
-          </button>
-        {% endfor %}</p>
-    {% endif %}
-    {% if backup_tokens %}
-      <p>{% trans "As a last resort, you can use a backup token:" %}</p>
-      <p>
-        <button name="wizard_goto_step" type="submit" value="backup"
-                class="btn">{% trans "Use Backup Token" %}</button>
-      </p>
-    {% endif %}
+    <div class="row">
+      <div class="col m1 l2 xl3"></div>
+      <div class="col s12 m10 l8 xl6">
+        <div class="card">
+          <div class="card-content">
+            {% if wizard.steps.current == 'auth' and socialaccount_providers %}
+              <div class="card-title">{% trans "Login with username and password" %}</div>
+            {% else %}
+              <div class="card-title">{% trans "Login" %}</div>
+            {% endif %}
+            {% if wizard.steps.current == "auth" and user.is_authenticated %}
+              <div class="alert warning">
+                <p>
+                  <i class="material-icons left">warning</i>
+                  {% blocktrans %}You have no permission to view this page. Please login with an other
+                    account.{% endblocktrans %}
+                </p>
+              </div>
+            {% elif wizard.steps.current == 'auth' %}
+              <div class="alert primary">
+                <p>
+                  <i class="material-icons left">info</i>
+                  {% blocktrans %}Please login to see this page.{% endblocktrans %}
+                </p>
+              </div>
+            {% endif %}
+            {% if not wizard.steps.current == "auth" %}
+              <div class="alert primary">
+                <p>
+                  <i class="material-icons left">info</i>
+                  {% if wizard.steps.current == 'token' %}
+                    {% if device.method == 'call' %}
+                      {% blocktrans %}
+                        We are calling your phone right now, please enter the
+                        digits you hear.
+                      {% endblocktrans %}
+                    {% elif device.method == 'sms' %}
+                      {% blocktrans %}
+                        We sent you a text message, please enter the tokens we
+                        sent.
+                      {% endblocktrans %}
+                    {% else %}
+                      {% blocktrans %}
+                        Please enter the tokens generated by your token
+                        generator.
+                      {% endblocktrans %}
+                    {% endif %}
+                  {% elif wizard.steps.current == 'backup' %}
+                    {% blocktrans %}
+                      Use this form for entering backup tokens for logging in.
+                      These tokens have been generated for you to print and keep safe. Please
+                      enter one of these backup tokens to login to your account.
+                    {% endblocktrans %}
+                  {% endif %}
+                </p>
+              </div>
+            {% endif %}
 
-    <button type="submit" class="btn green waves-effect waves-light">
-      {% trans "Login" %}
-      <i class="material-icons right">send</i>
-    </button>
+            {% include "two_factor/_wizard_forms.html" %}
+          </div>
+          <div class="card-action-light">
+            <button type="submit" class="btn green waves-effect waves-light">
+              {% trans "Login" %}
+              <i class="material-icons right">send</i>
+            </button>
+            {% if request.site.preferences.auth__allow_password_change and wizard.steps.current == "auth" %}
+              <a href="{% url "account_reset_password" %}" class="btn-flat right waves-effect waves-light">
+                {% trans "Reset password" %}
+              </a>
+            {% endif %}
+          </div>
+        </div>
 
+        {% if other_devices or backup_tokens %}
+          <div class="card">
+            <div class="card-content">
+              <div class="card-title">{% trans "Device currently not available?" %}</div>
+              {% if other_devices %}
+                <p>{% trans "Or, alternatively, use one of your backup phones:" %}</p>
+                <p>
+                  {% for other in other_devices %}
+                    <button name="challenge_device" value="{{ other.persistent_id }}" class="btn" type="submit">
+                      {{ other|device_action }}
+                    </button>
+                  {% endfor %}
+                </p>
+              {% endif %}
+              {% if backup_tokens %}
+                <p>{% trans "As a last resort, you can use a backup token:" %}</p>
+                <p>
+                  <button name="wizard_goto_step" type="submit" value="backup" class="btn">
+                    {% trans "Use Backup Token" %}
+                  </button>
+                </p>
+              {% endif %}
+            </div>
+          </div>
+        {% endif %}
+        {% if wizard.steps.current == 'auth' and socialaccount_providers %}
+          <div class="card">
+            <div class="card-content">
+              <div class="card-title">
+                {% trans "Use alternative login options" %}
+              </div>
+              {% include "socialaccount/snippets/provider_list.html" with process="login" %}
+            </div>
+          </div>
+        {% endif %}
+      </div>
+    </div>
   </form>
 
-  {% if wizard.steps.current == 'auth' and ALTERNATIVE_LOGIN_VIEWS %}
-    <h5>{% trans "Use alternative login options" %}</h5>
-    {% include "core/partials/alternative_login_options.html" %}
-  {% endif %}
-
 {% endblock %}
diff --git a/aleksis/core/templates/two_factor/core/phone_register.html b/aleksis/core/templates/two_factor/core/phone_register.html
index 4eae9ebfac173a6b4fc37ed3a21b5941cc28eb71..620cdd07b4310116406fd9e5ba2a15ecad4d1260 100644
--- a/aleksis/core/templates/two_factor/core/phone_register.html
+++ b/aleksis/core/templates/two_factor/core/phone_register.html
@@ -6,7 +6,7 @@
 {% endblock %}
 
 {% block content %}
-  <h4>{% block title %}{% trans "Add Backup Phone" %}{% endblock %}</h4>
+  <h1>{% block title %}{% trans "Add Backup Phone" %}{% endblock %}</h1>
 
   {% if wizard.steps.current == 'setup' %}
     <p>{% blocktrans %}You'll be adding a backup phone number to your
diff --git a/aleksis/core/templates/two_factor/core/setup.html b/aleksis/core/templates/two_factor/core/setup.html
index 8048403964a5bd4b7c9761d0189f18bb53524a22..a0e30472db11566a58b1c00e06b5ad6689b6e681 100644
--- a/aleksis/core/templates/two_factor/core/setup.html
+++ b/aleksis/core/templates/two_factor/core/setup.html
@@ -2,7 +2,7 @@
 {% load i18n %}
 
 {% block content %}
-  <h4>{% block title %}{% trans "Enable Two-Factor Authentication" %}{% endblock %}</h4>
+  <h1>{% block title %}{% trans "Enable Two-Factor Authentication" %}{% endblock %}</h1>
 
   {% if wizard.steps.current == 'welcome' %}
     <p class="flow-text">
diff --git a/aleksis/core/templates/two_factor/core/setup_complete.html b/aleksis/core/templates/two_factor/core/setup_complete.html
index 6520694bb0fbeda769573fa86e17bae89b514cf3..df3f5e93acb1ded3138b9aea2375ce7d529c9033 100644
--- a/aleksis/core/templates/two_factor/core/setup_complete.html
+++ b/aleksis/core/templates/two_factor/core/setup_complete.html
@@ -6,7 +6,7 @@
 {% endblock %}
 
 {% block content %}
-  <h4>{% block title %}{% trans "Two-Factor Authentication successfully enabled" %}{% endblock %}</h4>
+  <h1>{% block title %}{% trans "Two-Factor Authentication successfully enabled" %}{% endblock %}</h1>
 
   <div class="alert success">
     <p>
diff --git a/aleksis/core/templates/two_factor/profile/disable.html b/aleksis/core/templates/two_factor/profile/disable.html
index 36fa472b7550a11c6b0d2d0d17364e11820bb5ec..56bf20164b16b75a2b1d7e0759c9d43ba8d5b242 100644
--- a/aleksis/core/templates/two_factor/profile/disable.html
+++ b/aleksis/core/templates/two_factor/profile/disable.html
@@ -6,7 +6,7 @@
 {% endblock %}
 
 {% block content %}
-  <h4>{% block title %}{% trans "Disable Two-Factor Authentication" %}{% endblock %}</h4>
+  <h1>{% block title %}{% trans "Disable Two-Factor Authentication" %}{% endblock %}</h1>
 
   <p class="flow-text">
     {% blocktrans trimmed %}
diff --git a/aleksis/core/templates/two_factor/profile/profile.html b/aleksis/core/templates/two_factor/profile/profile.html
index 4161becad1bb418d62115634edd07a128afecf22..eaaa0ff6a5a1bde9463682100dfb9cc6f6b0f3dc 100644
--- a/aleksis/core/templates/two_factor/profile/profile.html
+++ b/aleksis/core/templates/two_factor/profile/profile.html
@@ -6,9 +6,9 @@
 {% endblock %}
 
 {% block content %}
-  <h4>
+  <h1>
     {% block title %}{% trans "Account Security" %}{% endblock %}
-  </h4>
+  </h1>
 
   {% if default_device %}
     {% if default_device_type == 'TOTPDevice' %}
@@ -20,7 +20,7 @@
     {% endif %}
 
     {% if available_phone_methods %}
-      <h5>{% trans "Backup Phone Numbers" %}</h5>
+      <h2>{% trans "Backup Phone Numbers" %}</h2>
       <p>{% blocktrans %}If your primary method is not available, we are able to
         send backup tokens to the phone numbers listed below.{% endblocktrans %}</p>
       <ul class="collection">
@@ -43,7 +43,7 @@
       </p>
     {% endif %}
 
-    <h5>{% trans "Backup Tokens" %}</h5>
+    <h2>{% trans "Backup Tokens" %}</h2>
     <p>
       {% blocktrans %}If you don't have any device with you, you can access
         your account using backup tokens.{% endblocktrans %}
@@ -60,7 +60,7 @@
       </a>
     </p>
 
-    <h5>{% trans "Disable Two-Factor Authentication" %}</h5>
+    <h2>{% trans "Disable Two-Factor Authentication" %}</h2>
     <p>
       {% blocktrans %}
         However we strongly discourage you to do so, you can
diff --git a/aleksis/core/templatetags/dashboard.py b/aleksis/core/templatetags/dashboard.py
index 4551cf5b065c49c0f26bc3520a8ad1d6a5626bd7..097997d1f03adb0bcd6487b85729a777d6fcc437 100644
--- a/aleksis/core/templatetags/dashboard.py
+++ b/aleksis/core/templatetags/dashboard.py
@@ -8,6 +8,6 @@ def include_widget(context, widget) -> dict:
     """Render a template with context from a defined widget."""
     template = loader.get_template(widget.get_template())
     request = context["request"]
-    context = widget.get_context(request)
+    context = widget._get_context_safe(request)
 
     return template.render(context)
diff --git a/aleksis/core/templatetags/html_helpers.py b/aleksis/core/templatetags/html_helpers.py
index 16238fcd00b0a88cb8eb8d6e25fb190ec432e1a5..5f78b1b195f9030cc97d96a0488c14845d450741 100644
--- a/aleksis/core/templatetags/html_helpers.py
+++ b/aleksis/core/templatetags/html_helpers.py
@@ -9,7 +9,11 @@ register = template.Library()
 def add_class_to_el(value: str, arg: str) -> str:
     """Add a CSS class to every occurence of an element type.
 
-    Example: {{ mymodel.myhtmlfield|add_class_to_el:"ul,browser-default"
+    :Example:
+
+    .. code-block::
+
+        {{ mymodel.myhtmlfield|add_class_to_el:"ul,browser-default" }}
     """
     el, cls = arg.split(",")
     soup = BeautifulSoup(value, "html.parser")
diff --git a/aleksis/core/tests/browser/test_selenium.py b/aleksis/core/tests/browser/test_selenium.py
index 8afd4b1584e3b5e6f35d69b2df73625ad8da5fad..e14ad8b9f123cd612848af566790d7e649d47bcc 100644
--- a/aleksis/core/tests/browser/test_selenium.py
+++ b/aleksis/core/tests/browser/test_selenium.py
@@ -77,5 +77,5 @@ class SeleniumTests(SeleniumTestCase):
         self._login()
         self._create_person()
         self.selenium.get(self.live_server_url + reverse("test_pdf"))
-        el = WebDriverWait(self.selenium, 10).until(lambda d: ".pdf" in self.selenium.current_url)
+        el = WebDriverWait(self.selenium, 20).until(lambda d: ".pdf" in self.selenium.current_url)
         self._screenshot("pdf.png")
diff --git a/aleksis/core/tests/models/test_group_sync.py b/aleksis/core/tests/models/test_group_sync.py
new file mode 100644
index 0000000000000000000000000000000000000000..808f122897b6b030ed2d95519b228b245c9992bd
--- /dev/null
+++ b/aleksis/core/tests/models/test_group_sync.py
@@ -0,0 +1,62 @@
+from django.contrib.auth.models import Group as DjangoGroup
+from django.contrib.auth.models import User
+
+import pytest
+
+from aleksis.core.models import Group, Person
+
+pytestmark = pytest.mark.django_db
+
+
+def test_create():
+    Group.objects.create(name="Foo")
+
+    assert DjangoGroup.objects.filter(name="Foo").exists()
+
+
+def test_assign_members():
+    g = Group.objects.create(name="Foo")
+    dj_g = DjangoGroup.objects.get(name="Foo")
+
+    u = User.objects.create(username="janedoe")
+    p = Person.objects.create(first_name="Jane", last_name="Doe", user=u)
+
+    g.members.add(p)
+
+    assert u in dj_g.user_set.all()
+
+
+def test_assign_owners():
+    g = Group.objects.create(name="Foo")
+    dj_g = DjangoGroup.objects.get(name="Foo")
+
+    u = User.objects.create(username="janedoe")
+    p = Person.objects.create(first_name="Jane", last_name="Doe", user=u)
+
+    g.owners.add(p)
+
+    assert u in dj_g.user_set.all()
+
+
+def test_assign_member_of():
+    g = Group.objects.create(name="Foo")
+    dj_g = DjangoGroup.objects.get(name="Foo")
+
+    u = User.objects.create(username="janedoe")
+    p = Person.objects.create(first_name="Jane", last_name="Doe", user=u)
+
+    p.member_of.add(g)
+
+    assert u in dj_g.user_set.all()
+
+
+def test_assign_owner_of():
+    g = Group.objects.create(name="Foo")
+    dj_g = DjangoGroup.objects.get(name="Foo")
+
+    u = User.objects.create(username="janedoe")
+    p = Person.objects.create(first_name="Jane", last_name="Doe", user=u)
+
+    p.owner_of.add(g)
+
+    assert u in dj_g.user_set.all()
diff --git a/aleksis/core/tests/models/test_pdffile.py b/aleksis/core/tests/models/test_pdffile.py
index 778ff8b5676a93279f9f9261e9c27ccbe6d85939..1f1e936a8c72a6dca3c4796a18a69b9370adbb0a 100644
--- a/aleksis/core/tests/models/test_pdffile.py
+++ b/aleksis/core/tests/models/test_pdffile.py
@@ -3,6 +3,7 @@ import re
 from datetime import datetime, timedelta
 
 from django.core.files import File
+from django.core.files.base import ContentFile
 from django.core.files.storage import default_storage
 from django.template.loader import render_to_string
 from django.test import TransactionTestCase, override_settings
@@ -25,14 +26,13 @@ class PDFFIleTest(TransactionTestCase):
     _test_pdf = os.path.join(os.path.dirname(os.path.abspath(__file__)), "test.pdf")
 
     def _get_test_html(self):
-        return render_to_string("core/pages/test_pdf.html")
+        return ContentFile(render_to_string("core/pages/test_pdf.html"), name="source.html")
 
     def test_pdf_file(self):
         dummy_person = Person.objects.create(first_name="Jane", last_name="Doe")
 
         html = self._get_test_html()
-        assert "html" in html
-        file_object = PDFFile.objects.create(person=dummy_person, html=html)
+        file_object = PDFFile.objects.create(person=dummy_person, html_file=html)
         assert isinstance(file_object.expires_at, datetime)
         assert file_object.expires_at > timezone.now()
         assert not bool(file_object.file)
@@ -40,12 +40,12 @@ class PDFFIleTest(TransactionTestCase):
         with open(self._test_pdf, "rb") as f:
             file_object.file.save("print.pdf", File(f))
         file_object.save()
-        re_base = r"pdfs/[a-zA-Z0-9]+\.pdf"
+        re_base = r"pdfs/print_[a-zA-Z0-9]+\.pdf"
         assert re.match(re_base, file_object.file.name)
 
     def test_delete_signal(self):
         dummy_person = Person.objects.create(first_name="Jane", last_name="Doe")
-        file_object = PDFFile.objects.create(person=dummy_person, html=self._get_test_html())
+        file_object = PDFFile.objects.create(person=dummy_person, html_file=self._get_test_html())
         with open(self._test_pdf, "rb") as f:
             file_object.file.save("print.pdf", File(f))
         file_object.save()
@@ -59,10 +59,10 @@ class PDFFIleTest(TransactionTestCase):
     def test_delete_expired_files(self):
         # Create test instances
         dummy_person = Person.objects.create(first_name="Jane", last_name="Doe")
-        file_object = PDFFile.objects.create(person=dummy_person, html=self._get_test_html())
+        file_object = PDFFile.objects.create(person=dummy_person, html_file=self._get_test_html())
         file_object2 = PDFFile.objects.create(
             person=dummy_person,
-            html=self._get_test_html(),
+            html_file=self._get_test_html(),
             expires_at=timezone.now() + timedelta(minutes=10),
         )
         with open(self._test_pdf, "rb") as f:
diff --git a/aleksis/core/tests/models/test_person.py b/aleksis/core/tests/models/test_person.py
index b42194f22424566bea2b99adc42b6995dbfaf626..dbe11f71812977c79b0b0fef5b4b975eca33b2e7 100644
--- a/aleksis/core/tests/models/test_person.py
+++ b/aleksis/core/tests/models/test_person.py
@@ -9,3 +9,9 @@ def test_full_name():
     _person = Person.objects.create(first_name="Jane", last_name="Doe")
 
     assert _person.full_name == "Doe, Jane"
+
+
+def test_delete():
+    _person = Person.objects.create(first_name="Jane", last_name="Doe")
+    _person.delete()
+    assert not Person.objects.filter(first_name="Jane", last_name="Doe").exists()
diff --git a/aleksis/core/urls.py b/aleksis/core/urls.py
index d48b7fa3824ad6db2849327af7735c8c9d7655d1..d1385037579d48142dc5bea9d25410fbb2115dfa 100644
--- a/aleksis/core/urls.py
+++ b/aleksis/core/urls.py
@@ -10,6 +10,7 @@ import debug_toolbar
 from ckeditor_uploader import views as ckeditor_uploader_views
 from django_js_reverse.views import urls_js
 from health_check.urls import urlpatterns as health_urls
+from oauth2_provider.views import ConnectDiscoveryInfoView
 from rules.contrib.views import permission_required
 from two_factor.urls import urlpatterns as tf_urls
 
@@ -17,25 +18,39 @@ from . import views
 
 urlpatterns = [
     path("", include("django_prometheus.urls")),
-    path("", include("pwa.urls"), name="pwa"),
+    path(settings.MEDIA_URL.removeprefix("/"), include("titofisto.urls")),
+    path("manifest.json", views.ManifestView.as_view(), name="manifest"),
+    path("serviceworker.js", views.ServiceWorkerView.as_view(), name="service_worker"),
+    path("offline/", views.OfflineView.as_view(), name="offline"),
     path("about/", views.about, name="about_aleksis"),
+    path("accounts/logout/", auth_views.LogoutView.as_view(), name="logout"),
+    path(
+        "accounts/password/change/",
+        views.CustomPasswordChangeView.as_view(),
+        name="account_change_password",
+    ),
+    path("accounts/", include("allauth.urls")),
+    path(
+        "accounts/social/connections/<int:pk>/delete",
+        views.SocialAccountDeleteView.as_view(),
+        name="delete_social_account_by_pk",
+    ),
     path("admin/", admin.site.urls),
     path("admin/uwsgi/", include("django_uwsgi.urls")),
     path("data_management/", views.data_management, name="data_management"),
     path("status/", views.SystemStatus.as_view(), name="system_status"),
     path("", include(tf_urls)),
-    path("celery_progress/", include("celery_progress.urls")),
+    path("celery_progress/<str:task_id>/", views.CeleryProgressView.as_view(), name="task_status"),
     path("accounts/logout/", auth_views.LogoutView.as_view(), name="logout"),
     path("school_terms/", views.SchoolTermListView.as_view(), name="school_terms"),
     path("school_terms/create/", views.SchoolTermCreateView.as_view(), name="create_school_term"),
     path("school_terms/<int:pk>/", views.SchoolTermEditView.as_view(), name="edit_school_term"),
     path("persons", views.persons, name="persons"),
-    path("persons/accounts", views.persons_accounts, name="persons_accounts"),
-    path("person", views.person, name="person"),
-    path("person/create", views.edit_person, name="create_person"),
-    path("person/<int:id_>", views.person, name="person_by_id"),
-    path("person/<int:id_>/edit", views.edit_person, name="edit_person_by_id"),
-    path("person/<int:id_>/delete", views.delete_person, name="delete_person_by_id"),
+    path("person/", views.person, name="person"),
+    path("person/create/", views.CreatePersonView.as_view(), name="create_person"),
+    path("person/<int:id_>/", views.person, name="person_by_id"),
+    path("person/<int:pk>/edit/", views.EditPersonView.as_view(), name="edit_person_by_id"),
+    path("person/<int:id_>/delete/", views.delete_person, name="delete_person_by_id"),
     path("groups", views.groups, name="groups"),
     path("groups/additional_fields", views.additional_fields, name="additional_fields"),
     path("groups/child_groups/", views.groups_child_groups, name="groups_child_groups"),
@@ -79,18 +94,43 @@ urlpatterns = [
     path("announcement/edit/<int:id_>/", views.announcement_form, name="edit_announcement"),
     path("announcement/delete/<int:id_>/", views.delete_announcement, name="delete_announcement"),
     path("search/searchbar/", views.searchbar_snippets, name="searchbar_snippets"),
-    path("search/", views.PermissionSearchView(), name="haystack_search"),
+    path("search/", views.PermissionSearchView.as_view(), name="haystack_search"),
     path("maintenance-mode/", include("maintenance_mode.urls")),
     path("impersonate/", include("impersonate.urls")),
+    path(
+        ".well-known/openid-configuration",
+        ConnectDiscoveryInfoView.as_view(),
+        name="oidc_configuration",
+    ),
+    path("oauth2/applications/", views.OAuth2ListView.as_view(), name="oauth2_applications"),
+    path(
+        "oauth2/applications/register/",
+        views.OAuth2RegisterView.as_view(),
+        name="register_oauth_application",
+    ),
+    path(
+        "oauth2/applications/<int:pk>/", views.OAuth2DetailView.as_view(), name="oauth2_application"
+    ),
+    path(
+        "oauth2/applications/<int:pk>/delete/",
+        views.OAuth2DeleteView.as_view(),
+        name="delete_oauth2_application",
+    ),
+    path(
+        "oauth2/applications/<int:pk>/edit/",
+        views.OAuth2EditView.as_view(),
+        name="edit_oauth2_application",
+    ),
+    path("oauth2/", include("oauth2_provider.urls", namespace="oauth2_provider")),
     path("__i18n__/", include("django.conf.urls.i18n")),
     path(
         "ckeditor/upload/",
-        permission_required("core.ckeditor_upload_files")(ckeditor_uploader_views.upload),
+        permission_required("core.ckeditor_upload_files_rule")(ckeditor_uploader_views.upload),
         name="ckeditor_upload",
     ),
     path(
         "ckeditor/browse/",
-        permission_required("core.ckeditor_upload_files")(ckeditor_uploader_views.browse),
+        permission_required("core.ckeditor_upload_files_rule")(ckeditor_uploader_views.browse),
         name="ckeditor_browse",
     ),
     path("select2/", include("django_select2.urls")),
@@ -248,9 +288,11 @@ urlpatterns = [
         name="assign_permission",
     ),
     path("pdfs/<int:pk>/", views.RedirectToPDFFile.as_view(), name="redirect_to_pdf_file"),
-    path("pdfs/<int:pk>/html/", views.HTMLForPDFFile.as_view(), name="html_for_pdf_file"),
 ]
 
+# Use custom server error handler to get a request object in the template
+handler500 = views.server_error
+
 # Add URLs for optional features
 if hasattr(settings, "TWILIO_ACCOUNT_SID"):
     from two_factor.gateways.twilio.urls import urlpatterns as tf_twilio_urls  # noqa
diff --git a/aleksis/core/util/apps.py b/aleksis/core/util/apps.py
index a2a6ced0e87b610dde1d74efe57b0fdf46f969c9..8157f1dcf2668aac249e107b3e37c1b3676ebeed 100644
--- a/aleksis/core/util/apps.py
+++ b/aleksis/core/util/apps.py
@@ -1,21 +1,26 @@
-from typing import Any, List, Optional, Sequence, Tuple
+from importlib import metadata
+from typing import TYPE_CHECKING, Any, Optional, Sequence
 
 import django.apps
 from django.contrib.auth.signals import user_logged_in, user_logged_out
 from django.db.models.signals import post_migrate, pre_migrate
 from django.http import HttpRequest
 
-import pkg_resources
 from dynamic_preferences.signals import preference_updated
 from license_expression import Licensing
 from spdx_license_list import LICENSES
 
 from .core_helpers import copyright_years
 
+if TYPE_CHECKING:
+    from oauth2_provider.models import AbstractApplication
+
 
 class AppConfig(django.apps.AppConfig):
     """An extended version of DJango's AppConfig container."""
 
+    default_auto_field = "django.db.models.BigAutoField"
+
     def ready(self):
         super().ready()
 
@@ -30,7 +35,7 @@ class AppConfig(django.apps.AppConfig):
         self.preference_updated(self)
 
     def get_distribution_name(self):
-        """Get pkg_resources distribution name of application package."""
+        """Get distribution name of application package."""
         if hasattr(self, "dist_name"):
             return self.dist_name
         elif self.name.lower().startswith("aleksis.apps."):
@@ -39,12 +44,12 @@ class AppConfig(django.apps.AppConfig):
         return None
 
     def get_distribution(self):
-        """Get pkg_resources distribution of application package."""
+        """Get distribution of application package."""
         dist_name = self.get_distribution_name()
         if dist_name:
             try:
-                dist = pkg_resources.get_distribution(dist_name)
-            except pkg_resources.DistributionNotFound:
+                dist = metadata.distribution(dist_name)
+            except metadata.PackageNotFoundError:
                 return None
 
             return dist
@@ -71,7 +76,7 @@ class AppConfig(django.apps.AppConfig):
                 return "unknown"
 
     @classmethod
-    def get_licence(cls) -> Tuple:
+    def get_licence(cls) -> tuple:
         """Get tuple of licence information of application package."""
         # Get string representation of licence in SPDX format
         licence = getattr(cls, "licence", None)
@@ -129,7 +134,7 @@ class AppConfig(django.apps.AppConfig):
         # TODO Try getting from distribution if not set
 
     @classmethod
-    def get_copyright(cls) -> Sequence[Tuple[str, str, str]]:
+    def get_copyright(cls) -> Sequence[tuple[str, str, str]]:
         """Get copyright information tuples for application package."""
         copyrights = getattr(cls, "copyright_info", tuple())
 
@@ -170,7 +175,7 @@ class AppConfig(django.apps.AppConfig):
         verbosity: int,
         interactive: bool,
         using: str,
-        plan: List[Tuple],
+        plan: list[tuple],
         apps: django.apps.registry.Apps,
         **kwargs,
     ) -> None:
@@ -212,6 +217,33 @@ class AppConfig(django.apps.AppConfig):
         """
         pass
 
+    @classmethod
+    def get_all_scopes(cls) -> dict[str, str]:
+        """Return all OAuth scopes and their descriptions for this app."""
+        return {}
+
+    @classmethod
+    def get_available_scopes(
+        cls,
+        application: Optional["AbstractApplication"] = None,
+        request: Optional[HttpRequest] = None,
+        *args,
+        **kwargs,
+    ) -> list[str]:
+        """Return a list of all OAuth scopes available to the request and application."""
+        return list(cls.get_all_scopes().keys())
+
+    @classmethod
+    def get_default_scopes(
+        cls,
+        application: Optional["AbstractApplication"] = None,
+        request: Optional[HttpRequest] = None,
+        *args,
+        **kwargs,
+    ) -> list[str]:
+        """Return a list of all OAuth scopes to always include for this request and application."""
+        return []
+
     def _maintain_default_data(self):
         from django.contrib.auth.models import Permission
         from django.contrib.contenttypes.models import ContentType
diff --git a/aleksis/core/util/auth_helpers.py b/aleksis/core/util/auth_helpers.py
new file mode 100644
index 0000000000000000000000000000000000000000..21acddda5acef95d6fa7c9636e999717d899ca0b
--- /dev/null
+++ b/aleksis/core/util/auth_helpers.py
@@ -0,0 +1,167 @@
+"""Helpers/overrides for django-allauth."""
+
+from typing import Optional
+
+from django.conf import settings
+from django.http import HttpRequest
+
+from allauth.account.adapter import DefaultAccountAdapter
+from allauth.socialaccount.adapter import DefaultSocialAccountAdapter
+from oauth2_provider.models import AbstractApplication
+from oauth2_provider.oauth2_validators import OAuth2Validator
+from oauth2_provider.scopes import BaseScopes
+from oauth2_provider.views.mixins import (
+    ClientProtectedResourceMixin as _ClientProtectedResourceMixin,
+)
+from oauthlib.common import Request as OauthlibRequest
+
+from .apps import AppConfig
+from .core_helpers import get_site_preferences, has_person
+
+
+class OurSocialAccountAdapter(DefaultSocialAccountAdapter):
+    """Customised adapter that recognises other authentication mechanisms."""
+
+    def validate_disconnect(self, account, accounts):
+        """Validate whether or not the socialaccount account can be safely disconnected.
+
+        Honours other authentication backends, i.e. ignores unusable passwords if LDAP is used.
+        """
+        if "django_auth_ldap.backend.LDAPBackend" in settings.AUTHENTICATION_BACKENDS:
+            # Ignore upstream validation error as we do not need a usable password
+            return None
+
+        # Let upstream decide whether we can disconnect or not
+        return super().validate_disconnect(account, accounts)
+
+
+class OurAccountAdapter(DefaultAccountAdapter):
+    """Customised adapter to allow to disable signup."""
+
+    def is_open_for_signup(self, request):
+        return get_site_preferences()["auth__signup_enabled"]
+
+
+class CustomOAuth2Validator(OAuth2Validator):
+    def get_additional_claims(self, request):
+        django_request = HttpRequest()
+        django_request.META = request.headers
+
+        claims = {
+            "preferred_username": request.user.username,
+        }
+
+        if "profile" in request.scopes:
+            if has_person(request.user):
+                claims["given_name"] = request.user.person.first_name
+                claims["family_name"] = request.user.person.last_name
+                claims["profile"] = django_request.build_absolute_uri(
+                    request.user.person.get_absolute_url()
+                )
+                if request.user.person.photo:
+                    claims["picture"] = django_request.build_absolute_uri(
+                        request.user.person.photo.url
+                    )
+            else:
+                claims["given_name"] = request.user.first_name
+                claims["family_name"] = request.user.last_name
+
+        if "email" in request.scopes:
+            if has_person(request.user):
+                claims["email"] = request.user.person.email
+            else:
+                claims["email"] = request.user.email
+
+        if "address" in request.scopes and has_person(request.user):
+            claims["address"] = {
+                "street_address": request.user.person.street
+                + " "
+                + request.user.person.housenumber,
+                "locality": request.user.person.place,
+                "postal_code": request.user.person.postal_code,
+            }
+
+        if "groups" in request.scopes and has_person(request.user):
+            claims["groups"] = request.user.person.groups.values_list("name", flat=True).all()
+
+        return claims
+
+
+class AppScopes(BaseScopes):
+    """Scopes backend for django-oauth-toolkit gathering scopes from apps.
+
+    Will call the respective method on all known AlekSIS app configs and
+    join the results.
+    """
+
+    def get_all_scopes(self) -> dict[str, str]:
+        scopes = {}
+        for app in AppConfig.__subclasses__():
+            scopes |= app.get_all_scopes()
+        return scopes
+
+    def get_available_scopes(
+        self,
+        application: Optional[AbstractApplication] = None,
+        request: Optional[HttpRequest] = None,
+        *args,
+        **kwargs
+    ) -> list[str]:
+        scopes = []
+        for app in AppConfig.__subclasses__():
+            scopes += app.get_available_scopes()
+        # Filter by allowed scopes of requesting application
+        if application and application.allowed_scopes:
+            scopes = list(filter(lambda scope: scope in application.allowed_scopes, scopes))
+        return scopes
+
+    def get_default_scopes(
+        self,
+        application: Optional[AbstractApplication] = None,
+        request: Optional[HttpRequest] = None,
+        *args,
+        **kwargs
+    ) -> list[str]:
+        scopes = []
+        for app in AppConfig.__subclasses__():
+            scopes += app.get_default_scopes()
+        # Filter by allowed scopes of requesting application
+        if application and application.allowed_scopes:
+            scopes = list(filter(lambda scope: scope in application.allowed_scopes, scopes))
+        return scopes
+
+
+class ClientProtectedResourceMixin(_ClientProtectedResourceMixin):
+    """Mixin for protecting resources with client authentication as mentioned in rfc:`3.2.1`.
+
+    This involves authenticating with any of: HTTP Basic Auth, Client Credentials and
+    Access token in that order. Breaks off after first validation.
+
+    This sub-class extends the functionality of Django OAuth Toolkit's mixin with support
+    for AlekSIS's `allowed_scopes` feature. For applications that have configured allowed
+    scopes, the required scopes for the view are checked to be a subset of the application's
+    allowed scopes (best to be combined with ScopedResourceMixin).
+    """
+
+    def authenticate_client(self, request: HttpRequest) -> bool:
+        """Return a boolean representing if client is authenticated with client credentials.
+
+        If the view has configured required scopes, they are verified against the application's
+        allowed scopes.
+        """
+        # Build an OAuth request so we can handle client information
+        core = self.get_oauthlib_core()
+        uri, http_method, body, headers = core._extract_params(request)
+        oauth_request = OauthlibRequest(uri, http_method, body, headers)
+
+        # Verify general authentication of the client
+        if not core.server.request_validator.authenticate_client(oauth_request):
+            # Client credentials were invalid
+            return False
+
+        # Verify scopes of configured application
+        # The OAuth request was enriched with a reference to the Application when using the
+        #  validator above.
+        required_scopes = set(self.get_scopes() or [])
+        allowed_scopes = set(AppScopes().get_available_scopes(oauth_request.client) or [])
+        return required_scopes.issubset(allowed_scopes)
diff --git a/aleksis/core/util/celery_progress.py b/aleksis/core/util/celery_progress.py
index e957dd4f938291b11a85d00e115d302f512e609a..ef42f1692dab60dc9bc88f3c6bba4ce4beef3222 100644
--- a/aleksis/core/util/celery_progress.py
+++ b/aleksis/core/util/celery_progress.py
@@ -2,8 +2,12 @@ from functools import wraps
 from numbers import Number
 from typing import Callable, Generator, Iterable, Optional, Sequence, Union
 
+from django.apps import apps
 from django.contrib import messages
+from django.http import HttpRequest
+from django.shortcuts import render
 
+from celery.result import AsyncResult
 from celery_progress.backend import PROGRESS_STATE, AbstractProgressRecorder
 
 from ..celery import app
@@ -48,19 +52,19 @@ class ProgressRecorder(AbstractProgressRecorder):
             # ...
             result = do_something.delay(foo, bar, baz=baz)
 
-            context = {
-                "title": _("Progress: Import data"),
-                "back_url": reverse("index"),
-                "progress": {
-                    "task_id": result.task_id,
-                    "title": _("Import objects …"),
-                    "success": _("The import was done successfully."),
-                    "error": _("There was a problem while importing data."),
-                },
-            }
-
             # Render progress view
-            return render(request, "core/progress.html", context)
+            return render_progress_page(
+                request,
+                result,
+                title=_("Progress: Import data"),
+                back_url=reverse("index"),
+                progress_title=_("Import objects …"),
+                success_message=_("The import was done successfully."),
+                error_message=_("There was a problem while importing data."),
+            )
+
+    Please take a look at the documentation of ``render_progress_page``
+    to get all available options.
     """
 
     def __init__(self, task):
@@ -160,3 +164,60 @@ def recorded_task(orig: Optional[Callable] = None, **kwargs) -> Union[Callable,
     if orig and not kwargs:
         return _real_decorator(orig)
     return _real_decorator
+
+
+def render_progress_page(
+    request: HttpRequest,
+    task_result: AsyncResult,
+    title: str,
+    progress_title: str,
+    success_message: str,
+    error_message: str,
+    back_url: Optional[str] = None,
+    redirect_on_success_url: Optional[str] = None,
+    button_title: Optional[str] = None,
+    button_url: Optional[str] = None,
+    button_icon: Optional[str] = None,
+    context: Optional[dict] = None,
+):
+    """Show a page to track the progress of a Celery task using a ``ProgressRecorder``.
+
+    :param task_result: The ``AsyncResult`` of the task to track
+    :param title: The title of the progress page
+    :param progress_title: The text shown under the progress bar
+    :param success_message: The message shown on task success
+    :param error_message: The message shown on task failure
+    :param back_url: The URL for the back button (leave empty to disable back button)
+    :param redirect_on_success_url: The URL to redirect on task success
+    :param button_title: The label for a button shown on task success
+        (leave empty to not show a button)
+    :param button_url: The URL for the button
+    :param button_icon: The icon for the button (leave empty to not show an icon)
+    :param context: Additional context for the progress page
+    """
+    if not context:
+        context = {}
+
+    # Create TaskUserAssignment to track permissions on this task
+    TaskUserAssignment = apps.get_model("core", "TaskUserAssignment")
+    assignment = TaskUserAssignment.create_for_task_id(task_result.task_id, request.user)
+
+    # Prepare context for progress page
+    context["title"] = title
+    context["back_url"] = back_url
+    context["progress"] = {
+        "task_id": task_result.task_id,
+        "title": progress_title,
+        "success": success_message,
+        "error": error_message,
+        "redirect_on_success": redirect_on_success_url,
+    }
+
+    if button_url and button_title:
+        context["additional_button"] = {
+            "href": button_url,
+            "caption": button_title,
+            "icon": button_icon,
+        }
+
+    return render(request, "core/pages/progress.html", context)
diff --git a/aleksis/core/util/core_helpers.py b/aleksis/core/util/core_helpers.py
index 69350b13967b0cf1d57a6fc6307adcbc90725fe6..5871fd8dd7f568d463c8f672373fa60b708caddf 100644
--- a/aleksis/core/util/core_helpers.py
+++ b/aleksis/core/util/core_helpers.py
@@ -1,16 +1,14 @@
-import sys
+import os
 from datetime import datetime, timedelta
-from importlib import import_module
+from importlib import import_module, metadata
 from itertools import groupby
 from operator import itemgetter
 from typing import Any, Callable, Optional, Sequence, Union
-
-if sys.version_info >= (3, 9):
-    from importlib import metadata
-else:
-    import importlib_metadata as metadata
+from warnings import warn
 
 from django.conf import settings
+from django.core.exceptions import ImproperlyConfigured
+from django.core.files import File
 from django.db.models import Model, QuerySet
 from django.http import HttpRequest
 from django.shortcuts import get_object_or_404
@@ -130,23 +128,42 @@ def lazy_preference(section: str, name: str) -> Callable[[str, str], Any]:
     return lazy(_get_preference, str)(section, name)
 
 
-def lazy_get_favicon_url(
-    title: str, size: int, rel: str, default: Optional[str] = None
-) -> Callable[[str, str], Any]:
-    """Lazily get the URL to a favicon image."""
+def get_or_create_favicon(title: str, default: str, is_favicon: bool = False) -> "Favicon":
+    """Ensure that there is always a favicon object."""
+    from favicon.models import Favicon  # noqa
 
-    @cache_memoize(3600)
-    def _get_favicon_url(size: int, rel: str) -> Any:
-        from favicon.models import Favicon  # noqa
+    if not os.path.exists(default):
+        warn("staticfiles are not ready yet, not creating default icons")
+        return
+    elif os.path.isdir(default):
+        raise ImproperlyConfigured(f"staticfiles are broken: unexpected directory at {default}")
 
-        try:
-            favicon = Favicon.on_site.get(title=title)
-        except Favicon.DoesNotExist:
-            return default
-        else:
-            return favicon.get_favicon(size, rel).faviconImage.url
+    favicon, created = Favicon.on_site.get_or_create(
+        title=title, defaults={"isFavicon": is_favicon}
+    )
+
+    changed = False
+
+    if favicon.isFavicon != is_favicon:
+        favicon.isFavicon = True
+        changed = True
+
+    if created:
+        favicon.faviconImage.save(os.path.basename(default), File(open(default, "rb")))
+        changed = True
+
+    if changed:
+        favicon.save()
+
+    return favicon
 
-    return lazy(_get_favicon_url, str)(size, rel)
+
+def get_pwa_icons():
+    from django.conf import settings  # noqa
+
+    favicon = get_or_create_favicon("pwa_icon", settings.DEFAULT_FAVICON_PATHS["pwa_icon"])
+    favicon_imgs = favicon.get_favicons(config_override=settings.PWA_ICONS_CONFIG)
+    return favicon_imgs
 
 
 def is_impersonate(request: HttpRequest) -> bool:
@@ -185,18 +202,30 @@ def custom_information_processor(request: HttpRequest) -> dict:
     """Provide custom information in all templates."""
     from ..models import CustomMenu
 
-    return {
+    pwa_icons = get_pwa_icons()
+    regrouped_pwa_icons = {}
+    for pwa_icon in pwa_icons:
+        regrouped_pwa_icons.setdefault(pwa_icon.rel, {})
+        regrouped_pwa_icons[pwa_icon.rel][pwa_icon.size] = pwa_icon
+
+    context = {
         "FOOTER_MENU": CustomMenu.get_default("footer"),
-        "ALTERNATIVE_LOGIN_VIEWS_LIST": [
-            a[0]
-            for a in settings.ALTERNATIVE_LOGIN_VIEWS
-            if a[0] in settings.AUTHENTICATION_BACKENDS
-        ],
-        "ALTERNATIVE_LOGIN_VIEWS": [
-            a for a in settings.ALTERNATIVE_LOGIN_VIEWS if a[0] in settings.AUTHENTICATION_BACKENDS
-        ],
+        "ADMINS": settings.ADMINS,
+        "PWA_ICONS": regrouped_pwa_icons,
+        "SENTRY_ENABLED": settings.SENTRY_ENABLED,
     }
 
+    if settings.SENTRY_ENABLED:
+        context["SENTRY_SETTINGS"] = settings.SENTRY_SETTINGS
+
+        import sentry_sdk
+
+        span = sentry_sdk.Hub.current.scope.span
+        if span is not None:
+            context["SENTRY_TRACE_ID"] = span.to_traceparent()
+
+    return context
+
 
 def now_tomorrow() -> datetime:
     """Return current time tomorrow."""
@@ -208,21 +237,18 @@ def objectgetter_optional(
 ) -> Callable[[HttpRequest, Optional[int]], Model]:
     """Get an object by pk, defaulting to None."""
 
-    def get_object(request: HttpRequest, id_: Optional[int] = None, **kwargs) -> Model:
+    def get_object(request: HttpRequest, id_: Optional[int] = None, **kwargs) -> Optional[Model]:
         if id_ is not None:
             return get_object_or_404(model, pk=id_)
         else:
-            return eval(default) if default_eval else default  # noqa:S307
+            try:
+                return eval(default) if default_eval else default  # noqa:S307
+            except (AttributeError, KeyError, IndexError):
+                return None
 
     return get_object
 
 
-def handle_uploaded_file(f, filename: str):
-    with open(filename, "wb+") as destination:
-        for chunk in f.chunks():
-            destination.write(chunk)
-
-
 @cache_memoize(3600)
 def get_content_type_by_perm(perm: str) -> Union["ContentType", None]:
     from django.contrib.contenttypes.models import ContentType  # noqa
@@ -254,3 +280,38 @@ def queryset_rules_filter(
 def unread_notifications_badge(request: HttpRequest) -> int:
     """Generate badge content with the number of unread notifications."""
     return request.user.person.unread_notifications_count
+
+
+def monkey_patch() -> None:  # noqa
+    """Monkey-patch dependencies for special behaviour."""
+    # Unwrap promises in JSON serializer instead of stringifying
+    from django.core.serializers import json
+    from django.utils.functional import Promise
+
+    class DjangoJSONEncoder(json.DjangoJSONEncoder):
+        def default(self, o: Any) -> Any:
+            if isinstance(o, Promise) and hasattr(o, "copy"):
+                return o.copy()
+            return super().default(o)
+
+    json.DjangoJSONEncoder = DjangoJSONEncoder
+
+
+def get_allowed_object_ids(request: HttpRequest, models: list) -> list:
+    """Get all objects of all given models the user of a given request is allowed to view."""
+    allowed_object_ids = []
+
+    for model in models:
+        app_label = model._meta.app_label
+        model_name = model.__name__.lower()
+
+        # Loop through the pks of all objects of the current model the user is allowed to view
+        # and put the corresponding ids into a django-haystack-style-formatted list
+        allowed_object_ids += [
+            f"{app_label}.{model_name}.{pk}"
+            for pk in queryset_rules_filter(
+                request, model.objects.all(), f"{app_label}.view_{model_name}_rule"
+            ).values_list("pk", flat=True)
+        ]
+
+    return allowed_object_ids
diff --git a/aleksis/core/util/ldap.py b/aleksis/core/util/ldap.py
index 5a17cbbb5aea91030fab467c21897826f74eb6aa..96b058ac0060975cdd08281be94ba4df371aa7db 100644
--- a/aleksis/core/util/ldap.py
+++ b/aleksis/core/util/ldap.py
@@ -20,13 +20,13 @@ class LDAPBackend(_LDAPBackend):
         Django database in order to not require it to have global admin permissions
         on the LDAP directory.
         """
-        user = ldap_user.authenticate(password)
-
-        if not user:
-            # Fail early and do not try other backends
-            raise PermissionDenied("LDAP failed to authenticate user")
+        user = super().authenticate_ldap_user(ldap_user, password)
 
         if self.settings.SET_USABLE_PASSWORD:
+            if not user:
+                # Fail early and do not try other backends
+                raise PermissionDenied("LDAP failed to authenticate user")
+
             # Set a usable password so users can change their LDAP password
             user.set_password(password)
             user.save()
diff --git a/aleksis/core/util/pdf.py b/aleksis/core/util/pdf.py
index 18640a0e8667d91b7c31b8130b597e7992985e8d..3f27680d9d8b3fc569a457ec78a2ce7c154c23d1 100644
--- a/aleksis/core/util/pdf.py
+++ b/aleksis/core/util/pdf.py
@@ -1,23 +1,27 @@
 import os
 import subprocess  # noqa
 from tempfile import TemporaryDirectory
-from typing import Optional
+from typing import Optional, Tuple
+from urllib.parse import urljoin
 
+from django.conf import settings
 from django.core.files import File
+from django.core.files.base import ContentFile
 from django.http.request import HttpRequest
 from django.http.response import HttpResponse
-from django.shortcuts import get_object_or_404, render
+from django.shortcuts import get_object_or_404
 from django.template.loader import render_to_string
 from django.urls import reverse
 from django.utils import timezone
 from django.utils.translation import get_language
 from django.utils.translation import gettext as _
 
+from celery.result import AsyncResult
 from celery_progress.backend import ProgressRecorder
 
 from aleksis.core.celery import app
 from aleksis.core.models import PDFFile
-from aleksis.core.util.celery_progress import recorded_task
+from aleksis.core.util.celery_progress import recorded_task, render_progress_page
 
 
 @recorded_task
@@ -63,6 +67,26 @@ def generate_pdf(
     recorder.set_progress(1, 1)
 
 
+def generate_pdf_from_template(
+    template_name: str, context: Optional[dict] = None, request: Optional[HttpRequest] = None
+) -> Tuple[PDFFile, AsyncResult]:
+    """Start a PDF generation task and return the matching file object and Celery result."""
+    html_template = render_to_string(template_name, context, request)
+
+    file_object = PDFFile.objects.create(html_file=ContentFile(html_template, name="source.html"))
+
+    # As this method may be run in background and there is no request available,
+    # we have to use a predefined URL from settings then
+    if request:
+        html_url = request.build_absolute_uri(file_object.html_file.url)
+    else:
+        html_url = urljoin(settings.BASE_URL, file_object.html_file.url)
+
+    result = generate_pdf.delay(file_object.pk, html_url, lang=get_language())
+
+    return file_object, result
+
+
 def render_pdf(request: HttpRequest, template_name: str, context: dict = None) -> HttpResponse:
     """Start PDF generation and show progress page.
 
@@ -71,34 +95,23 @@ def render_pdf(request: HttpRequest, template_name: str, context: dict = None) -
     if not context:
         context = {}
 
-    html_template = render_to_string(template_name, context)
-
-    file_object = PDFFile.objects.create(person=request.user.person, html=html_template)
-    html_url = request.build_absolute_uri(file_object.html_url)
-
-    result = generate_pdf.delay(file_object.pk, html_url, lang=get_language())
+    file_object, result = generate_pdf_from_template(template_name, context, request)
 
     redirect_url = reverse("redirect_to_pdf_file", args=[file_object.pk])
 
-    progress_context = {
-        "title": _("Progress: Generate PDF file"),
-        "back_url": context.get("back_url", "index"),
-        "progress": {
-            "task_id": result.task_id,
-            "title": _("Generating PDF file …"),
-            "success": _("The PDF file has been generated successfully."),
-            "error": _("There was a problem while generating the PDF file."),
-            "redirect_on_success": redirect_url,
-        },
-        "additional_button": {
-            "href": redirect_url,
-            "caption": _("Download PDF"),
-            "icon": "picture_as_pdf",
-        },
-    }
-
-    # Render progress view
-    return render(request, "core/pages/progress.html", progress_context)
+    return render_progress_page(
+        request,
+        result,
+        title=_("Progress: Generate PDF file"),
+        progress_title=_("Generating PDF file …"),
+        success_message=_("The PDF file has been generated successfully."),
+        error_message=_("There was a problem while generating the PDF file."),
+        redirect_on_success_url=redirect_url,
+        back_url=context.get("back_url", reverse("index")),
+        button_title=_("Download PDF"),
+        button_url=redirect_url,
+        button_icon="picture_as_pdf",
+    )
 
 
 def clean_up_expired_pdf_files() -> None:
diff --git a/aleksis/core/util/predicates.py b/aleksis/core/util/predicates.py
index c6e6035301be735ac439bd21a229446ad490208c..bcba7e8637d81aa9c554aef65cdb431b059753a9 100644
--- a/aleksis/core/util/predicates.py
+++ b/aleksis/core/util/predicates.py
@@ -5,6 +5,7 @@ from django.contrib.auth.models import User
 from django.db.models import Model
 from django.http import HttpRequest
 
+from django_otp import user_has_device
 from guardian.backends import ObjectPermissionBackend
 from guardian.shortcuts import get_objects_for_user
 from rules import predicate
@@ -74,14 +75,19 @@ def has_any_object(perm: str, klass):
 
     Build predicate which checks whether a user has access
     to objects with the provided permission or rule.
+    Differentiates between object-related permissions and rules.
     """
     name = f"has_any_object:{perm}"
 
     @predicate(name)
     def fn(user: User) -> bool:
         ct_perm = get_content_type_by_perm(perm)
+        # In case an object-related permission with the same ContentType class as the given class
+        # is passed, the optimized django-guardian get_objects_for_user function is used.
         if ct_perm and ct_perm.model_class() == klass:
             return get_objects_for_user(user, perm, klass).exists()
+        # In other cases, it is checked for each object of the given model whether the current user
+        # fulfills the given rule.
         else:
             return queryset_rules_filter(user, klass.objects.all(), perm).exists()
 
@@ -114,7 +120,7 @@ def is_current_person(user: User, obj: Model) -> bool:
 @predicate
 def is_group_owner(user: User, group: Group) -> bool:
     """Predicate which checks if the user is a owner of the provided group."""
-    return group.owners.filter(owners=user.person).exists()
+    return user.person in group.owners.all()
 
 
 @predicate
@@ -142,3 +148,9 @@ def contains_site_preference_value(section: str, pref: str, value: str):
         return bool(value in get_site_preferences()[f"{section}__{pref}"])
 
     return fn
+
+
+@predicate
+def has_activated_2fa(user: User) -> bool:
+    """Check if the user has activated two-factor authentication."""
+    return user_has_device(user)
diff --git a/aleksis/core/views.py b/aleksis/core/views.py
index 1c55cbc3459acfd129e63828fa36d14fa095065b..76f44a4938dc26bb4b3c4370e9d5f48f94abc2d1 100644
--- a/aleksis/core/views.py
+++ b/aleksis/core/views.py
@@ -2,38 +2,55 @@ from typing import Any, Dict, Optional, Type
 from urllib.parse import urlencode
 
 from django.apps import apps
+from django.conf import settings
 from django.contrib.auth.models import Group as DjangoGroup
 from django.contrib.auth.models import Permission, User
 from django.contrib.contenttypes.models import ContentType
-from django.core.exceptions import PermissionDenied
+from django.core.exceptions import PermissionDenied, ValidationError
 from django.core.paginator import Paginator
 from django.db.models import QuerySet
 from django.forms.models import BaseModelForm, modelform_factory
-from django.http import Http404, HttpRequest, HttpResponse, HttpResponseNotFound
+from django.http import (
+    Http404,
+    HttpRequest,
+    HttpResponse,
+    HttpResponseNotFound,
+    HttpResponseRedirect,
+    HttpResponseServerError,
+    JsonResponse,
+)
 from django.shortcuts import get_object_or_404, redirect, render
+from django.template import loader
 from django.urls import reverse, reverse_lazy
 from django.utils.decorators import method_decorator
+from django.utils.translation import get_language
 from django.utils.translation import gettext_lazy as _
 from django.views.decorators.cache import never_cache
+from django.views.defaults import ERROR_500_TEMPLATE_NAME
 from django.views.generic import FormView
 from django.views.generic.base import TemplateView, View
 from django.views.generic.detail import DetailView, SingleObjectMixin
+from django.views.generic.edit import DeleteView
 from django.views.generic.list import ListView
 
 import reversion
+from allauth.account.views import PasswordChangeView
+from allauth.socialaccount.adapter import get_adapter
+from allauth.socialaccount.models import SocialAccount
+from celery_progress.views import get_progress
 from django_celery_results.models import TaskResult
 from django_filters.views import FilterView
 from django_tables2 import RequestConfig, SingleTableMixin, SingleTableView
 from dynamic_preferences.forms import preference_form_builder
 from guardian.shortcuts import GroupObjectPermission, UserObjectPermission, get_objects_for_user
+from haystack.generic_views import SearchView
 from haystack.inputs import AutoQuery
 from haystack.query import SearchQuerySet
-from haystack.views import SearchView
+from haystack.utils.loading import UnifiedIndex
 from health_check.views import MainView
 from reversion import set_user
 from reversion.views import RevisionMixin
 from rules.contrib.views import PermissionRequiredMixin, permission_required
-from templated_email import send_templated_mail
 
 from aleksis.core.data_checks import DataCheckRegistry, check_data
 
@@ -54,10 +71,10 @@ from .forms import (
     EditAdditionalFieldForm,
     EditGroupForm,
     EditGroupTypeForm,
-    EditPersonForm,
     GroupPreferenceForm,
+    OAuthApplicationForm,
+    PersonForm,
     PersonPreferenceForm,
-    PersonsAccountsFormSet,
     SchoolTermForm,
     SelectPermissionForm,
     SitePreferenceForm,
@@ -73,9 +90,11 @@ from .models import (
     Group,
     GroupType,
     Notification,
+    OAuthApplication,
     PDFFile,
     Person,
     SchoolTerm,
+    TaskUserAssignment,
 )
 from .registries import (
     group_preferences_registry,
@@ -96,7 +115,14 @@ from .tables import (
 )
 from .util import messages
 from .util.apps import AppConfig
-from .util.core_helpers import get_site_preferences, has_person, objectgetter_optional
+from .util.celery_progress import render_progress_page
+from .util.core_helpers import (
+    get_allowed_object_ids,
+    get_pwa_icons,
+    get_site_preferences,
+    has_person,
+    objectgetter_optional,
+)
 from .util.forms import PreferenceLayout
 from .util.pdf import render_pdf
 
@@ -112,7 +138,59 @@ class RenderPDFView(TemplateView):
         return render_pdf(request, self.template_name, context)
 
 
-@permission_required("core.view_dashboard")
+class ServiceWorkerView(View):
+    """Render serviceworker.js under root URL.
+
+    This can't be done by static files,
+    because the PWA has a scope and
+    only accepts service worker files from the root URL.
+    """
+
+    def get(self, request: HttpRequest, *args, **kwargs) -> HttpResponse:
+        return HttpResponse(
+            open(settings.SERVICE_WORKER_PATH, "rt"), content_type="application/javascript"
+        )
+
+
+class ManifestView(View):
+    """Build manifest.json for PWA."""
+
+    def get(self, request: HttpRequest, *args, **kwargs) -> HttpResponse:
+        prefs = get_site_preferences()
+        pwa_imgs = get_pwa_icons()
+
+        icons = [
+            {
+                "src": favicon_img.faviconImage.url,
+                "sizes": f"{favicon_img.size}x{favicon_img.size}",
+            }
+            for favicon_img in pwa_imgs
+        ]
+
+        manifest = {
+            "name": prefs["general__title"],
+            "short_name": prefs["general__title"],
+            "description": prefs["general__description"],
+            "start_url": "/",
+            "scope": "/",
+            "lang": get_language(),
+            "display": "standalone",
+            "orientation": "any",
+            "status_bar": "default",
+            "background_color": "#ffffff",
+            "theme_color": prefs["theme__primary"],
+            "icons": icons,
+        }
+        return JsonResponse(manifest)
+
+
+class OfflineView(TemplateView):
+    """Show an error page if there is no internet connection."""
+
+    template_name = "offline.html"
+
+
+@permission_required("core.view_dashboard_rule")
 def index(request: HttpRequest) -> HttpResponse:
     """View for dashboard."""
     context = {}
@@ -149,13 +227,13 @@ def index(request: HttpRequest) -> HttpResponse:
 
 
 class NotificationsListView(PermissionRequiredMixin, ListView):
-    permission_required = "core.view_notifications"
+    permission_required = "core.view_notifications_rule"
     template_name = "core/notifications.html"
 
     def get_queryset(self) -> QuerySet:
         return self.request.user.person.notifications.order_by("-created")
 
-    def get_context_data(self, **kwargs: Any) -> Dict[str, Any]:
+    def get_context_data(self, **kwargs: Any) -> dict[str, Any]:
         self.get_queryset().filter(read=False).update(read=True)
         return super().get_context_data(**kwargs)
 
@@ -176,7 +254,7 @@ class SchoolTermListView(PermissionRequiredMixin, SingleTableView):
 
     model = SchoolTerm
     table_class = SchoolTermTable
-    permission_required = "core.view_schoolterm"
+    permission_required = "core.view_schoolterm_rule"
     template_name = "core/school_term/list.html"
 
 
@@ -186,7 +264,7 @@ class SchoolTermCreateView(PermissionRequiredMixin, AdvancedCreateView):
 
     model = SchoolTerm
     form_class = SchoolTermForm
-    permission_required = "core.add_schoolterm"
+    permission_required = "core.add_schoolterm_rule"
     template_name = "core/school_term/create.html"
     success_url = reverse_lazy("school_terms")
     success_message = _("The school term has been created.")
@@ -204,7 +282,7 @@ class SchoolTermEditView(PermissionRequiredMixin, AdvancedEditView):
     success_message = _("The school term has been saved.")
 
 
-@permission_required("core.view_persons")
+@permission_required("core.view_persons_rule")
 def persons(request: HttpRequest) -> HttpResponse:
     """List view listing all persons."""
     context = {}
@@ -227,7 +305,7 @@ def persons(request: HttpRequest) -> HttpResponse:
 
 
 @permission_required(
-    "core.view_person", fn=objectgetter_optional(Person, "request.user.person", True)
+    "core.view_person_rule", fn=objectgetter_optional(Person, "request.user.person", True)
 )
 def person(request: HttpRequest, id_: Optional[int] = None) -> HttpResponse:
     """Detail view for one person; defaulting to logged-in person."""
@@ -247,7 +325,7 @@ def person(request: HttpRequest, id_: Optional[int] = None) -> HttpResponse:
     return render(request, "core/person/full.html", context)
 
 
-@permission_required("core.view_group", fn=objectgetter_optional(Group, None, False))
+@permission_required("core.view_group_rule", fn=objectgetter_optional(Group, None, False))
 def group(request: HttpRequest, id_: int) -> HttpResponse:
     """Detail view for one group."""
     context = {}
@@ -280,7 +358,7 @@ def group(request: HttpRequest, id_: int) -> HttpResponse:
     return render(request, "core/group/full.html", context)
 
 
-@permission_required("core.view_groups")
+@permission_required("core.view_groups_rule")
 def groups(request: HttpRequest) -> HttpResponse:
     """List view for listing all groups."""
     context = {}
@@ -301,28 +379,7 @@ def groups(request: HttpRequest) -> HttpResponse:
 
 
 @never_cache
-@permission_required("core.link_persons_accounts")
-def persons_accounts(request: HttpRequest) -> HttpResponse:
-    """View allowing to batch-process linking of users to persons."""
-    context = {}
-
-    # Get all persons
-    persons_qs = Person.objects.all()
-
-    # Form set with one form per known person
-    persons_accounts_formset = PersonsAccountsFormSet(request.POST or None, queryset=persons_qs)
-
-    if request.method == "POST":
-        if persons_accounts_formset.is_valid():
-            persons_accounts_formset.save()
-
-    context["persons_accounts_formset"] = persons_accounts_formset
-
-    return render(request, "core/person/accounts.html", context)
-
-
-@never_cache
-@permission_required("core.assign_child_groups_to_groups")
+@permission_required("core.assign_child_groups_to_groups_rule")
 def groups_child_groups(request: HttpRequest) -> HttpResponse:
     """View for batch-processing assignment from child groups to groups."""
     context = {}
@@ -359,59 +416,38 @@ def groups_child_groups(request: HttpRequest) -> HttpResponse:
     return render(request, "core/group/child_groups.html", context)
 
 
-@never_cache
-@permission_required("core.edit_person", fn=objectgetter_optional(Person))
-def edit_person(request: HttpRequest, id_: Optional[int] = None) -> HttpResponse:
-    """Edit view for a single person, defaulting to logged-in person."""
-    context = {}
+@method_decorator(never_cache, name="dispatch")
+class CreatePersonView(PermissionRequiredMixin, AdvancedCreateView):
+    form_class = PersonForm
+    model = Person
+    permission_required = "core.create_person_rule"
+    template_name = "core/person/create.html"
+    success_message = _("The person has been saved.")
 
-    person = objectgetter_optional(Person)(request, id_)
-    context["person"] = person
 
-    if id_:
-        # Edit form for existing group
-        edit_person_form = EditPersonForm(
-            request, 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, request.POST or None, request.FILES or None)
-        else:
-            raise PermissionDenied()
-    if request.method == "POST":
-        if edit_person_form.is_valid():
-            if person and person == request.user.person:
-                # Check if user edited non-editable field
-                notification_fields = get_site_preferences()[
-                    "account__notification_on_person_change"
-                ]
-                send_notification_fields = set(edit_person_form.changed_data).intersection(
-                    set(notification_fields)
-                )
-                context["send_notification_fields"] = send_notification_fields
-                if send_notification_fields:
-                    context["send_notification_fields"] = send_notification_fields
-                    send_templated_mail(
-                        template_name="person_changed",
-                        from_email=request.user.person.mail_sender_via,
-                        headers={
-                            "Reply-To": request.user.person.mail_sender,
-                            "Sender": request.user.person.mail_sender,
-                        },
-                        recipient_list=[
-                            get_site_preferences()["account__person_change_notification_contact"]
-                        ],
-                        context=context,
-                    )
-            with reversion.create_revision():
-                set_user(request.user)
-                edit_person_form.save(commit=True)
-            messages.success(request, _("The person has been saved."))
+@method_decorator(never_cache, name="dispatch")
+class EditPersonView(PermissionRequiredMixin, RevisionMixin, AdvancedEditView):
+    form_class = PersonForm
+    model = Person
+    permission_required = "core.edit_person_rule"
+    context_object_name = "person"
+    template_name = "core/person/edit.html"
+    success_message = _("The person has been saved.")
+
+    def get_form_kwargs(self):
+        kwargs = super().get_form_kwargs()
+        kwargs["request"] = self.request
+        return kwargs
 
-    context["edit_person_form"] = edit_person_form
+    def form_valid(self, form):
+        if self.object == self.request.user.person:
+            # Get all changed fields and send a notification about them
+            notification_fields = get_site_preferences()["account__notification_on_person_change"]
+            send_notification_fields = set(form.changed_data).intersection(set(notification_fields))
 
-    return render(request, "core/person/edit.html", context)
+            if send_notification_fields:
+                self.object.notify_about_changed_data(send_notification_fields)
+        return super().form_valid(form)
 
 
 def get_group_by_id(request: HttpRequest, id_: Optional[int] = None):
@@ -422,7 +458,7 @@ def get_group_by_id(request: HttpRequest, id_: Optional[int] = None):
 
 
 @never_cache
-@permission_required("core.edit_group", fn=objectgetter_optional(Group, None, False))
+@permission_required("core.edit_group_rule", fn=objectgetter_optional(Group, None, False))
 def edit_group(request: HttpRequest, id_: Optional[int] = None) -> HttpResponse:
     """View to edit or create a group."""
     context = {}
@@ -435,7 +471,7 @@ def edit_group(request: HttpRequest, id_: Optional[int] = None) -> HttpResponse:
         edit_group_form = EditGroupForm(request.POST or None, instance=group)
     else:
         # Empty form to create a new group
-        if request.user.has_perm("core.create_group"):
+        if request.user.has_perm("core.create_group_rule"):
             edit_group_form = EditGroupForm(request.POST or None)
         else:
             raise PermissionDenied()
@@ -455,7 +491,7 @@ def edit_group(request: HttpRequest, id_: Optional[int] = None) -> HttpResponse:
     return render(request, "core/group/edit.html", context)
 
 
-@permission_required("core.manage_data")
+@permission_required("core.manage_data_rule")
 def data_management(request: HttpRequest) -> HttpResponse:
     """View with special menu for data management."""
     context = {}
@@ -466,7 +502,7 @@ class SystemStatus(PermissionRequiredMixin, MainView):
     """View giving information about the system status."""
 
     template_name = "core/pages/system_status.html"
-    permission_required = "core.view_system_status"
+    permission_required = "core.view_system_status_rule"
     context = {}
 
     def get(self, request, *args, **kwargs):
@@ -480,17 +516,22 @@ class SystemStatus(PermissionRequiredMixin, MainView):
                     TaskResult.objects.filter(task_name=job).order_by("date_done").last()
                 )
 
-        context = {"plugins": self.plugins, "status_code": status_code, "tasks": task_results}
+        context = {
+            "plugins": self.plugins,
+            "status_code": status_code,
+            "tasks": task_results,
+            "DEBUG": settings.DEBUG,
+        }
         return self.render_to_response(context, status=status_code)
 
 
 class TestPDFGenerationView(PermissionRequiredMixin, RenderPDFView):
     template_name = "core/pages/test_pdf.html"
-    permission_required = "core.test_pdf"
+    permission_required = "core.test_pdf_rule"
 
 
 @permission_required(
-    "core.mark_notification_as_read", fn=objectgetter_optional(Notification, None, False)
+    "core.mark_notification_as_read_rule", fn=objectgetter_optional(Notification, None, False)
 )
 def notification_mark_read(request: HttpRequest, id_: int) -> HttpResponse:
     """Mark a notification read."""
@@ -503,7 +544,7 @@ def notification_mark_read(request: HttpRequest, id_: int) -> HttpResponse:
     return redirect("index")
 
 
-@permission_required("core.view_announcements")
+@permission_required("core.view_announcements_rule")
 def announcements(request: HttpRequest) -> HttpResponse:
     """List view of announcements."""
     context = {}
@@ -517,7 +558,7 @@ def announcements(request: HttpRequest) -> HttpResponse:
 
 @never_cache
 @permission_required(
-    "core.create_or_edit_announcement", fn=objectgetter_optional(Announcement, None, False)
+    "core.create_or_edit_announcement_rule", fn=objectgetter_optional(Announcement, None, False)
 )
 def announcement_form(request: HttpRequest, id_: Optional[int] = None) -> HttpResponse:
     """View to create or edit an announcement."""
@@ -547,7 +588,7 @@ def announcement_form(request: HttpRequest, id_: Optional[int] = None) -> HttpRe
 
 
 @permission_required(
-    "core.delete_announcement", fn=objectgetter_optional(Announcement, None, False)
+    "core.delete_announcement_rule", fn=objectgetter_optional(Announcement, None, False)
 )
 def delete_announcement(request: HttpRequest, id_: int) -> HttpResponse:
     """View to delete an announcement."""
@@ -559,13 +600,16 @@ def delete_announcement(request: HttpRequest, id_: int) -> HttpResponse:
     return redirect("announcements")
 
 
-@permission_required("core.search")
+@permission_required("core.search_rule")
 def searchbar_snippets(request: HttpRequest) -> HttpResponse:
     """View to return HTML snippet with searchbar autocompletion results."""
     query = request.GET.get("q", "")
     limit = int(request.GET.get("limit", "5"))
-
-    results = SearchQuerySet().filter(text=AutoQuery(query))[:limit]
+    indexed_models = UnifiedIndex().get_indexed_models()
+    allowed_object_ids = get_allowed_object_ids(request, indexed_models)
+    results = (
+        SearchQuerySet().filter(id__in=allowed_object_ids).filter(text=AutoQuery(query))[:limit]
+    )
     context = {"results": results}
 
     return render(request, "search/searchbar_snippets.html", context)
@@ -574,13 +618,15 @@ def searchbar_snippets(request: HttpRequest) -> HttpResponse:
 class PermissionSearchView(PermissionRequiredMixin, SearchView):
     """Wrapper to apply permission to haystack's search view."""
 
-    permission_required = "core.search"
+    permission_required = "core.search_rule"
 
-    def create_response(self):
-        context = self.get_context()
-        if not self.has_permission():
-            return self.handle_no_permission()
-        return render(self.request, self.template, context)
+    def get_context_data(self, *, object_list=None, **kwargs):
+        queryset = object_list if object_list is not None else self.object_list
+        indexed_models = UnifiedIndex().get_indexed_models()
+        allowed_object_ids = get_allowed_object_ids(self.request, indexed_models)
+        queryset = queryset.filter(id__in=allowed_object_ids)
+
+        return super().get_context_data(object_list=queryset, **kwargs)
 
 
 @never_cache
@@ -599,21 +645,21 @@ def preferences(
         instance = request.site
         form_class = SitePreferenceForm
 
-        if not request.user.has_perm("core.change_site_preferences", instance):
+        if not request.user.has_perm("core.change_site_preferences_rule", instance):
             raise PermissionDenied()
     elif registry_name == "person":
         registry = person_preferences_registry
         instance = objectgetter_optional(Person, "request.user.person", True)(request, pk)
         form_class = PersonPreferenceForm
 
-        if not request.user.has_perm("core.change_person_preferences", instance):
+        if not request.user.has_perm("core.change_person_preferences_rule", instance):
             raise PermissionDenied()
     elif registry_name == "group":
         registry = group_preferences_registry
         instance = objectgetter_optional(Group, None, False)(request, pk)
         form_class = GroupPreferenceForm
 
-        if not request.user.has_perm("core.change_group_preferences", instance):
+        if not request.user.has_perm("core.change_group_preferences_rule", instance):
             raise PermissionDenied()
     else:
         # Invalid registry name passed from URL
@@ -647,7 +693,7 @@ def preferences(
     return render(request, "dynamic_preferences/form.html", context)
 
 
-@permission_required("core.delete_person", fn=objectgetter_optional(Person))
+@permission_required("core.delete_person_rule", fn=objectgetter_optional(Person))
 def delete_person(request: HttpRequest, id_: int) -> HttpResponse:
     """View to delete an person."""
     person = objectgetter_optional(Person)(request, id_)
@@ -662,7 +708,7 @@ def delete_person(request: HttpRequest, id_: int) -> HttpResponse:
     return redirect("persons")
 
 
-@permission_required("core.delete_group", fn=objectgetter_optional(Group))
+@permission_required("core.delete_group_rule", fn=objectgetter_optional(Group))
 def delete_group(request: HttpRequest, id_: int) -> HttpResponse:
     """View to delete an group."""
     group = objectgetter_optional(Group)(request, id_)
@@ -678,7 +724,7 @@ def delete_group(request: HttpRequest, id_: int) -> HttpResponse:
 
 @never_cache
 @permission_required(
-    "core.change_additionalfield", fn=objectgetter_optional(AdditionalField, None, False)
+    "core.change_additionalfield_rule", fn=objectgetter_optional(AdditionalField, None, False)
 )
 def edit_additional_field(request: HttpRequest, id_: Optional[int] = None) -> HttpResponse:
     """View to edit or create a additional_field."""
@@ -693,7 +739,7 @@ def edit_additional_field(request: HttpRequest, id_: Optional[int] = None) -> Ht
             request.POST or None, instance=additional_field
         )
     else:
-        if request.user.has_perm("core.create_additionalfield"):
+        if request.user.has_perm("core.create_additionalfield_rule"):
             # Empty form to create a new additional_field
             edit_additional_field_form = EditAdditionalFieldForm(request.POST or None)
         else:
@@ -712,7 +758,7 @@ def edit_additional_field(request: HttpRequest, id_: Optional[int] = None) -> Ht
     return render(request, "core/additional_field/edit.html", context)
 
 
-@permission_required("core.view_additionalfield")
+@permission_required("core.view_additionalfields_rule")
 def additional_fields(request: HttpRequest) -> HttpResponse:
     """List view for listing all additional fields."""
     context = {}
@@ -731,7 +777,7 @@ def additional_fields(request: HttpRequest) -> HttpResponse:
 
 
 @permission_required(
-    "core.delete_additionalfield", fn=objectgetter_optional(AdditionalField, None, False)
+    "core.delete_additionalfield_rule", fn=objectgetter_optional(AdditionalField, None, False)
 )
 def delete_additional_field(request: HttpRequest, id_: int) -> HttpResponse:
     """View to delete an additional field."""
@@ -743,7 +789,7 @@ def delete_additional_field(request: HttpRequest, id_: int) -> HttpResponse:
 
 
 @never_cache
-@permission_required("core.change_grouptype", fn=objectgetter_optional(GroupType, None, False))
+@permission_required("core.change_grouptype_rule", fn=objectgetter_optional(GroupType, None, False))
 def edit_group_type(request: HttpRequest, id_: Optional[int] = None) -> HttpResponse:
     """View to edit or create a group_type."""
     context = {}
@@ -771,7 +817,7 @@ def edit_group_type(request: HttpRequest, id_: Optional[int] = None) -> HttpResp
     return render(request, "core/group_type/edit.html", context)
 
 
-@permission_required("core.view_grouptype")
+@permission_required("core.view_grouptypes_rule")
 def group_types(request: HttpRequest) -> HttpResponse:
     """List view for listing all group types."""
     context = {}
@@ -787,7 +833,7 @@ def group_types(request: HttpRequest) -> HttpResponse:
     return render(request, "core/group_type/list.html", context)
 
 
-@permission_required("core.delete_grouptype", fn=objectgetter_optional(GroupType, None, False))
+@permission_required("core.delete_grouptype_rule", fn=objectgetter_optional(GroupType, None, False))
 def delete_group_type(request: HttpRequest, id_: int) -> HttpResponse:
     """View to delete an group_type."""
     group_type = objectgetter_optional(GroupType, None, False)(request, id_)
@@ -798,42 +844,44 @@ def delete_group_type(request: HttpRequest, id_: int) -> HttpResponse:
 
 
 class DataCheckView(PermissionRequiredMixin, ListView):
-    permission_required = "core.view_datacheckresults"
+    permission_required = "core.view_datacheckresults_rule"
     model = DataCheckResult
     template_name = "core/data_check/list.html"
     context_object_name = "results"
 
     def get_queryset(self) -> QuerySet:
-        return DataCheckResult.objects.filter(solved=False).order_by("check")
+        return (
+            DataCheckResult.objects.filter(content_type__app_label__in=apps.app_configs.keys())
+            .filter(solved=False)
+            .order_by("check")
+        )
 
-    def get_context_data(self, **kwargs: Any) -> Dict[str, Any]:
+    def get_context_data(self, **kwargs: Any) -> dict[str, Any]:
         context = super().get_context_data(**kwargs)
         context["registered_checks"] = DataCheckRegistry.data_checks
         return context
 
 
 class RunDataChecks(PermissionRequiredMixin, View):
-    permission_required = "core.run_data_checks"
+    permission_required = "core.run_data_checks_rule"
 
     def get(self, request: HttpRequest, *args, **kwargs) -> HttpResponse:
         result = check_data.delay()
 
-        context = {
-            "title": _("Progress: Run data checks"),
-            "back_url": reverse("check_data"),
-            "progress": {
-                "task_id": result.task_id,
-                "title": _("Run data checks …"),
-                "success": _("The data checks were run successfully."),
-                "error": _("There was a problem while running data checks."),
-            },
-        }
-        return render(request, "core/pages/progress.html", context)
+        return render_progress_page(
+            request,
+            result,
+            title=_("Progress: Run data checks"),
+            progress_title=_("Run data checks …"),
+            success_message=_("The data checks were run successfully."),
+            error_message=_("There was a problem while running data checks."),
+            back_url=reverse("check_data"),
+        )
 
 
 class SolveDataCheckView(PermissionRequiredMixin, RevisionMixin, DetailView):
     queryset = DataCheckResult.objects.all()
-    permission_required = "core.solve_data_problem"
+    permission_required = "core.solve_data_problem_rule"
 
     def get(self, request: HttpRequest, *args, **kwargs) -> HttpResponse:
         solve_option = self.kwargs["solve_option"]
@@ -860,10 +908,10 @@ class DashboardWidgetListView(PermissionRequiredMixin, SingleTableView):
 
     model = DashboardWidget
     table_class = DashboardWidgetTable
-    permission_required = "core.view_dashboardwidget"
+    permission_required = "core.view_dashboardwidget_rule"
     template_name = "core/dashboard_widget/list.html"
 
-    def get_context_data(self, **kwargs: Any) -> Dict[str, Any]:
+    def get_context_data(self, **kwargs: Any) -> dict[str, Any]:
         context = super().get_context_data(**kwargs)
         context["widget_types"] = [
             (ContentType.objects.get_for_model(m, False), m)
@@ -881,7 +929,7 @@ class DashboardWidgetEditView(PermissionRequiredMixin, AdvancedEditView):
 
     model = DashboardWidget
     fields = "__all__"
-    permission_required = "core.edit_dashboardwidget"
+    permission_required = "core.edit_dashboardwidget_rule"
     template_name = "core/dashboard_widget/edit.html"
     success_url = reverse_lazy("dashboard_widgets")
     success_message = _("The dashboard widget has been saved.")
@@ -897,7 +945,7 @@ class DashboardWidgetCreateView(PermissionRequiredMixin, AdvancedCreateView):
         ct = get_object_or_404(ContentType, app_label=app_label, model=model)
         return ct.model_class()
 
-    def get_context_data(self, **kwargs: Any) -> Dict[str, Any]:
+    def get_context_data(self, **kwargs: Any) -> dict[str, Any]:
         context = super().get_context_data(**kwargs)
         context["model"] = self.model
         return context
@@ -911,7 +959,7 @@ class DashboardWidgetCreateView(PermissionRequiredMixin, AdvancedCreateView):
         return super().post(request, *args, **kwargs)
 
     fields = "__all__"
-    permission_required = "core.add_dashboardwidget"
+    permission_required = "core.add_dashboardwidget_rule"
     template_name = "core/dashboard_widget/create.html"
     success_url = reverse_lazy("dashboard_widgets")
     success_message = _("The dashboard widget has been created.")
@@ -921,7 +969,7 @@ class DashboardWidgetDeleteView(PermissionRequiredMixin, AdvancedDeleteView):
     """Delete view for dashboard widgets."""
 
     model = DashboardWidget
-    permission_required = "core.delete_dashboardwidget"
+    permission_required = "core.delete_dashboardwidget_rule"
     template_name = "core/pages/delete.html"
     success_url = reverse_lazy("dashboard_widgets")
     success_message = _("The dashboard widget has been deleted.")
@@ -930,13 +978,13 @@ class DashboardWidgetDeleteView(PermissionRequiredMixin, AdvancedDeleteView):
 class EditDashboardView(PermissionRequiredMixin, View):
     """View for editing dashboard widget order."""
 
-    permission_required = "core.edit_dashboard"
+    permission_required = "core.edit_dashboard_rule"
 
     def get_context_data(self, request, **kwargs):
         context = {}
         self.default_dashboard = kwargs.get("default", False)
 
-        if self.default_dashboard and not request.user.has_perm("core.edit_default_dashboard"):
+        if self.default_dashboard and not request.user.has_perm("core.edit_default_dashboard_rule"):
             raise PermissionDenied()
 
         context["default_dashboard"] = self.default_dashboard
@@ -1132,6 +1180,61 @@ class GroupObjectPermissionDeleteView(PermissionRequiredMixin, AdvancedDeleteVie
     template_name = "core/pages/delete.html"
 
 
+class OAuth2ListView(PermissionRequiredMixin, ListView):
+    """List view for all the applications."""
+
+    permission_required = "core.view_oauthapplications_rule"
+    context_object_name = "applications"
+    template_name = "oauth2_provider/application/list.html"
+
+    def get_queryset(self):
+        return OAuthApplication.objects.all()
+
+
+class OAuth2DetailView(PermissionRequiredMixin, DetailView):
+    """Detail view for an application instance."""
+
+    context_object_name = "application"
+    permission_required = "core.view_oauthapplication_rule"
+    template_name = "oauth2_provider/application/detail.html"
+
+    def get_queryset(self):
+        return OAuthApplication.objects.all()
+
+
+class OAuth2DeleteView(PermissionRequiredMixin, AdvancedDeleteView):
+    """View used to delete an application."""
+
+    permission_required = "core.delete_oauthapplication_rule"
+    context_object_name = "application"
+    success_url = reverse_lazy("oauth2_applications")
+    template_name = "core/pages/delete.html"
+
+    def get_queryset(self):
+        return OAuthApplication.objects.all()
+
+
+class OAuth2EditView(PermissionRequiredMixin, AdvancedEditView):
+    """View used to edit an application."""
+
+    permission_required = "core.edit_oauthapplication_rule"
+    context_object_name = "application"
+    template_name = "oauth2_provider/application/edit.html"
+    form_class = OAuthApplicationForm
+
+    def get_queryset(self):
+        return OAuthApplication.objects.all()
+
+
+class OAuth2RegisterView(PermissionRequiredMixin, AdvancedCreateView):
+    """View used to register an application."""
+
+    permission_required = "core.create_oauthapplication_rule"
+    context_object_name = "application"
+    template_name = "oauth2_provider/application/create.html"
+    form_class = OAuthApplicationForm
+
+
 class RedirectToPDFFile(SingleObjectMixin, View):
     """Redirect to a generated PDF file."""
 
@@ -1144,13 +1247,70 @@ class RedirectToPDFFile(SingleObjectMixin, View):
         return redirect(file_object.file.url)
 
 
-class HTMLForPDFFile(SingleObjectMixin, View):
-    """Return rendered HTML for generating a PDF file."""
+class CeleryProgressView(View):
+    """Wrap celery-progress view to check permissions before."""
 
-    model = PDFFile
+    def get(self, request: HttpRequest, task_id: str, *args, **kwargs) -> HttpResponse:
+        if request.user.is_anonymous:
+            raise Http404()
+        if not TaskUserAssignment.objects.filter(
+            task_result__task_id=task_id, user=request.user
+        ).exists():
+            raise Http404()
+        return get_progress(request, task_id, *args, **kwargs)
 
-    def get(self, request, *args, **kwargs):
-        file_object = self.get_object()
-        if request.GET.get("secret") != file_object.secret:
-            raise PermissionDenied()
-        return HttpResponse(file_object.html)
+
+class CustomPasswordChangeView(PermissionRequiredMixin, PasswordChangeView):
+    """Custom password change view to allow to disable changing of password."""
+
+    permission_required = "core.can_change_password"
+
+    def __init__(self, *args, **kwargs):
+        if get_site_preferences()["auth__allow_password_change"]:
+            self.template_name = "account/password_change.html"
+        else:
+            self.template_name = "account/password_change_disabled.html"
+
+        super().__init__(*args, **kwargs)
+
+
+class SocialAccountDeleteView(DeleteView):
+    """Custom view to delete django-allauth social account."""
+
+    template_name = "core/pages/delete.html"
+    success_url = reverse_lazy("socialaccount_connections")
+
+    def get_queryset(self):
+        return SocialAccount.objects.filter(user=self.request.user)
+
+    def delete(self, request, *args, **kwargs):
+        self.object = self.get_object()
+        success_url = self.get_success_url()
+        try:
+            get_adapter(self.request).validate_disconnect(
+                self.object, SocialAccount.objects.filter(user=self.request.user)
+            )
+        except ValidationError:
+            messages.error(
+                self.request,
+                _(
+                    "The third-party account could not be disconnected "
+                    "because it is the only login method available."
+                ),
+            )
+        else:
+            self.object.delete()
+            messages.success(
+                self.request, _("The third-party account has been successfully disconnected.")
+            )
+        return HttpResponseRedirect(success_url)
+
+
+def server_error(
+    request: HttpRequest, template_name: str = ERROR_500_TEMPLATE_NAME
+) -> HttpResponseServerError:
+    """Ensure the request is passed to the error page."""
+    template = loader.get_template(template_name)
+    context = {"request": request}
+
+    return HttpResponseServerError(template.render(context))
diff --git a/dev.sh b/dev.sh
deleted file mode 100755
index fa77b4c47f287c48d7839801223bbeddac5c88e1..0000000000000000000000000000000000000000
--- a/dev.sh
+++ /dev/null
@@ -1,177 +0,0 @@
-#!/usr/bin/env mksh
-
-remove_pip_metadata() {
-    find . -type d -name pip-wheel-metadata -print0 | xargs -0r rm -rf --
-}
-
-case "$1" in
-    "install-all")
-	set -e
-	cd "$(dirname "$0")"
-	remove_pip_metadata
-	poetry lock
-	poetry install
-	for d in apps/official/*; do
-	    remove_pip_metadata
-	    poetry run sh -c "cd $d; poetry lock; poetry install"
-	done
-	remove_pip_metadata
-	poetry run ./manage.py compilemessages
-	poetry run ./manage.py yarn install
-	poetry run ./manage.py collectstatic --no-input
-	set +e
-	exit
-	;;
-    "makemessages")
-	cd "$(dirname "$0")"
-	manage_py=$(realpath manage.py)
-	locales="-l ar -l de_DE -l fr -l nb_NO -l tr_TR -l la"
-	for d in aleksis/core apps/official/*/aleksis/apps/*; do
-		echo; echo "Entering $d."
-		poetry run sh -c "cd $d; $manage_py makemessages --no-wrap -e html,txt,py,email -i static $locales"
-		poetry run sh -c "cd $d; $manage_py makemessages --no-wrap -d djangojs $locales"
-	done
-	exit
-	;;
-    "autopep8")
-	cd "$(dirname "$0")"
-	for d in aleksis/core apps/official/*/aleksis/apps/*; do
-		echo; echo "Entering $d."
-		poetry run sh -c "cd $d; autopep8 -i -r ."
-	done
-	exit
-	;;
-    "pylama")
-	cd "$(dirname "$0")"
-	tox_ini=$(realpath tox.ini)
-	for d in aleksis/core apps/official/*/aleksis/apps/*; do
-		echo; echo "Entering $d."
-		poetry run sh -c "cd $d; pylama -a -o $tox_ini ."
-	done
-	exit
-	;;
-    "gource")
-	for d in . apps/official/*; do
-		gource --output-custom-log - "$d"
-	done | sort -n | gource --log-format custom --background-image aleksis/core/static/img/aleksis-icon.png "$@" -
-	exit
-	;;
-
-    "devstats-commits")
-	# Copyright © 2018
-	#	mirabilos <m@mirbsd.org>
-	# Copyright © 2017
-	#	mirabilos <t.glaser@tarent.de>
-	# Copyright © 2015, 2017, 2020
-	#	mirabilos <thorsten.glaser@teckids.org>
-	#
-	# Provided that these terms and disclaimer and all copyright notices
-	# are retained or reproduced in an accompanying document, permission
-	# is granted to deal in this work without restriction, including un‐
-	# limited rights to use, publicly perform, distribute, sell, modify,
-	# merge, give away, or sublicence.
-	#
-	# This work is provided “AS IS” and WITHOUT WARRANTY of any kind, to
-	# the utmost extent permitted by applicable law, neither express nor
-	# implied; without malicious intent or gross negligence. In no event
-	# may a licensor, author or contributor be held liable for indirect,
-	# direct, other damage, loss, or other issues arising in any way out
-	# of dealing in the work, even if advised of the possibility of such
-	# damage or existence of a defect, except proven that it results out
-	# of said person’s immediate fault when using the work as intended.
-
-	set -e
-	set -o pipefail
-	unset LANGUAGE
-	export LC_ALL=C.UTF-8
-	set -o utf8-mode
-
-	for d in . apps/official/*; do
-		cd "$d"
-		if [[ ! -s pyproject.toml ]]; then
-			print -ru2 "E: missing pyproject.toml in ${d@Q}"
-			print -ru2 "N: maybe you forgot the submodules?"
-			print -ru2 "N: try git submodule update --init --recursive"
-			exit 1
-		fi
-		cd "$OLDPWD"
-	done
-	for d in . apps/official/*; do
-		cd "$d"
-		git log --pretty=tformat:%aN
-		cd "$OLDPWD"
-	done | sort | uniq -c | sort -nr |&
-	maxnum=0
-	maxlen=0
-	set -A nums
-	set -A names
-	nlines=0
-	while IFS= read -pr line; do
-		line=${line##*( )}
-		num=${line%% *}
-		line=${line##+([0-9]) }
-		#print -r -- "<$num><$line>"
-		(( maxnum = num > maxnum ? num : maxnum ))
-		len=${%line}
-		if (( len == -1 )); then
-			len=${#line}
-			print -ru2 -- "W: assuming length $len for author ${line@Q}"
-		fi
-		(( maxlen = len > maxlen ? len : maxlen ))
-		nums[nlines]=$num
-		names[nlines++]=$line
-	done
-	w=$COLUMNS
-	if (( (w -= 1 + maxlen + 1) < 1 )); then
-		print -ru2 -- "E: terminal too small, need $((-w+1)) more columns"
-		exit 1
-	fi
-	if (( maxnum < 1 )); then
-		print -ru2 -- "E: no commits"
-		exit 1
-	fi
-	set +e
-	typeset -R$maxlen pname
-	mbar=██
-	nlen=0
-	num=$maxnum
-	while ((# num > 0 )); do
-		mbar+=â–ˆ
-		((# ++nlen ))
-		((# num /= 10 ))
-	done
-	typeset -R$nlen pnum
-	print '\e[0m'
-	line=-1
-	while (( ++line < nlines )); do
-		bar=
-		((# num = (nums[line] * w * 8) / maxnum ))
-		while ((# num >= 8 )); do
-			bar+=â–ˆ
-			((# num -= 8 ))
-		done
-		case $num {
-		(7) bar+=â–‰ ;;
-		(6) bar+=â–Š ;;
-		(5) bar+=â–‹ ;;
-		(4) bar+=▌ ;;
-		(3) bar+=▍ ;;
-		(2) bar+=â–Ž ;;
-		(1) bar+=▏ ;;
-		}
-		pname=${names[line]}
-		if [[ $bar = "$mbar"* ]]; then
-			pnum=${nums[line]}
-			bar=$'\e[7m '$pnum$' \e[0m'${bar#"$mbar"}
-		else
-			bar+=" ${nums[line]}"
-		fi
-		print -r -- "$pname $bar"
-	done
-	exit
-	;;
-    *)
-	print -ru2 -- "E: unknown command ${1@Q}"
-	exit 1
-	;;
-esac
diff --git a/docs/admin/01_install.rst b/docs/admin/01_install.rst
new file mode 100644
index 0000000000000000000000000000000000000000..5251a054be53d1e587fbadc0c9c4b848d25df249
--- /dev/null
+++ b/docs/admin/01_install.rst
@@ -0,0 +1,254 @@
+Install AlekSIS
+===============
+
+From PyPI
+---------
+
+In this section we will install AlekSIS with `uWSGI` and `nGINX` on Debian
+bullseye.
+
+Filesystem locations
+~~~~~~~~~~~~~~~~~~~~
+
+AlekSIS will need and use the following paths:
+
+ * `/etc/aleksis` for configuration files
+ * `/var/lib/aleksis/media` for file storage (Django media)
+ * `/var/backups/aleksis` for backups of database and media files
+ * `/usr/local/share/aleksis/static` for static files
+ * `/usr/local/share/aleksis/node_modules` for frontend dependencies
+
+You can change any of the paths as you like.
+
+Prerequisites
+~~~~~~~~~~~~~
+
+For an installation on a dedicated server, the following prerequisites are needed:
+
+ * Debian 11
+ * PostgreSQL
+ * Redis
+ * uWSGI
+ * nginx
+ * Python 3.9
+ * Some system dependencies to build Python modules and manage frontend files
+ * The aforementioned paths
+
+Install system packages
+~~~~~~~~~~~~~~~~~~~~~~~
+
+Install some packages from the Debian package system.
+
+.. code-block:: shell
+
+   apt install uwsgi \
+               uwsgi-plugin-python3 \
+               nginx-full \
+               python3 \
+               python3-dev \
+               libldap2-dev \
+               libpq-dev \
+               libsasl2-dev \
+               yarnpkg \
+               python3-virtualenv \
+               chromium \
+               redis-server \
+               postgresql
+
+Create PostgreSQL user and database
+~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
+
+Generate a secure password for the database, then create the user and database.
+
+.. code-block:: shell
+
+   sudo -u postgres createuser -D -P -R -S aleksis
+   sudo -u postgres createdb -E UTF-8 -O aleksis -T template0 -l C.UTF-8 aleksis
+
+When asked, use the password generated above.
+
+Create the directories for storage
+~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
+
+.. code-block:: shell
+
+   mkdir -p /etc/aleksis \
+            /usr/share/aleksis/{static,node_modules} \
+            /var/lib/aleksis/media \
+            /var/backups/aleksis
+   chown -R www-data:www-data /var/lib/aleksis
+
+Create AlekSIS configuration file
+~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
+
+AlekSIS is configured in files in `/etc/aleksis`. Create a basic configuration file
+for the environment defined above by opening `/etc/aleksis/aleksis.toml` with your
+favourite text editor and adding the following configuration.
+
+.. code-block:: toml
+
+   static = { root = "/usr/local/share/aleksis/static", url = "/static/" }
+   media = { root = "/var/lib/aleksis/media", url = "/media/" }
+   node_modules = { root = "/usr/local/share/aleksis/node_modules" }
+   secret_key = "SomeRandomValue"
+
+   [http]
+   allowed_hosts = ["aleksis.example.com"]
+
+   [database]
+   host = "localhost"
+   name = "aleksis"
+   username = "aleksis"
+   password = "password_generated_above"
+
+   [backup]
+   location = "/var/backups/aleksis"
+
+   [auth.superuser]
+   username = "admin"
+   password = "admin"
+   email = "root@localhost"
+
+Install AlekSIS itself
+~~~~~~~~~~~~~~~~~~~~~~
+
+To install AlekSIS now, and run all post-install tasks, run the following commands.
+They will pull the AlekSIS standard distribution from `PyPI`_ and install it to the
+system-wide `dist-packages` of Python. Afterwards, it will download frontend dependencies
+from `yarnpkg`, collect static files, and migrate the database to the final schema.
+
+.. code-block:: shell
+
+   pip3 install aleksis
+   aleksis-admin yarn install
+   aleksis-admin collectstatic
+   aleksis-admin migrate
+   aleksis-admin createinitialrevisions
+
+Configure uWSGI
+~~~~~~~~~~~~~~~
+
+uWSGI is an application server that will manage the server processes and requests.
+It will also run the Celery broker and scheduler for you.
+
+Configure a uWSGI app by opening `/etc/uwsgi/apps-available/aleksis.ini` in an
+editor and inserting:
+
+.. code-block:: toml
+
+   [uwsgi]
+   vhost = true
+   plugins = python3
+   master = true
+   enable-threads = true
+   processes = 20
+   wsgi-file = /usr/local/lib/python3.9/dist-packages/aleksis/core/wsgi.py
+   chdir = /var/lib/aleksis
+   lazy = true
+   lazy-apps = true
+   attach-daemon = celery -A aleksis.core worker --concurrency=4
+   attach-daemon = celery -A aleksis.core beat
+
+Afterwards, enable the app using:
+
+.. code-block:: shell
+
+   ln -s /etc/uwsgi/apps-available/aleksis.ini /etc/uwsgi/apps-enabled/aleksis.ini
+   service uwsgi restart
+
+Configure the nginx webserver
+~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
+
+First, you should get a TLS certificate, e.g. by using `Let's Encrypt`_.
+
+Then, create a virtual host in nginx, by editing `/etc/nginx/sites-available/aleksis.example.com`.
+
+.. code-block:: nginx
+   upstream aleksis {
+     server unix:///run/uwsgi/app/aleksis/socket;
+   }
+
+   server {
+     listen 80;
+     listen [::]:80;
+
+     server_name aleksis.example.com;
+
+     return 301 https://$server_name$request_uri;
+   }
+
+   server {
+     listen 443 ssl http2;
+     listen [::]:443 ssl http2;
+
+     ssl_certificate /etc/letsencrypt/certs/aleksis.example.com/fullchain.pem;
+     ssl_certificate_key /etc/letsencrypt/certs/aleksis.example.com/privkey.pem;
+     ssl_trusted_certificate /etc/letsencrypt/certs/aleksis.example.com/chain.pem;
+
+     server_name aleksis.example.com;
+
+     access_log /var/log/nginx/access.log;
+
+     location /static {
+       alias /usr/local/share/aleksis/static;
+     }
+
+     location / {
+       uwsgi_pass aleksis;
+       include uwsgi_params;
+       proxy_redirect off;
+       proxy_pass_header Authorization;
+     }
+   }
+
+Enable the virtual host:
+
+.. code-block:: shell
+
+   ln -s /etc/nginx/sites-available/aleksis.example.com /etc/nginx/sites-enabled/aleksis.example.com
+   service nginx restart
+
+Finalisation
+~~~~~~~~~~~~
+
+Your AlekSIS installation should now be reachable and you can login with the administration
+account configured above.
+
+With Docker
+-----------
+
+AlekSIS can also be installed using Docker, either only AlekSIS itself, or the
+full stack, including Redis, using docker-compose
+
+Full stack using docker-compose
+~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
+
+First, install Docker and docker-compose on your system. Also install git
+to get the docker-compose file and image definition.
+
+.. code-block:: shell
+
+   apt install docker.io docker-compose git
+
+Now, clone the distribution repository, which contains the docker-compose
+file.
+
+.. code-block:: shell
+
+   git clone https://edugit.org/AlekSIS/official/AlekSIS
+
+You should review the file `docker-compose.yaml` for any environment variables
+you want to change.
+
+Finally, bring the stack up using:
+
+.. code-block:: shell
+
+  docker-compose up -d
+
+AlekSIS will be reachable on port 80 if you forgot to configure the environment.
+You are responsible for adding a reverse proxy like nginx providing TLS, etc.
+
+.. _Dynaconf: https://dynaconf.readthedocs.io/en/latest/
+.. _Let's Encrypt: https://certbot.eff.org/instructions
+.. _PyPI: https://pypi.org
diff --git a/docs/admin/02_ldap.rst b/docs/admin/02_ldap.rst
index 5ef33259b580fc68909c58f4b4a8b1fd507fcd4b..b8fbd657dedc3d3e7a6ef04a7fa36d9b7f1b66bf 100644
--- a/docs/admin/02_ldap.rst
+++ b/docs/admin/02_ldap.rst
@@ -5,24 +5,24 @@ AlekSIS can authenticate users against an LDAP directory (like OpenLDAP or
 Active Directory). The AlekSIS core can only authenticate and synchronise
 authenticated users to AlekSIS’ database. There are apps that help with
 tasks like mass-importing accounts and linking accounts to persons in
-the BiscuIY system (see below).
+the AlekSIS system (see below).
 
 
 Installing packages for LDAP support
 ------------------------------------
 
 Installing the necessary librairies for LDAP support unfortunately is not
-very straightforward under all circumstances.
+very straightforward under all circumstances. On Debian, install these packages::
 
-TBA.
+  sudo apt install python3-ldap libldap2-dev libssl-dev libsasl2-dev python3-dev
 
 
 Configuration of LDAP support
 -----------------------------
 
-Configuration is done under the `default.ldap` section in AlekSIS’
+Configuration is done under the ``default.ldap`` section in AlekSIS’
 configuration file. For example, add something like the following to your
-configuration (normally in `/etc/aleksis`; you can either append to an
+configuration (normally in ``/etc/aleksis``; you can either append to an
 existing file or add a new one)::
 
   [ldap]
@@ -32,3 +32,9 @@ existing file or add a new one)::
   [ldap.users]
   search = { base = "ou=people,dc=myschool,dc=edu", filter = "(uid=%(user)s)" }
   map = { first_name = "givenName", last_name = "sn", email = "mail" }
+
+  [ldap.groups]
+  search = { base = "ou=groups,dc=myschool,dc=edu" }
+  type = "groupOfNames"
+  # Users in group "admins" are superusers
+  flags = { is_superuser = "cn=admins,ou=groups,dc=myschool,dc=edu" }
diff --git a/docs/admin/03_psql.rst b/docs/admin/03_psql.rst
deleted file mode 100644
index 6d5ab57ae3b157ca906f3cfb7df1d3897efdfbd7..0000000000000000000000000000000000000000
--- a/docs/admin/03_psql.rst
+++ /dev/null
@@ -1,41 +0,0 @@
-Installing AlekSIS with PostgreSQL backend
-==========================================
-
-PostgreSQL is the only supported database backend for AlekSIS. If you are
-installing AlekSIS manually, you need to properly set it up.
-
-Install the PostgreSQL server
------------------------------
-
-On Debian, install the postgresql server package with::
-
-  sudo apt install postgresql
-
-
-Create a database and user
---------------------------
-
-On Debian, you can use the following commands to create the database and a
-user who owns it::
-
-  sudo -u postgres createuser -D -P -R -S aleksis
-  sudo -u postgres createdb -E UTF-8 -O aleksis -T template0 -l C.UTF-8 aleksis
-
-When asked for the database user password, choose a secure, preferrably
-random, password. You can generate one using the pwgen utility if you like::
-
-  pwgen 16 1
-
-
-Configure AlekSIS to use PostgreSQL
------------------------------------
-
-Fill in the configuration under `/etc/aleksis/aleksis.toml` (or a file with any other name in this directory)::
-
-  [database]
-  host = "localhost"
-  name = "aleksis"
-  username = "aleksis"
-  password = "Y0urV3ryR4nd0mP4ssw0rd"
-
-Don't forget to run the migrations, like described in the basic setup guide.
diff --git a/docs/admin/05_configuration_options.rst b/docs/admin/05_configuration_options.rst
new file mode 100644
index 0000000000000000000000000000000000000000..19321db371728406086e35e3a4d7dd1e6979f367
--- /dev/null
+++ b/docs/admin/05_configuration_options.rst
@@ -0,0 +1,68 @@
+Configuration options
+=====================
+
+AlekSIS provides lots of options to configure your instance.
+
+Configuration file
+------------------
+
+All settings which are required for running an AlekSIS instance are stored in your configuration file ``/etc/aleksis/aleksis.toml``.
+
+Example configuration file::
+
+    # General config for static, media and secret key, required
+    static = { root = "/srv/www/aleksis/data/static", url = "/static/" }
+    media = { root = "/srv/www/aleksis/data/media", url = "/media/" }
+    secret_key = "Xoc8eiwah3neehid2Xi3oomoh4laem"
+
+    # Admin contat, optional
+    [contact]
+    admins = [["AlekSIS - Admins", "root@example.com"]]
+    from = 'aleksis@example.com'
+
+    # Allowed hosts, required
+    [http]
+    allowed_hosts = ["localhost"]
+
+    # Database for whole AlekSIS data, required
+    [database]
+    host = "localhost"
+    name = "aleksis"
+    username = "aleksis"
+    password = "aleksis"
+
+    # Maintenance mode and debug, optional
+    [maintenance]
+    debug = true
+
+    # Two factor authentication with yubikey enabled, optional
+    [2fa]
+    enabled = true
+    yubikey = { enabled = true }
+
+    # Authentication via LDAP, optional
+    [ldap]
+    uri = "ldaps://ldap.myschool.edu"
+    bind = { dn = "cn=reader,dc=myschool,dc=edu", password = "secret" }
+    map = { first_name = "givenName", last_name = "sn", email = "mail" }
+
+    [ldap.users]
+    search = { base = "ou=people,dc=myschool,dc=edu", filter = "(uid=%(user)s)" }
+
+    [ldap.groups]
+    search = { base = "ou=groups,dc=myschool,dc=edu" }
+    type = "groupOfNames"
+    # Users in group "admins" are superusers
+    flags = { is_superuser = "cn=admins,ou=groups,dc=myschool,dc=edu" }
+
+    # Search index, optional
+    [search]
+    backend = "whoosh"
+    index = "/srv/www/aleksis/data/whoosh_index"
+
+Configuration in frontend
+-------------------------
+
+Everything that must not be configured before the AlekSIS instance fully starts can be configured in frontend, such as site title and logo.
+
+You can find the configuration options in your AlekSIS instance under ``Admin → Configuration``.
diff --git a/docs/conf.py b/docs/conf.py
index 5817b0ad2b1195aeb704d7561fa57aec5a6d880a..a6a811a1566cdd11c80a5af40657992669160930 100644
--- a/docs/conf.py
+++ b/docs/conf.py
@@ -29,9 +29,9 @@ copyright = "2019, 2020, AlekSIS team"
 author = "AlekSIS team"
 
 # The short X.Y version
-version = "2.0"
+version = "2.2"
 # The full version, including alpha/beta/rc tags
-release = "2.0a1"
+release = "2.2.dev0"
 
 
 # -- General configuration ---------------------------------------------------
@@ -84,7 +84,7 @@ pygments_style = None
 # The theme to use for HTML and HTML Help pages.  See the documentation for
 # a list of builtin themes.
 #
-html_theme = "alabaster"
+html_theme = "sphinx_materialdesign_theme"
 
 # Theme options are theme-specific and customize the look and feel of a theme
 # further.  For a list of options available for each theme, see the
diff --git a/docs/dev/01_setup.rst b/docs/dev/01_setup.rst
index 122a9baa14ce942886bd8d1a73fc5584f77cad19..dd3121a822e53c603b21511b790f5ef6950f295a 100644
--- a/docs/dev/01_setup.rst
+++ b/docs/dev/01_setup.rst
@@ -11,22 +11,41 @@ framework and selected apps.
 
 Also, `Yarn`_ is needed to resolve JavaScript dependencies.
 
+For repository management, `myrepos` is required.
+
+Setup database and message broker
+---------------------------------
+
+AlekSIS requires `PostgreSQL`_ (version 13 or newer) as database
+backend. To provide a database names `aleksis` with a user named
+`aleksis` on Debian::
+
+  sudo apt install postgresql-13
+  sudo -u postgres createuser -P aleksis
+  sudo -u postgres createdb -O aleksis aleksis
+
+Additionally, `Redis`_ is used as message broker and for caching.
+The default configuration of the server in Debian is sufficient::
+
+  sudo apt install redis-server
+
 Get the source tree
 -------------------
 
 To download AlekSIS and all officially bundled apps in their
 development version, use Git like so::
 
-  git clone --recurse-submodules https://edugit.org/AlekSIS/AlekSIS
+  git clone https://edugit.org/AlekSIS/official/AlekSIS
 
-If you do not want to download the bundled apps, leave out the
-``--recurse-submodules`` option.
+This first downloads a meta repository that contains a config file for mr.
+To clone the AlekSIS-Core and all official (and onboarding) apps, run::
 
+  mr update
 
 Install native dependencies
 ---------------------------
 
-Some system libraries are required to install AlekSIS::
+Some system libraries are required to install AlekSIS. On Debian, for example, this would be done with::
 
   sudo apt install build-essential libpq-dev libpq5 libssl-dev python3-dev python3-pip python3-venv yarnpkg gettext chromium
 
@@ -36,7 +55,7 @@ Get Poetry
 Make sure to have Poetry installed like described in its
 documentation. Right now, we encourage using pip to install Poetry
 once system-wide (this will change once distributions pick up
-Poetry). On Debian, for example, this would be done with::
+Poetry).::
 
   sudo pip3 install poetry
 
@@ -47,10 +66,15 @@ Install AlekSIS in its own virtual environment
 ----------------------------------------------
 
 Poetry will automatically manage virtual environments per project, so
-installing AlekSIS is a matter of::
+installing AlekSIS is a matter of switching into the Core's directory and running the initial AlekSIS installation::
 
+  cd apps/official/AlekSIS-Core
   poetry install
 
+Now it's recommended to run a shell that uses the newly created venv::
+
+  poetry shell
+
 
 Regular tasks
 -------------
@@ -63,13 +87,13 @@ some maintenance tasks need to be done:
 3. Run database migrations
 
 All three steps can be done with the ``poetry run`` command and
-``manage.py``::
+``aleksis-admin``::
 
-  poetry run ./manage.py yarn install
-  poetry run ./manage.py collectstatic
-  poetry run ./manage.py compilemessages
-  poetry run ./manage.py migrate
-  poetry run ./manage.py createinitialrevisions
+  poetry run aleksis-admin yarn install
+  poetry run aleksis-admin collectstatic
+  poetry run aleksis-admin compilemessages
+  poetry run aleksis-admin migrate
+  poetry run aleksis-admin createinitialrevisions
 
 (You might need database settings for the `migrate` command; see below.)
 
@@ -78,13 +102,13 @@ Running the development server
 
 The development server can be started using Django's ``runserver`` command.
 If you want to automatically start other necessary tools in development,
-like the `Celery` worker and scheduler, use ``runuwsgi`` instead.
+like the `Celery`_ worker and scheduler, use ``runuwsgi`` instead.
 You can either configure AlekSIS like in a production environment, or pass
 basic settings in as environment variable. Here is an example that runs the
 development server against a local PostgreSQL database with password
 `aleksis` (all else remains default) and with the `debug` setting enabled::
 
-  ALEKSIS_debug=true ALEKSIS_database__password=aleksis poetry run ./manage.py runuwsgi
+  ALEKSIS_debug=true ALEKSIS_database__password=aleksis poetry run aleksis-admin runuwsgi
 
 .. figure:: /screenshots/index.png
    :scale: 50%
@@ -96,4 +120,6 @@ development server against a local PostgreSQL database with password
 .. _Poetry: https://poetry.eustace.io/
 .. _Poetry installation methods: https://poetry.eustace.io/docs/#installation
 .. _Yarn: https://yarnpkg.com
+.. _PostgreSQL: https://www.postgresql.org/
+.. _Redis: https://redis.io/
 .. _Celery: https://celeryproject.org/
diff --git a/docs/dev/02_install_apps.rst b/docs/dev/02_install_apps.rst
index db3fe87f6ff6407ad0ca0db0aec0177720be70dd..bbfbb9b4007e533c1e910ae829ddf80f382b7d99 100644
--- a/docs/dev/02_install_apps.rst
+++ b/docs/dev/02_install_apps.rst
@@ -4,18 +4,18 @@ Installing apps into development environment
 Officially bundled apps
 -----------------------
 
-Officially bundlede apps are available in the ``apps/official/``
-sub-folder as Git submodules. If you followed the documentation, they
+Officially bundled apps are available in the ``apps/official/``
+sub-folder of the meta repository. If you followed the documentation, they
 will already be checked out in the version required for the bundle you
 are running.
 
-Installing apps into the existing virtual environment is a bit awkward::
+Installing apps into the existing virtual environment of `AlekSIS-Core` can
+be easily done after starting `poetry shell`::
 
-  poetry run sh -c "cd apps/official/AlekSIS-App-Exlibris; poetry install"
-
-This will install the Exlibris app (library management) app by using a
-shell for first ``cd``'ing into the app directory and then using
-poetry to install the app.
+  poetry install
 
 Do not forget to run the maintenance tasks described earlier after
 installing any app.
+
+**Heads up:** This is not suitable for working on the core, because it
+will install the `AlekSIS-Core` version used by the app using `pip` again.
diff --git a/docs/dev/04_materialize_templates.rst b/docs/dev/04_materialize_templates.rst
new file mode 100644
index 0000000000000000000000000000000000000000..34d582166caa4688ad5e6b2e841d082a4dfd649c
--- /dev/null
+++ b/docs/dev/04_materialize_templates.rst
@@ -0,0 +1,93 @@
+Materialize templates
+======================
+
+AlekSIS frontend uses the `MaterializeCSS`_ framework and the `django-material`_ library.
+
+Internationalization
+--------------------
+
+Load the ``i18n`` template tag and start translating strings in templates with
+the following template tags::
+
+    {% blocktrans %}String{% endblocktrans %}
+    {% trans "String" %}
+
+``{% blocktrans %}`` is mostly used for multiple words or multiline, while ``{%
+trans %}`` is used for single words.
+
+Title and headlines
+-------------------
+
+To add a main headline or browser title to your template, you can add the
+following blocks to your template::
+
+    {% block browser_title %}Title{% endblock %}
+    {% block page_title %}Headline{% endblock %}
+
+To fully remove page or browser title, use these template tags::
+
+    {% block no_browser_title %}{% endblock %}
+    {% block no_page_title %}{% endblock %}
+
+Forms in templates
+------------------
+
+The django MaterializeCSS integrations provides support for forms in
+template.
+
+You just have to load the ``material_form`` templatetag in the ``{% load %}``
+block.
+
+The following snippet generates the form::
+
+    <form method="post" enctype="multipart/form-data">
+        {% csrf_token %}
+        {% form form=edit_person_form %}{% endform %}
+        {% include "core/partials/save_button.html" %}
+    </form>
+
+``edit_person_form`` is the variable name of the form in your ``context``.
+
+``{% include "core/partials/save_button.html" %}`` includes a template snippet
+from AlekSIS core.  You can modify the buttons icon and translatable caption
+like this::
+
+    {% trans "Edit" as caption %}
+    {% include "core/partials/save_button.html" with caption=caption, icon="person" %}
+
+
+In your ``forms.py`` you can configure the layout of the fields like in the EditPersonForm::
+
+    class EditPersonForm(ExtensibleForm):
+    """Form to edit an existing person object in the frontend."""
+
+    layout = Layout(
+        Fieldset(
+            _("Base data"),
+            "short_name",
+            Row("user", "primary_group"),
+            "is_active",
+            Row("first_name", "additional_name", "last_name"),
+        ),
+        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"), "guardians",
+        ),
+    )
+
+Tables in templates
+-------------------
+
+To display tables generated by ``django-tables2`` in your template, you have to load the ``render_table`` template tag from ``django_tables2``::
+
+    {% load render_table from django_tables2 %}
+
+After you've loaded the template tag, you can simply generate the table like this::
+
+    {% render_table persons_table %}
+
+``persons_table`` is the variable name of the table in your ``context``.
+
+.. _MaterializeCSS: https://materializecss.com/
+.. _django-material: https://pypi.org/project/django-material/
diff --git a/docs/dev/05_extensible_models.rst b/docs/dev/05_extensible_models.rst
new file mode 100644
index 0000000000000000000000000000000000000000..1e291b1a7d6c8d730020b755d891f5a2a19b99c5
--- /dev/null
+++ b/docs/dev/05_extensible_models.rst
@@ -0,0 +1,11 @@
+Extensible models
+=================
+
+In AlekSIS you can use ``ExtensibleModels`` to add model fields to other
+apps models.
+
+If you want to make your apps models extensible, use the ``ExtensibleModel``
+class as parent class of your models.
+
+.. automodule:: aleksis.core.mixins
+   :members:
diff --git a/docs/dev/06_merging_app_settings.rst b/docs/dev/06_merging_app_settings.rst
new file mode 100644
index 0000000000000000000000000000000000000000..5c8bc1f65cced9153946bf36fbd005740e5fabb7
--- /dev/null
+++ b/docs/dev/06_merging_app_settings.rst
@@ -0,0 +1,29 @@
+Merging of app settings
+=======================
+
+AlekSIS provides features to merge app settings into main ``settings.py``.
+
+Currently mergable settings
+---------------------------
+
+ * INSTALLED_APPS
+ * DATABASES
+ * YARN_INSTALLED_APPS
+ * ANY_JS
+
+If you want to add another database for your AlekSIS app, you have to add
+the following into your ``settings.py``::
+
+    DATABASES = {
+        "database": {
+            "ENGINE": "django.db.backends.postgresql",
+            "NAME": "database",
+            "USER": "database",
+            "PASSWORD": "Y0urV3ryR4nd0mP4ssw0rd",
+            "HOST": "127.0.0.1",
+            "PORT": 5432,
+        }
+
+If you install new apps and want to configure these, or need some other settings you can easily add
+settings to your ``settings.py``.  Only settings that does not exist in the
+main ``settings.py`` will be respected.
diff --git a/docs/index.rst b/docs/index.rst
index 11863e5e9bd4590693add06b465a73ca3152e14c..aded3b28d2e41186b0f871d0064627bc807a7a29 100644
--- a/docs/index.rst
+++ b/docs/index.rst
@@ -3,8 +3,8 @@
    You can adapt this file completely to your liking, but it should at least
    contain the root `toctree` directive.
 
-Welcome to AlekSIS’ documentation!
-==================================
+Welcome to AlekSIS-Core's documentation!
+========================================
 
 .. toctree::
    :maxdepth: 2
diff --git a/docs/ref/core/01_checks.rst b/docs/ref/core/01_checks.rst
new file mode 100644
index 0000000000000000000000000000000000000000..ef096f388bf270049197cf2ffe66e45956edb009
--- /dev/null
+++ b/docs/ref/core/01_checks.rst
@@ -0,0 +1,5 @@
+Checks
+======
+
+.. automodule:: aleksis.core.checks
+        :members:
diff --git a/docs/ref/core/02_managers.rst b/docs/ref/core/02_managers.rst
new file mode 100644
index 0000000000000000000000000000000000000000..23248579f12e5a5e62ce8dccdb1a129b30fe0598
--- /dev/null
+++ b/docs/ref/core/02_managers.rst
@@ -0,0 +1,5 @@
+Managers
+========
+
+.. automodule:: aleksis.core.managers
+   :members:
diff --git a/docs/ref/core/03_mixins.rst b/docs/ref/core/03_mixins.rst
new file mode 100644
index 0000000000000000000000000000000000000000..5933fb3e1791b70cacfcc97cd3b25e1b8344f2cb
--- /dev/null
+++ b/docs/ref/core/03_mixins.rst
@@ -0,0 +1,5 @@
+Mixins
+======
+
+.. automodule:: aleksis.core.mixins
+   :members:
diff --git a/docs/ref/core/01_models.rst b/docs/ref/core/04_models.rst
similarity index 100%
rename from docs/ref/core/01_models.rst
rename to docs/ref/core/04_models.rst
diff --git a/docs/ref/core/05_registries.rst b/docs/ref/core/05_registries.rst
new file mode 100644
index 0000000000000000000000000000000000000000..97d2137617478b7bf8cf23e3c5751f596255aa6b
--- /dev/null
+++ b/docs/ref/core/05_registries.rst
@@ -0,0 +1,5 @@
+Core preference registries
+==========================
+
+.. automodule:: aleksis.core.registries
+   :members:
diff --git a/docs/ref/core/06_search_indexes.rst b/docs/ref/core/06_search_indexes.rst
new file mode 100644
index 0000000000000000000000000000000000000000..ba9944ace9a203d10b9e15a58177ce18a7cc83bb
--- /dev/null
+++ b/docs/ref/core/06_search_indexes.rst
@@ -0,0 +1,5 @@
+Search indexes
+==============
+
+.. automodule:: aleksis.core.search_indexes
+        :members:
diff --git a/docs/ref/core/07_tables.rst b/docs/ref/core/07_tables.rst
new file mode 100644
index 0000000000000000000000000000000000000000..ee4eeec7ac2db851bcd733588eeea271a0a95594
--- /dev/null
+++ b/docs/ref/core/07_tables.rst
@@ -0,0 +1,5 @@
+Core tables
+===========
+
+.. automodule:: aleksis.core.tables
+        :members:
diff --git a/docs/ref/core/08_tasks.rst b/docs/ref/core/08_tasks.rst
new file mode 100644
index 0000000000000000000000000000000000000000..76a3a430f924f41784d382a0b42a208799f18640
--- /dev/null
+++ b/docs/ref/core/08_tasks.rst
@@ -0,0 +1,5 @@
+Core Celery tasks
+=================
+
+.. automodule:: aleksis.core.tasks
+        :members:
diff --git a/docs/ref/core/09_utils.rst b/docs/ref/core/09_utils.rst
new file mode 100644
index 0000000000000000000000000000000000000000..74f6f9311644bfd168abfbe0cc4a9db0121a1620
--- /dev/null
+++ b/docs/ref/core/09_utils.rst
@@ -0,0 +1,43 @@
+Core utillity functions
+=======================
+
+
+General helper functions in core
+---------------------------------
+    .. automodule:: aleksis.core.util.core_helpers
+        :members:
+
+Predicates for permission systemd
+---------------------------------
+.. automodule:: aleksis.core.util.predicates
+        :members:
+
+Messages
+--------
+.. automodule:: aleksis.core.util.messages
+        :members:
+
+General helper functions for models
+-----------------------------------
+.. automodule:: aleksis.core.util.model_helpers
+        :members:
+
+Helper functions for SASS
+-------------------------
+.. automodule:: aleksis.core.util.sass_helpers
+        :members:
+
+Utillity function for AlekSIS app container
+-------------------------------------------
+.. automodule:: aleksis.core.util.apps
+        :members:
+
+AlekSIS core middlewares
+------------------------
+.. automodule:: aleksis.core.util.middlewares
+        :members:
+
+Search utillity
+------------------------
+.. automodule:: aleksis.core.util.search
+        :members:
diff --git a/docs/ref/core/10_views.rst b/docs/ref/core/10_views.rst
new file mode 100644
index 0000000000000000000000000000000000000000..15c111f8b97bcfd330dd625cdc864b28b3fa2a44
--- /dev/null
+++ b/docs/ref/core/10_views.rst
@@ -0,0 +1,5 @@
+Core views
+==========
+
+.. automodule:: aleksis.core.views
+   :members:
diff --git a/docs/ref/core/11_filters.rst b/docs/ref/core/11_filters.rst
new file mode 100644
index 0000000000000000000000000000000000000000..04a5aa623a3a339cfbc5884b742e5de14ef317d8
--- /dev/null
+++ b/docs/ref/core/11_filters.rst
@@ -0,0 +1,5 @@
+Core filters
+============
+
+.. automodule:: aleksis.core.filters
+   :members:
diff --git a/docs/ref/core/12_template_tags.rst b/docs/ref/core/12_template_tags.rst
new file mode 100644
index 0000000000000000000000000000000000000000..a4c79715f317fa925a8e2f4dfa7508f4e0d6040c
--- /dev/null
+++ b/docs/ref/core/12_template_tags.rst
@@ -0,0 +1,22 @@
+Template tags
+=============
+
+AlekSIS provides some templatetags to display data in templates.
+
+Dashboard
+---------
+
+.. automodule:: aleksis.core.templatetags.dashboard
+   :members:
+
+Data helpers
+------------
+
+.. automodule:: aleksis.core.templatetags.data_helpers
+   :members:
+
+HTML helpers
+------------
+
+.. automodule:: aleksis.core.templatetags.html_helpers
+   :members:
diff --git a/poetry.lock b/poetry.lock
index 2cd7e64d590d49870a2f0c4ce583a291e7379984..ed1f58deae33e3a86254575ee92d8c9b3c08c0e4 100644
--- a/poetry.lock
+++ b/poetry.lock
@@ -6,9 +6,14 @@ category = "dev"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "aleksis-builddeps"
-version = "3"
+version = "4"
 description = "AlekSIS (School Information System) — Build/Dev dependencies for apps"
 category = "dev"
 optional = false
@@ -27,7 +32,7 @@ flake8-docstrings = ">=1.5.0,<2.0.0"
 flake8-fixme = ">=1.1.1,<2.0.0"
 flake8-isort = ">=4.0.0,<5.0.0"
 flake8-mypy = ">=17.8.0,<18.0.0"
-flake8-rst-docstrings = ">=0.1.0,<0.2.0"
+flake8-rst-docstrings = ">=0.2.0,<0.3.0"
 freezegun = ">=1.1.0,<2.0.0"
 isort = ">=5.0.0,<6.0.0"
 pytest = ">=6.0,<7.0"
@@ -39,6 +44,7 @@ safety = ">=1.8.5,<2.0.0"
 selenium = ">=3.141.0,<4.0.0"
 sphinx = ">=3.0,<4.0"
 sphinx-autodoc-typehints = ">=1.7,<2.0"
+sphinx_materialdesign_theme = ">=0.1.11,<0.2.0"
 sphinxcontrib-django = ">=0.5.0,<0.6.0"
 
 [package.source]
@@ -57,6 +63,11 @@ python-versions = ">=3.6"
 [package.dependencies]
 vine = "5.0.0"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "appdirs"
 version = "1.4.4"
@@ -65,6 +76,11 @@ category = "dev"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "appnope"
 version = "0.1.2"
@@ -73,20 +89,27 @@ category = "main"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "asgiref"
-version = "3.3.4"
+version = "3.4.1"
 description = "ASGI specs, helper code, and adapters"
 category = "main"
 optional = false
 python-versions = ">=3.6"
 
-[package.dependencies]
-typing-extensions = {version = "*", markers = "python_version < \"3.8\""}
-
 [package.extras]
 tests = ["pytest", "pytest-asyncio", "mypy (>=0.800)"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "asn1crypto"
 version = "1.4.0"
@@ -95,6 +118,11 @@ category = "dev"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "atomicwrites"
 version = "1.4.0"
@@ -103,23 +131,33 @@ category = "dev"
 optional = false
 python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "attrs"
-version = "20.3.0"
+version = "21.2.0"
 description = "Classes Without Boilerplate"
 category = "dev"
 optional = false
-python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
+python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
 
 [package.extras]
-dev = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "zope.interface", "furo", "sphinx", "pre-commit"]
-docs = ["furo", "sphinx", "zope.interface"]
-tests = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "zope.interface"]
-tests_no_zope = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six"]
+dev = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "mypy", "pytest-mypy-plugins", "zope.interface", "furo", "sphinx", "sphinx-notfound-page", "pre-commit"]
+docs = ["furo", "sphinx", "zope.interface", "sphinx-notfound-page"]
+tests = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "mypy", "pytest-mypy-plugins", "zope.interface"]
+tests_no_zope = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "mypy", "pytest-mypy-plugins"]
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "babel"
-version = "2.9.0"
+version = "2.9.1"
 description = "Internationalization utilities"
 category = "dev"
 optional = false
@@ -128,6 +166,11 @@ python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
 [package.dependencies]
 pytz = ">=2015.7"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "backcall"
 version = "0.2.0"
@@ -136,6 +179,11 @@ category = "main"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "bandit"
 version = "1.7.0"
@@ -151,21 +199,31 @@ PyYAML = ">=5.3.1"
 six = ">=1.10.0"
 stevedore = ">=1.20.0"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "beautifulsoup4"
-version = "4.9.3"
+version = "4.10.0"
 description = "Screen-scraping library"
 category = "main"
 optional = false
-python-versions = "*"
+python-versions = ">3.0.0"
 
 [package.dependencies]
-soupsieve = {version = ">1.2", markers = "python_version >= \"3.0\""}
+soupsieve = ">1.2"
 
 [package.extras]
 html5lib = ["html5lib"]
 lxml = ["lxml"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "billiard"
 version = "3.6.4.0"
@@ -174,6 +232,11 @@ category = "main"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "black"
 version = "19.10b0"
@@ -194,19 +257,29 @@ typed-ast = ">=1.4.0"
 [package.extras]
 d = ["aiohttp (>=3.3.2)", "aiohttp-cors"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "bleach"
-version = "3.3.0"
+version = "4.1.0"
 description = "An easy safelist-based HTML-sanitizing tool."
 category = "main"
 optional = false
-python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
+python-versions = ">=3.6"
 
 [package.dependencies]
 packaging = "*"
 six = ">=1.9.0"
 webencodings = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "boolean.py"
 version = "3.8"
@@ -215,26 +288,34 @@ category = "main"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "boto3"
-version = "1.17.53"
+version = "1.19.12"
 description = "The AWS SDK for Python"
 category = "main"
 optional = true
-python-versions = ">= 2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, !=3.5.*"
+python-versions = ">= 3.6"
 
 [package.dependencies]
-botocore = ">=1.20.53,<1.21.0"
+botocore = ">=1.22.12,<1.23.0"
 jmespath = ">=0.7.1,<1.0.0"
-s3transfer = ">=0.3.0,<0.4.0"
+s3transfer = ">=0.5.0,<0.6.0"
+
+[package.extras]
+crt = ["botocore[crt] (>=1.21.0,<2.0a0)"]
 
 [[package]]
 name = "botocore"
-version = "1.20.53"
+version = "1.22.12"
 description = "Low-level, data-driven core of boto 3."
 category = "main"
 optional = true
-python-versions = ">= 2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, !=3.5.*"
+python-versions = ">= 3.6"
 
 [package.dependencies]
 jmespath = ">=0.7.1,<1.0.0"
@@ -242,12 +323,17 @@ python-dateutil = ">=2.1,<3.0.0"
 urllib3 = ">=1.25.4,<1.27"
 
 [package.extras]
-crt = ["awscrt (==0.10.8)"]
+crt = ["awscrt (==0.12.5)"]
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "bs4"
 version = "0.0.1"
-description = "Dummy package for Beautiful Soup"
+description = "Screen-scraping library"
 category = "main"
 optional = false
 python-versions = "*"
@@ -255,6 +341,11 @@ python-versions = "*"
 [package.dependencies]
 beautifulsoup4 = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "calendarweek"
 version = "0.5.0"
@@ -266,22 +357,27 @@ python-versions = ">=3.7,<4.0"
 [package.extras]
 django = ["Django (>=2.2,<4.0)"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "celery"
-version = "5.0.5"
+version = "5.1.2"
 description = "Distributed Task Queue."
 category = "main"
 optional = false
 python-versions = ">=3.6,"
 
 [package.dependencies]
-billiard = ">=3.6.3.0,<4.0"
+billiard = ">=3.6.4.0,<4.0"
 click = ">=7.0,<8.0"
 click-didyoumean = ">=0.0.3"
 click-plugins = ">=1.1.1"
 click-repl = ">=0.1.6"
 Django = {version = ">=1.11", optional = true, markers = "extra == \"django\""}
-kombu = ">=5.0.0,<6.0"
+kombu = ">=5.1.0,<6.0"
 pytz = ">0.0-dev"
 redis = {version = ">=3.2.0", optional = true, markers = "extra == \"redis\""}
 vine = ">=5.0.0,<6.0"
@@ -289,10 +385,10 @@ vine = ">=5.0.0,<6.0"
 [package.extras]
 arangodb = ["pyArango (>=1.3.2)"]
 auth = ["cryptography"]
-azureblockblob = ["azure-storage (==0.36.0)", "azure-common (==1.1.5)", "azure-storage-common (==1.1.0)"]
+azureblockblob = ["azure-storage-blob (==12.6.0)"]
 brotli = ["brotli (>=1.0.0)", "brotlipy (>=0.7.0)"]
 cassandra = ["cassandra-driver (<3.21.0)"]
-consul = ["python-consul"]
+consul = ["python-consul2"]
 cosmosdbsql = ["pydocumentdb (==2.3.2)"]
 couchbase = ["couchbase (>=3.0.0)"]
 couchdb = ["pycouchdb"]
@@ -302,7 +398,6 @@ elasticsearch = ["elasticsearch"]
 eventlet = ["eventlet (>=0.26.1)"]
 gevent = ["gevent (>=1.0.0)"]
 librabbitmq = ["librabbitmq (>=1.5.0)"]
-lzma = ["backports.lzma"]
 memcache = ["pylibmc"]
 mongodb = ["pymongo[srv] (>=3.3.0)"]
 msgpack = ["msgpack"]
@@ -320,6 +415,11 @@ yaml = ["PyYAML (>=3.10)"]
 zookeeper = ["kazoo (>=1.3.1)"]
 zstd = ["zstandard"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "celery-haystack-ng"
 version = "0.20.post2"
@@ -333,9 +433,14 @@ celery = ">=4.0"
 django-appconf = ">=0.4.1"
 django-haystack = ">=2.0"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "celery-progress"
-version = "0.1.0"
+version = "0.1.1"
 description = "Drop in, configurable, dependency-free progress bars for your Django/Celery applications."
 category = "main"
 optional = false
@@ -346,21 +451,55 @@ rabbitmq = ["channels-rabbitmq"]
 redis = ["channels-redis"]
 websockets = ["channels"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "certifi"
-version = "2020.12.5"
+version = "2021.10.8"
 description = "Python package for providing Mozilla's CA Bundle."
 category = "main"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
-name = "chardet"
-version = "4.0.0"
-description = "Universal encoding detector for Python 2 and 3"
+name = "cffi"
+version = "1.15.0"
+description = "Foreign Function Interface for Python calling C code."
 category = "main"
 optional = false
-python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
+python-versions = "*"
+
+[package.dependencies]
+pycparser = "*"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
+[[package]]
+name = "charset-normalizer"
+version = "2.0.7"
+description = "The Real First Universal Charset Detector. Open, modern and actively maintained alternative to Chardet."
+category = "main"
+optional = false
+python-versions = ">=3.5.0"
+
+[package.extras]
+unicode_backport = ["unicodedata2"]
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "click"
@@ -370,16 +509,26 @@ category = "main"
 optional = false
 python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "click-didyoumean"
-version = "0.0.3"
-description = "Enable git-like did-you-mean feature in click."
+version = "0.3.0"
+description = "Enables git-like *did-you-mean* feature in click"
 category = "main"
 optional = false
-python-versions = "*"
+python-versions = ">=3.6.2,<4.0.0"
 
 [package.dependencies]
-click = "*"
+click = ">=7"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "click-plugins"
@@ -395,9 +544,14 @@ click = ">=4.0"
 [package.extras]
 dev = ["pytest (>=3.6)", "pytest-cov", "wheel", "coveralls"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "click-repl"
-version = "0.1.6"
+version = "0.2.0"
 description = "REPL plugin for Click"
 category = "main"
 optional = false
@@ -408,6 +562,11 @@ click = "*"
 prompt-toolkit = "*"
 six = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "colorama"
 version = "0.4.4"
@@ -416,6 +575,11 @@ category = "main"
 optional = false
 python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "colour"
 version = "0.1.5"
@@ -427,6 +591,11 @@ python-versions = "*"
 [package.extras]
 test = ["nose"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "configobj"
 version = "5.0.6"
@@ -438,16 +607,50 @@ python-versions = "*"
 [package.dependencies]
 six = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "coverage"
-version = "5.5"
+version = "6.1.1"
 description = "Code coverage measurement for Python"
 category = "dev"
 optional = false
-python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, <4"
+python-versions = ">=3.6"
 
 [package.extras]
-toml = ["toml"]
+toml = ["tomli"]
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
+[[package]]
+name = "cryptography"
+version = "35.0.0"
+description = "cryptography is a package which provides cryptographic recipes and primitives to Python developers."
+category = "main"
+optional = false
+python-versions = ">=3.6"
+
+[package.dependencies]
+cffi = ">=1.12"
+
+[package.extras]
+docs = ["sphinx (>=1.6.5,!=1.8.0,!=3.1.0,!=3.1.1)", "sphinx-rtd-theme"]
+docstest = ["doc8", "pyenchant (>=1.6.11)", "twine (>=1.12.0)", "sphinxcontrib-spelling (>=4.0.1)"]
+pep8test = ["black", "flake8", "flake8-import-order", "pep8-naming"]
+sdist = ["setuptools_rust (>=0.11.4)"]
+ssh = ["bcrypt (>=3.1.5)"]
+test = ["pytest (>=6.2.0)", "pytest-cov", "pytest-subtests", "pytest-xdist", "pretend", "iso8601", "pytz", "hypothesis (>=1.11.4,!=3.79.2)"]
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "curlylint"
@@ -467,14 +670,56 @@ toml = ">=0.9.4"
 [package.extras]
 dev = ["black (==19.10b0)", "flake8 (==3.8.4)", "mypy (==0.812)", "pytest (==6.2.2)", "coverage (==5.4)"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "decorator"
-version = "5.0.7"
+version = "5.1.0"
 description = "Decorators for Humans"
 category = "main"
 optional = false
 python-versions = ">=3.5"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
+[[package]]
+name = "defusedxml"
+version = "0.7.1"
+description = "XML bomb protection for Python stdlib modules"
+category = "main"
+optional = false
+python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
+[[package]]
+name = "deprecated"
+version = "1.2.13"
+description = "Python @deprecated decorator to deprecate old python classes, functions or methods."
+category = "main"
+optional = false
+python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
+
+[package.dependencies]
+wrapt = ">=1.10,<2"
+
+[package.extras]
+dev = ["tox", "bump2version (<1)", "sphinx (<2)", "importlib-metadata (<3)", "importlib-resources (<4)", "configparser (<5)", "sphinxcontrib-websupport (<2)", "zipp (<2)", "PyTest (<5)", "PyTest-Cov (<2.6)", "pytest", "pytest-cov"]
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "dj-database-url"
 version = "0.5.0"
@@ -483,9 +728,14 @@ category = "dev"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django"
-version = "3.2"
+version = "3.2.9"
 description = "A high-level Python Web framework that encourages rapid development and clean, pragmatic design."
 category = "main"
 optional = false
@@ -500,9 +750,34 @@ sqlparse = ">=0.2.2"
 argon2 = ["argon2-cffi (>=19.1.0)"]
 bcrypt = ["bcrypt"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
-name = "django-any-js"
-version = "1.1"
+name = "django-allauth"
+version = "0.45.0"
+description = "Integrated set of Django applications addressing authentication, registration, account management as well as 3rd party (social) account authentication."
+category = "main"
+optional = false
+python-versions = "*"
+
+[package.dependencies]
+Django = ">=2.0"
+pyjwt = {version = ">=1.7", extras = ["crypto"]}
+python3-openid = ">=3.0.8"
+requests = "*"
+requests-oauthlib = ">=0.3.0"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
+[[package]]
+name = "django-any-js"
+version = "1.1"
 description = "Include JavaScript/CSS libraries with readable template tags"
 category = "main"
 optional = false
@@ -511,20 +786,30 @@ python-versions = ">=3.7,<4.0"
 [package.dependencies]
 Django = ">=2.2,<4.0"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-appconf"
-version = "1.0.4"
+version = "1.0.5"
 description = "A helper class for handling configuration defaults of packaged apps gracefully."
 category = "main"
 optional = false
-python-versions = "*"
+python-versions = ">=3.6"
 
 [package.dependencies]
 django = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-auth-ldap"
-version = "2.4.0"
+version = "3.0.0"
 description = "Django LDAP authentication backend."
 category = "main"
 optional = true
@@ -534,9 +819,14 @@ python-versions = ">=3.6"
 Django = ">=2.2"
 python-ldap = ">=3.1"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-bleach"
-version = "0.6.1"
+version = "0.9.0"
 description = "Easily use bleach with Django models and templates"
 category = "main"
 optional = false
@@ -546,31 +836,30 @@ python-versions = "*"
 bleach = ">=1.5.0"
 Django = ">=1.11"
 
-[[package]]
-name = "django-bulk-update"
-version = "2.2.0"
-description = "Bulk update using one query over Django ORM."
-category = "main"
-optional = false
-python-versions = "*"
-
-[package.dependencies]
-Django = ">=1.8"
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "django-cachalot"
-version = "2.3.5"
+version = "2.4.4"
 description = "Caches your Django ORM queries and automatically invalidates them."
 category = "main"
 optional = false
 python-versions = "*"
 
 [package.dependencies]
-Django = ">=2"
+Django = ">=2.2,<3.3"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "django-cache-memoize"
-version = "0.1.8"
+version = "0.1.10"
 description = "Django utility for a memoization decorator that uses the Django cache framework."
 category = "main"
 optional = false
@@ -579,20 +868,30 @@ python-versions = ">=3.5"
 [package.extras]
 dev = ["flake8", "tox", "twine", "therapist", "black"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-celery-beat"
-version = "2.2.0"
+version = "2.2.1"
 description = "Database-backed Periodic Tasks."
 category = "main"
 optional = false
 python-versions = "*"
 
 [package.dependencies]
-celery = ">=4.4,<6.0"
+celery = ">=5.0,<6.0"
 Django = ">=2.2,<4.0"
 django-timezone-field = ">=4.1.0,<5.0"
 python-crontab = ">=2.3.4"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-celery-email"
 version = "3.0.0"
@@ -606,20 +905,30 @@ celery = ">=4.0"
 django = ">=2.2"
 django-appconf = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-celery-results"
-version = "2.0.1"
+version = "2.2.0"
 description = "Celery result backends for Django."
 category = "main"
 optional = false
 python-versions = "*"
 
 [package.dependencies]
-celery = ">=4.4,<6.0"
+celery = ">=5.0,<6.0"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "django-ckeditor"
-version = "6.0.0"
+version = "6.1.0"
 description = "Django admin CKEditor integration."
 category = "main"
 optional = false
@@ -628,22 +937,37 @@ python-versions = "*"
 [package.dependencies]
 django-js-asset = ">=1.2.2"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-cleanup"
-version = "5.1.0"
+version = "5.2.0"
 description = "Deletes old files."
 category = "main"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-colorfield"
-version = "0.4.1"
+version = "0.4.5"
 description = "simple color field for your models with a nice color-picker in the admin-interface."
 category = "main"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-dbbackup"
 version = "3.3.0"
@@ -657,9 +981,14 @@ Django = ">=1.5"
 pytz = "*"
 six = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-debug-toolbar"
-version = "3.2.1"
+version = "3.2.2"
 description = "A configurable set of panels that display various debug information about the current request/response."
 category = "main"
 optional = false
@@ -669,9 +998,14 @@ python-versions = ">=3.6"
 Django = ">=2.2"
 sqlparse = ">=0.2.0"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-dynamic-preferences"
-version = "1.10.1"
+version = "1.11.0"
 description = "Dynamic global and instance settings for your django project"
 category = "main"
 optional = false
@@ -682,9 +1016,14 @@ django = ">=1.11"
 persisting-theory = ">=0.2.1"
 six = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-extensions"
-version = "3.1.2"
+version = "3.1.3"
 description = "Extensions for Django"
 category = "main"
 optional = false
@@ -693,9 +1032,14 @@ python-versions = ">=3.6"
 [package.dependencies]
 Django = ">=2.2"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-favicon-plus-reloaded"
-version = "1.0.4"
+version = "1.1.3"
 description = "simple Django app which allows you to upload a image and it renders a wide variety for html link tags to display the favicon"
 category = "main"
 optional = false
@@ -705,6 +1049,11 @@ python-versions = "*"
 django = "*"
 pillow = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-filter"
 version = "2.4.0"
@@ -716,20 +1065,30 @@ python-versions = ">=3.5"
 [package.dependencies]
 Django = ">=2.2"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-formtools"
-version = "2.2"
+version = "2.3"
 description = "A set of high-level abstractions for Django forms"
 category = "main"
 optional = false
-python-versions = "*"
+python-versions = ">=3.6"
 
 [package.dependencies]
-Django = ">=1.11"
+Django = ">=2.2"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "django-guardian"
-version = "2.3.0"
+version = "2.4.0"
 description = "Implementation of per object permissions for Django."
 category = "main"
 optional = false
@@ -738,24 +1097,14 @@ python-versions = ">=3.5"
 [package.dependencies]
 Django = ">=2.2"
 
-[[package]]
-name = "django-hattori"
-version = "0.2.1"
-description = "Command to anonymize sensitive data."
-category = "main"
-optional = false
-python-versions = "*"
-
-[package.dependencies]
-Django = ">=1.8"
-django-bulk-update = ">=2.2.0"
-Faker = ">=0.8.13"
-six = "*"
-tqdm = ">=4.23.4"
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "django-haystack"
-version = "3.0"
+version = "3.1.1"
 description = "Pluggable search for Django."
 category = "main"
 optional = false
@@ -764,16 +1113,29 @@ python-versions = "*"
 [package.dependencies]
 Django = ">=2.2"
 
+[package.extras]
+elasticsearch = ["elasticsearch (>=5,<6)"]
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-health-check"
-version = "3.16.3"
+version = "3.16.4"
 description = "Run checks on services like databases, queue servers, celery processes, etc."
 category = "main"
 optional = false
 python-versions = "*"
 
 [package.dependencies]
-django = ">=1.11"
+django = ">=2.2"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "django-impersonate"
@@ -783,13 +1145,23 @@ category = "main"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-ipware"
-version = "3.0.2"
-description = "A Django utility application that returns client's real IP address"
+version = "4.0.0"
+description = "A Django application to retrieve user's IP address"
 category = "main"
 optional = false
-python-versions = "*"
+python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, !=3.5.*"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "django-js-asset"
@@ -799,6 +1171,11 @@ category = "main"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-js-reverse"
 version = "0.9.1"
@@ -810,6 +1187,11 @@ python-versions = "*"
 [package.dependencies]
 Django = ">=1.5"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-jsonstore"
 version = "0.5.0"
@@ -822,17 +1204,27 @@ python-versions = "*"
 Django = ">=1.11"
 six = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-maintenance-mode"
-version = "0.16.0"
+version = "0.16.1"
 description = "django-maintenance-mode shows a 503 error page when maintenance-mode is on."
 category = "main"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-material"
-version = "1.7.6"
+version = "1.9.0"
 description = "Material design for django forms and admin"
 category = "main"
 optional = false
@@ -841,6 +1233,11 @@ python-versions = "*"
 [package.dependencies]
 six = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-menu-generator-ng"
 version = "1.2.3"
@@ -849,31 +1246,50 @@ category = "main"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
-name = "django-middleware-global-request"
-version = "0.1.2"
-description = "Django middleware that keep request instance for every thread."
+name = "django-model-utils"
+version = "4.2.0"
+description = "Django model mixins and utilities"
 category = "main"
 optional = false
 python-versions = "*"
 
 [package.dependencies]
-django = "*"
+Django = ">=2.0.1"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
-name = "django-model-utils"
-version = "4.1.1"
-description = "Django model mixins and utilities"
+name = "django-oauth-toolkit"
+version = "1.5.0"
+description = "OAuth2 Provider for Django"
 category = "main"
 optional = false
 python-versions = "*"
 
 [package.dependencies]
-Django = ">=2.0.1"
+django = ">=2.2"
+jwcrypto = ">=0.8.0"
+oauthlib = ">=3.1.0"
+requests = ">=2.13.0"
+six = "*"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "django-otp"
-version = "1.0.3"
+version = "1.1.1"
 description = "A pluggable framework for adding two-factor authentication to Django using one-time passwords."
 category = "main"
 optional = false
@@ -885,6 +1301,11 @@ django = ">=2.2"
 [package.extras]
 qrcode = ["qrcode"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-otp-yubikey"
 version = "1.0.0.post1"
@@ -897,9 +1318,14 @@ python-versions = "*"
 django-otp = ">=1.0.0"
 YubiOTP = ">=0.2.2"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-phonenumber-field"
-version = "5.1.0"
+version = "5.2.0"
 description = "An international phone number field for django models."
 category = "main"
 optional = false
@@ -913,6 +1339,11 @@ phonenumbers = {version = ">=7.0.2", optional = true, markers = "extra == \"phon
 phonenumbers = ["phonenumbers (>=7.0.2)"]
 phonenumberslite = ["phonenumberslite (>=7.0.2)"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-polymorphic"
 version = "3.0.0"
@@ -924,6 +1355,11 @@ python-versions = "*"
 [package.dependencies]
 Django = ">=2.1"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-prometheus"
 version = "2.1.0"
@@ -934,30 +1370,29 @@ python-versions = "*"
 
 [package.dependencies]
 prometheus-client = ">=0.7"
-
-[[package]]
-name = "django-pwa"
-version = "1.0.10"
-description = "A Django app to include a manifest.json and Service Worker instance to enable progressive web app behavior"
-category = "main"
-optional = false
-python-versions = "*"
-
-[package.dependencies]
-django = ">=1.8"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "django-redis"
-version = "4.12.1"
+version = "5.0.0"
 description = "Full featured redis cache backend for Django."
 category = "main"
 optional = false
-python-versions = ">=3.5"
+python-versions = ">=3.6"
 
 [package.dependencies]
 Django = ">=2.2"
 redis = ">=3.0.0"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-render-block"
 version = "0.8.1"
@@ -969,20 +1404,30 @@ python-versions = ">=3.5"
 [package.dependencies]
 django = ">=2.2"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-reversion"
-version = "3.0.9"
+version = "4.0.1"
 description = "An extension to the Django web framework that provides version control for model instances."
 category = "main"
 optional = false
 python-versions = ">=3.6"
 
 [package.dependencies]
-django = ">=1.11"
+django = ">=2.0"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "django-sass-processor"
-version = "1.0.1"
+version = "1.0.0"
 description = "SASS processor to compile SCSS files into *.css, while rendering, or offline."
 category = "main"
 optional = false
@@ -991,9 +1436,14 @@ python-versions = "*"
 [package.extras]
 management_command = ["django-compressor (>=2.4)"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-select2"
-version = "7.7.1"
+version = "7.9.0"
 description = "Select2 option fields for Django"
 category = "main"
 optional = false
@@ -1006,17 +1456,14 @@ django-appconf = ">=0.6.0"
 [package.extras]
 test = ["pytest", "pytest-cov", "pytest-django", "selenium"]
 
-[[package]]
-name = "django-settings-context-processor"
-version = "0.2"
-description = "Makes specified django settings visible in template rendering context."
-category = "main"
-optional = false
-python-versions = "*"
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "django-storages"
-version = "1.11.1"
+version = "1.12.3"
 description = "Support for many storage backends in Django"
 category = "main"
 optional = true
@@ -1026,16 +1473,21 @@ python-versions = ">=3.5"
 Django = ">=2.2"
 
 [package.extras]
-azure = ["azure-storage-blob (>=1.3.1,<12.0.0)"]
+azure = ["azure-storage-blob (>=12.0.0)"]
 boto3 = ["boto3 (>=1.4.4)"]
 dropbox = ["dropbox (>=7.2.1)"]
-google = ["google-cloud-storage (>=1.15.0)"]
+google = ["google-cloud-storage (>=1.27.0)"]
 libcloud = ["apache-libcloud"]
 sftp = ["paramiko"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-stubs"
-version = "1.8.0"
+version = "1.9.0"
 description = "Mypy stubs for Django"
 category = "dev"
 optional = false
@@ -1043,13 +1495,21 @@ python-versions = ">=3.6"
 
 [package.dependencies]
 django = "*"
-django-stubs-ext = "*"
-mypy = ">=0.790"
+django-stubs-ext = ">=0.3.0"
+mypy = ">=0.910"
+toml = "*"
+types-pytz = "*"
+types-PyYAML = "*"
 typing-extensions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-stubs-ext"
-version = "0.2.0"
+version = "0.3.1"
 description = "Monkey-patching and extensions for django-stubs"
 category = "dev"
 optional = false
@@ -1057,10 +1517,16 @@ python-versions = ">=3.6"
 
 [package.dependencies]
 django = "*"
+typing-extensions = "*"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "django-tables2"
-version = "2.3.4"
+version = "2.4.1"
 description = "Table/data-grid framework for Django"
 category = "main"
 optional = false
@@ -1072,9 +1538,14 @@ Django = ">=1.11"
 [package.extras]
 tablib = ["tablib"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-templated-email"
-version = "2.3.0"
+version = "3.0.0"
 description = "A Django oriented templated / transaction email abstraction"
 category = "main"
 optional = false
@@ -1082,11 +1553,15 @@ python-versions = "*"
 
 [package.dependencies]
 django-render-block = ">=0.5"
-six = ">=1"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "django-timezone-field"
-version = "4.1.2"
+version = "4.2.1"
 description = "A Django app providing database and form fields for pytz timezone objects."
 category = "main"
 optional = false
@@ -1099,6 +1574,27 @@ pytz = "*"
 [package.extras]
 rest_framework = ["djangorestframework (>=3.0.0)"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
+[[package]]
+name = "django-titofisto"
+version = "0.1.2.post1"
+description = "Django Time-Token File Storage"
+category = "main"
+optional = false
+python-versions = ">=3.9,<4.0"
+
+[package.dependencies]
+Django = ">2.2,<4.0"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-two-factor-auth"
 version = "1.13.1"
@@ -1124,9 +1620,14 @@ phonenumberslite = ["phonenumberslite (>=7.0.9,<8.99)"]
 sms = ["twilio (>=6.0)"]
 yubikey = ["django-otp-yubikey"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-uwsgi-ng"
-version = "1.1.1"
+version = "1.1.2"
 description = "uWSGI stuff for Django projects"
 category = "main"
 optional = false
@@ -1135,14 +1636,24 @@ python-versions = "*"
 [package.extras]
 uwsgi = ["uwsgi"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-widget-tweaks"
-version = "1.4.8"
+version = "1.4.9"
 description = "Tweak the form field rendering in templates, not in python-level form definitions."
 category = "main"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "django-yarnpkg"
 version = "6.0.1"
@@ -1155,6 +1666,27 @@ python-versions = "*"
 django = "*"
 six = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
+[[package]]
+name = "djangorestframework"
+version = "3.12.4"
+description = "Web APIs for Django, made easy."
+category = "main"
+optional = false
+python-versions = ">=3.5"
+
+[package.dependencies]
+django = ">=2.2"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "docutils"
 version = "0.16"
@@ -1163,6 +1695,11 @@ category = "dev"
 optional = false
 python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "dparse"
 version = "0.5.1"
@@ -1179,13 +1716,18 @@ toml = "*"
 [package.extras]
 pipenv = ["pipenv"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "dynaconf"
-version = "3.1.4"
+version = "3.1.7"
 description = "The dynamic configurator for your Python Project"
 category = "main"
 optional = false
-python-versions = "*"
+python-versions = ">=3.7"
 
 [package.dependencies]
 configobj = {version = "*", optional = true, markers = "extra == \"ini\""}
@@ -1201,32 +1743,29 @@ toml = ["toml"]
 vault = ["hvac"]
 yaml = ["ruamel.yaml"]
 
-[[package]]
-name = "faker"
-version = "8.1.0"
-description = "Faker is a Python package that generates fake data for you."
-category = "main"
-optional = false
-python-versions = ">=3.6"
-
-[package.dependencies]
-python-dateutil = ">=2.4"
-text-unidecode = "1.3"
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "flake8"
-version = "3.9.1"
+version = "3.9.2"
 description = "the modular source code checker: pep8 pyflakes and co"
 category = "dev"
 optional = false
 python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,>=2.7"
 
 [package.dependencies]
-importlib-metadata = {version = "*", markers = "python_version < \"3.8\""}
 mccabe = ">=0.6.0,<0.7.0"
 pycodestyle = ">=2.7.0,<2.8.0"
 pyflakes = ">=2.3.0,<2.4.0"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "flake8-bandit"
 version = "2.1.2"
@@ -1241,9 +1780,14 @@ flake8 = "*"
 flake8-polyfill = "*"
 pycodestyle = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "flake8-black"
-version = "0.2.1"
+version = "0.2.3"
 description = "flake8 plugin to call black as a code style validator"
 category = "dev"
 optional = false
@@ -1252,6 +1796,12 @@ python-versions = "*"
 [package.dependencies]
 black = "*"
 flake8 = ">=3.0.0"
+toml = "*"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "flake8-builtins"
@@ -1267,16 +1817,26 @@ flake8 = "*"
 [package.extras]
 test = ["coverage", "coveralls", "mock", "pytest", "pytest-cov"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "flake8-django"
-version = "1.1.1"
-description = "Plugin to catch bad style specific to Django Projects"
+version = "1.1.2"
+description = "Plugin to catch bad style specific to Django Projects."
 category = "dev"
 optional = false
-python-versions = "*"
+python-versions = ">=3.6,<4.0"
 
 [package.dependencies]
-flake8 = "*"
+flake8 = ">=3.8.4,<4.0.0"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "flake8-docstrings"
@@ -1290,6 +1850,11 @@ python-versions = "*"
 flake8 = ">=3"
 pydocstyle = ">=2.1"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "flake8-fixme"
 version = "1.1.1"
@@ -1298,21 +1863,31 @@ category = "dev"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "flake8-isort"
-version = "4.0.0"
+version = "4.1.1"
 description = "flake8 plugin that integrates isort ."
 category = "dev"
 optional = false
 python-versions = "*"
 
 [package.dependencies]
-flake8 = ">=3.2.1,<4"
+flake8 = ">=3.2.1,<5"
 isort = ">=4.3.5,<6"
 testfixtures = ">=6.8.0,<7"
 
 [package.extras]
-test = ["pytest (>=4.0.2,<6)", "toml"]
+test = ["pytest-cov"]
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "flake8-mypy"
@@ -1327,6 +1902,11 @@ attrs = "*"
 flake8 = ">=3.0.0"
 mypy = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "flake8-polyfill"
 version = "1.0.2"
@@ -1338,9 +1918,14 @@ python-versions = "*"
 [package.dependencies]
 flake8 = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "flake8-rst-docstrings"
-version = "0.1.2"
+version = "0.2.3"
 description = "Python docstring reStructuredText (RST) validator"
 category = "dev"
 optional = false
@@ -1348,10 +1933,14 @@ python-versions = ">=3.3"
 
 [package.dependencies]
 flake8 = ">=3.0.0"
-pydocstyle = ">=3.0.0"
 pygments = "*"
 restructuredtext-lint = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "freezegun"
 version = "1.1.0"
@@ -1363,27 +1952,61 @@ python-versions = ">=3.5"
 [package.dependencies]
 python-dateutil = ">=2.7"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "gitdb"
-version = "4.0.7"
+version = "4.0.9"
 description = "Git Object Database"
 category = "dev"
 optional = false
-python-versions = ">=3.4"
+python-versions = ">=3.6"
 
 [package.dependencies]
-smmap = ">=3.0.1,<5"
+smmap = ">=3.0.1,<6"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "gitpython"
-version = "3.1.14"
-description = "Python Git Library"
+version = "3.1.24"
+description = "GitPython is a python library used to interact with Git repositories"
 category = "dev"
 optional = false
-python-versions = ">=3.4"
+python-versions = ">=3.7"
 
 [package.dependencies]
 gitdb = ">=4.0.1,<5"
+typing-extensions = {version = ">=3.7.4.3", markers = "python_version < \"3.10\""}
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
+[[package]]
+name = "haystack-redis"
+version = "0.0.1"
+description = "Use redis as a persistence layer for Whoosh and Haystack"
+category = "main"
+optional = false
+python-versions = "*"
+
+[package.dependencies]
+django-haystack = "*"
+redis = "*"
+whoosh = "*"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "html2text"
@@ -1393,13 +2016,23 @@ category = "main"
 optional = false
 python-versions = ">=3.5"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "idna"
-version = "2.10"
+version = "3.3"
 description = "Internationalized Domain Names in Applications (IDNA)"
 category = "main"
 optional = false
-python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
+python-versions = ">=3.5"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "imagesize"
@@ -1409,33 +2042,27 @@ category = "dev"
 optional = false
 python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
-name = "importlib-metadata"
-version = "3.10.1"
-description = "Read metadata from Python packages"
-category = "main"
-optional = false
-python-versions = ">=3.6"
-
-[package.dependencies]
-typing-extensions = {version = ">=3.6.4", markers = "python_version < \"3.8\""}
-zipp = ">=0.5"
-
-[package.extras]
-docs = ["sphinx", "jaraco.packaging (>=8.2)", "rst.linker (>=1.9)"]
-testing = ["pytest (>=4.6)", "pytest-checkdocs (>=2.4)", "pytest-flake8", "pytest-cov", "pytest-enabler (>=1.0.1)", "packaging", "pep517", "pyfakefs", "flufl.flake8", "pytest-black (>=0.3.7)", "pytest-mypy", "importlib-resources (>=1.3)"]
-
-[[package]]
-name = "iniconfig"
-version = "1.1.1"
-description = "iniconfig: brain-dead simple config-ini parsing"
-category = "dev"
+name = "iniconfig"
+version = "1.1.1"
+description = "iniconfig: brain-dead simple config-ini parsing"
+category = "dev"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "ipython"
-version = "7.22.0"
+version = "7.29.0"
 description = "IPython: Productive Interactive Computing"
 category = "main"
 optional = false
@@ -1447,6 +2074,7 @@ backcall = "*"
 colorama = {version = "*", markers = "sys_platform == \"win32\""}
 decorator = "*"
 jedi = ">=0.16"
+matplotlib-inline = "*"
 pexpect = {version = ">4.3", markers = "sys_platform != \"win32\""}
 pickleshare = "*"
 prompt-toolkit = ">=2.0.0,<3.0.0 || >3.0.0,<3.0.1 || >3.0.1,<3.1.0"
@@ -1454,7 +2082,7 @@ pygments = "*"
 traitlets = ">=4.2"
 
 [package.extras]
-all = ["Sphinx (>=1.3)", "ipykernel", "ipyparallel", "ipywidgets", "nbconvert", "nbformat", "nose (>=0.10.1)", "notebook", "numpy (>=1.16)", "pygments", "qtconsole", "requests", "testpath"]
+all = ["Sphinx (>=1.3)", "ipykernel", "ipyparallel", "ipywidgets", "nbconvert", "nbformat", "nose (>=0.10.1)", "notebook", "numpy (>=1.17)", "pygments", "qtconsole", "requests", "testpath"]
 doc = ["Sphinx (>=1.3)"]
 kernel = ["ipykernel"]
 nbconvert = ["nbconvert"]
@@ -1462,28 +2090,31 @@ nbformat = ["nbformat"]
 notebook = ["notebook", "ipywidgets"]
 parallel = ["ipyparallel"]
 qtconsole = ["qtconsole"]
-test = ["nose (>=0.10.1)", "requests", "testpath", "pygments", "nbformat", "ipykernel", "numpy (>=1.16)"]
+test = ["nose (>=0.10.1)", "requests", "testpath", "pygments", "nbformat", "ipykernel", "numpy (>=1.17)"]
 
-[[package]]
-name = "ipython-genutils"
-version = "0.2.0"
-description = "Vestigial utilities from IPython"
-category = "main"
-optional = false
-python-versions = "*"
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "isort"
-version = "5.8.0"
+version = "5.10.0"
 description = "A Python utility / library to sort Python imports."
 category = "dev"
 optional = false
-python-versions = ">=3.6,<4.0"
+python-versions = ">=3.6.1,<4.0"
 
 [package.extras]
-pipfile_deprecated_finder = ["pipreqs", "requirementslib"]
-requirements_deprecated_finder = ["pipreqs", "pip-api"]
 colors = ["colorama (>=0.4.3,<0.5.0)"]
+requirements_deprecated_finder = ["pip-api", "pipreqs"]
+pipfile_deprecated_finder = ["pipreqs", "requirementslib"]
+plugins = ["setuptools"]
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "jedi"
@@ -1500,19 +2131,29 @@ parso = ">=0.8.0,<0.9.0"
 qa = ["flake8 (==3.8.3)", "mypy (==0.782)"]
 testing = ["Django (<3.1)", "colorama", "docopt", "pytest (<6.0.0)"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "jinja2"
-version = "2.11.3"
+version = "3.0.2"
 description = "A very fast and expressive template engine."
 category = "dev"
 optional = false
-python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
+python-versions = ">=3.6"
 
 [package.dependencies]
-MarkupSafe = ">=0.23"
+MarkupSafe = ">=2.0"
 
 [package.extras]
-i18n = ["Babel (>=0.8)"]
+i18n = ["Babel (>=2.7)"]
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "jmespath"
@@ -1522,20 +2163,42 @@ category = "main"
 optional = true
 python-versions = ">=2.6, !=3.0.*, !=3.1.*, !=3.2.*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
+[[package]]
+name = "jwcrypto"
+version = "1.0"
+description = "Implementation of JOSE Web standards"
+category = "main"
+optional = false
+python-versions = "*"
+
+[package.dependencies]
+cryptography = ">=2.3"
+deprecated = "*"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "kombu"
-version = "5.0.2"
+version = "5.2.0"
 description = "Messaging library for Python."
 category = "main"
 optional = false
-python-versions = ">=3.6"
+python-versions = ">=3.7"
 
 [package.dependencies]
-amqp = ">=5.0.0,<6.0.0"
-importlib-metadata = {version = ">=0.18", markers = "python_version < \"3.8\""}
+amqp = ">=5.0.6,<6.0.0"
+vine = "*"
 
 [package.extras]
-azureservicebus = ["azure-servicebus (>=0.21.1)"]
+azureservicebus = ["azure-servicebus (>=7.0.0)"]
 azurestoragequeues = ["azure-storage-queue"]
 consul = ["python-consul (>=0.6.0)"]
 librabbitmq = ["librabbitmq (>=1.5.2)"]
@@ -1546,13 +2209,18 @@ qpid = ["qpid-python (>=0.26)", "qpid-tools (>=0.26)"]
 redis = ["redis (>=3.3.11)"]
 slmq = ["softlayer-messaging (>=1.0.3)"]
 sqlalchemy = ["sqlalchemy"]
-sqs = ["boto3 (>=1.4.4)", "pycurl (==7.43.0.2)"]
+sqs = ["boto3 (>=1.4.4)", "pycurl (==7.43.0.2)", "urllib3 (<1.26)"]
 yaml = ["PyYAML (>=3.10)"]
 zookeeper = ["kazoo (>=1.3.1)"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "libsass"
-version = "0.20.1"
+version = "0.21.0"
 description = "Sass for Python: A straightforward binding of libsass for Python."
 category = "main"
 optional = false
@@ -1561,6 +2229,11 @@ python-versions = "*"
 [package.dependencies]
 six = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "license-expression"
 version = "1.2"
@@ -1572,13 +2245,39 @@ python-versions = "*"
 [package.dependencies]
 "boolean.py" = ">=3.6,<4.0.0"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "markupsafe"
-version = "1.1.1"
+version = "2.0.1"
 description = "Safely add untrusted strings to HTML/XML markup."
 category = "dev"
 optional = false
-python-versions = ">=2.7,!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*"
+python-versions = ">=3.6"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
+[[package]]
+name = "matplotlib-inline"
+version = "0.1.3"
+description = "Inline Matplotlib backend for Jupyter"
+category = "main"
+optional = false
+python-versions = ">=3.5"
+
+[package.dependencies]
+traitlets = "*"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "mccabe"
@@ -1588,9 +2287,14 @@ category = "dev"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "mypy"
-version = "0.812"
+version = "0.910"
 description = "Optional static typing for Python"
 category = "dev"
 optional = false
@@ -1598,11 +2302,17 @@ python-versions = ">=3.5"
 
 [package.dependencies]
 mypy-extensions = ">=0.4.3,<0.5.0"
-typed-ast = ">=1.4.0,<1.5.0"
+toml = "*"
 typing-extensions = ">=3.7.4"
 
 [package.extras]
 dmypy = ["psutil (>=4.0)"]
+python2 = ["typed-ast (>=1.4.0,<1.5.0)"]
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "mypy-extensions"
@@ -1612,16 +2322,44 @@ category = "dev"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
+[[package]]
+name = "oauthlib"
+version = "3.1.1"
+description = "A generic, spec-compliant, thorough implementation of the OAuth request-signing logic"
+category = "main"
+optional = false
+python-versions = ">=3.6"
+
+[package.extras]
+rsa = ["cryptography (>=3.0.0,<4)"]
+signals = ["blinker (>=1.4.0)"]
+signedtoken = ["cryptography (>=3.0.0,<4)", "pyjwt (>=2.0.0,<3)"]
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "packaging"
-version = "20.9"
+version = "21.2"
 description = "Core utilities for Python packages"
 category = "main"
 optional = false
-python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
+python-versions = ">=3.6"
 
 [package.dependencies]
-pyparsing = ">=2.0.2"
+pyparsing = ">=2.0.2,<3"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "parso"
@@ -1635,6 +2373,11 @@ python-versions = ">=3.6"
 qa = ["flake8 (==3.8.3)", "mypy (==0.782)"]
 testing = ["docopt", "pytest (<6.0.0)"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "parsy"
 version = "1.1.0"
@@ -1643,22 +2386,37 @@ category = "dev"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "pathspec"
-version = "0.8.1"
+version = "0.9.0"
 description = "Utility library for gitignore style pattern matching of file paths."
 category = "dev"
 optional = false
-python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
+python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,>=2.7"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "pbr"
-version = "5.5.1"
+version = "5.7.0"
 description = "Python Build Reasonableness"
 category = "dev"
 optional = false
 python-versions = ">=2.6"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "persisting-theory"
 version = "0.2.1"
@@ -1667,6 +2425,11 @@ category = "main"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "pexpect"
 version = "4.8.0"
@@ -1678,25 +2441,40 @@ python-versions = "*"
 [package.dependencies]
 ptyprocess = ">=0.5"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "pg8000"
-version = "1.19.2"
+version = "1.22.0"
 description = "PostgreSQL interface library"
 category = "dev"
 optional = false
 python-versions = ">=3.6"
 
 [package.dependencies]
-scramp = "1.4.0"
+scramp = ">=1.4.1"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "phonenumbers"
-version = "8.12.21"
+version = "8.12.36"
 description = "Python version of Google's common library for parsing, formatting, storing and validating international phone numbers."
 category = "main"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "pickleshare"
 version = "0.7.5"
@@ -1705,31 +2483,44 @@ category = "main"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "pillow"
-version = "8.2.0"
+version = "8.4.0"
 description = "Python Imaging Library (Fork)"
 category = "main"
 optional = false
 python-versions = ">=3.6"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "pluggy"
-version = "0.13.1"
+version = "1.0.0"
 description = "plugin and hook calling mechanisms for python"
 category = "dev"
 optional = false
-python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
-
-[package.dependencies]
-importlib-metadata = {version = ">=0.12", markers = "python_version < \"3.8\""}
+python-versions = ">=3.6"
 
 [package.extras]
 dev = ["pre-commit", "tox"]
+testing = ["pytest", "pytest-benchmark"]
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "prometheus-client"
-version = "0.10.1"
+version = "0.12.0"
 description = "Python client for the Prometheus monitoring system."
 category = "main"
 optional = false
@@ -1738,17 +2529,27 @@ python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
 [package.extras]
 twisted = ["twisted"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "prompt-toolkit"
-version = "3.0.18"
+version = "3.0.22"
 description = "Library for building powerful interactive command lines in Python"
 category = "main"
 optional = false
-python-versions = ">=3.6.1"
+python-versions = ">=3.6.2"
 
 [package.dependencies]
 wcwidth = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "psutil"
 version = "5.8.0"
@@ -1760,13 +2561,23 @@ python-versions = ">=2.6, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
 [package.extras]
 test = ["ipaddress", "mock", "unittest2", "enum34", "pywin32", "wmi"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "psycopg2"
-version = "2.8.6"
+version = "2.9.1"
 description = "psycopg2 - Python-PostgreSQL Database Adapter"
 category = "main"
 optional = false
-python-versions = ">=2.7,!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*"
+python-versions = ">=3.6"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "ptyprocess"
@@ -1776,13 +2587,23 @@ category = "main"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "py"
-version = "1.10.0"
+version = "1.11.0"
 description = "library with cross-python path, ini-parsing, io, code, log facilities"
 category = "dev"
 optional = false
-python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
+python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "pyasn1"
@@ -1792,6 +2613,11 @@ category = "main"
 optional = true
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "pyasn1-modules"
 version = "0.2.8"
@@ -1803,6 +2629,11 @@ python-versions = "*"
 [package.dependencies]
 pyasn1 = ">=0.4.6,<0.5.0"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "pycodestyle"
 version = "2.7.0"
@@ -1811,17 +2642,40 @@ category = "dev"
 optional = false
 python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
+[[package]]
+name = "pycparser"
+version = "2.20"
+description = "C parser in Python"
+category = "main"
+optional = false
+python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "pycryptodome"
-version = "3.10.1"
+version = "3.11.0"
 description = "Cryptographic library for Python"
 category = "main"
 optional = false
 python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "pydocstyle"
-version = "6.0.0"
+version = "6.1.1"
 description = "Python docstring style checker"
 category = "dev"
 optional = false
@@ -1830,6 +2684,14 @@ python-versions = ">=3.6"
 [package.dependencies]
 snowballstemmer = "*"
 
+[package.extras]
+toml = ["toml"]
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "pyflakes"
 version = "2.3.1"
@@ -1838,26 +2700,45 @@ category = "dev"
 optional = false
 python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "pygments"
-version = "2.8.1"
+version = "2.10.0"
 description = "Pygments is a syntax highlighting package written in Python."
 category = "main"
 optional = false
 python-versions = ">=3.5"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "pyjwt"
-version = "1.7.1"
+version = "2.3.0"
 description = "JSON Web Token implementation in Python"
 category = "main"
 optional = false
-python-versions = "*"
+python-versions = ">=3.6"
+
+[package.dependencies]
+cryptography = {version = ">=3.3.1", optional = true, markers = "extra == \"crypto\""}
 
 [package.extras]
-crypto = ["cryptography (>=1.4)"]
-flake8 = ["flake8", "flake8-import-order", "pep8-naming"]
-test = ["pytest (>=4.0.1,<5.0.0)", "pytest-cov (>=2.6.0,<3.0.0)", "pytest-runner (>=4.2,<5.0.0)"]
+crypto = ["cryptography (>=3.3.1)"]
+dev = ["sphinx", "sphinx-rtd-theme", "zope.interface", "cryptography (>=3.3.1)", "pytest (>=6.0.0,<7.0.0)", "coverage[toml] (==5.0.4)", "mypy", "pre-commit"]
+docs = ["sphinx", "sphinx-rtd-theme", "zope.interface"]
+tests = ["pytest (>=6.0.0,<7.0.0)", "coverage[toml] (==5.0.4)"]
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "pyparsing"
@@ -1867,9 +2748,14 @@ category = "main"
 optional = false
 python-versions = ">=2.6, !=3.0.*, !=3.1.*, !=3.2.*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "pytest"
-version = "6.2.3"
+version = "6.2.5"
 description = "pytest: simple powerful testing with Python"
 category = "dev"
 optional = false
@@ -1879,19 +2765,23 @@ python-versions = ">=3.6"
 atomicwrites = {version = ">=1.0", markers = "sys_platform == \"win32\""}
 attrs = ">=19.2.0"
 colorama = {version = "*", markers = "sys_platform == \"win32\""}
-importlib-metadata = {version = ">=0.12", markers = "python_version < \"3.8\""}
 iniconfig = "*"
 packaging = "*"
-pluggy = ">=0.12,<1.0.0a1"
+pluggy = ">=0.12,<2.0"
 py = ">=1.8.2"
 toml = "*"
 
 [package.extras]
 testing = ["argcomplete", "hypothesis (>=3.56)", "mock", "nose", "requests", "xmlschema"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "pytest-cov"
-version = "2.11.1"
+version = "2.12.1"
 description = "Pytest plugin for measuring coverage."
 category = "dev"
 optional = false
@@ -1900,13 +2790,19 @@ python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
 [package.dependencies]
 coverage = ">=5.2.1"
 pytest = ">=4.6"
+toml = "*"
 
 [package.extras]
-testing = ["fields", "hunter", "process-tests (==2.0.2)", "six", "pytest-xdist", "virtualenv"]
+testing = ["fields", "hunter", "process-tests", "six", "pytest-xdist", "virtualenv"]
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "pytest-django"
-version = "4.2.0"
+version = "4.4.0"
 description = "A Django plugin for pytest."
 category = "dev"
 optional = false
@@ -1919,6 +2815,11 @@ pytest = ">=5.4.0"
 docs = ["sphinx", "sphinx-rtd-theme"]
 testing = ["django", "django-configurations (>=2.0)"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "pytest-django-testing-postgresql"
 version = "0.1.post0"
@@ -1931,6 +2832,11 @@ python-versions = "*"
 dj-database-url = "*"
 "testing.postgresql" = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "pytest-sugar"
 version = "0.9.4"
@@ -1944,9 +2850,14 @@ packaging = ">=14.1"
 pytest = ">=2.9"
 termcolor = ">=1.1.0"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "python-crontab"
-version = "2.5.1"
+version = "2.6.0"
 description = "Python Crontab API"
 category = "main"
 optional = false
@@ -1959,9 +2870,14 @@ python-dateutil = "*"
 cron-description = ["cron-descriptor"]
 cron-schedule = ["croniter"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "python-dateutil"
-version = "2.8.1"
+version = "2.8.2"
 description = "Extensions to the standard Python datetime module"
 category = "main"
 optional = false
@@ -1970,6 +2886,24 @@ python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,>=2.7"
 [package.dependencies]
 six = ">=1.5"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
+[[package]]
+name = "python-gnupg"
+version = "0.4.7"
+description = "A wrapper for the Gnu Privacy Guard (GPG or GnuPG)"
+category = "main"
+optional = false
+python-versions = "*"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "python-ldap"
 version = "3.3.1"
@@ -1982,21 +2916,56 @@ python-versions = ">=2.7,!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*"
 pyasn1 = ">=0.3.7"
 pyasn1_modules = ">=0.1.5"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
+[[package]]
+name = "python3-openid"
+version = "3.2.0"
+description = "OpenID support for modern servers and consumers."
+category = "main"
+optional = false
+python-versions = "*"
+
+[package.dependencies]
+defusedxml = "*"
+
+[package.extras]
+mysql = ["mysql-connector-python"]
+postgresql = ["psycopg2"]
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "pytz"
-version = "2021.1"
+version = "2021.3"
 description = "World timezone definitions, modern and historical"
 category = "main"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "pyyaml"
-version = "5.4.1"
+version = "6.0"
 description = "YAML parser and emitter for Python"
 category = "dev"
 optional = false
-python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, !=3.5.*"
+python-versions = ">=3.6"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "qrcode"
@@ -2016,6 +2985,11 @@ maintainer = ["zest.releaser"]
 pil = ["pillow"]
 test = ["pytest", "pytest-cov", "mock"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "redis"
 version = "3.5.3"
@@ -2027,31 +3001,66 @@ python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
 [package.extras]
 hiredis = ["hiredis (>=0.1.3)"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "regex"
-version = "2021.4.4"
+version = "2021.11.2"
 description = "Alternative regular expression module, to replace re."
 category = "dev"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "requests"
-version = "2.25.1"
+version = "2.26.0"
 description = "Python HTTP for Humans."
 category = "main"
 optional = false
-python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
+python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, !=3.5.*"
 
 [package.dependencies]
 certifi = ">=2017.4.17"
-chardet = ">=3.0.2,<5"
-idna = ">=2.5,<3"
+charset-normalizer = {version = ">=2.0.0,<2.1.0", markers = "python_version >= \"3\""}
+idna = {version = ">=2.5,<4", markers = "python_version >= \"3\""}
 urllib3 = ">=1.21.1,<1.27"
 
 [package.extras]
-security = ["pyOpenSSL (>=0.14)", "cryptography (>=1.3.4)"]
 socks = ["PySocks (>=1.5.6,!=1.5.7)", "win-inet-pton"]
+use_chardet_on_py3 = ["chardet (>=3.0.2,<5)"]
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
+[[package]]
+name = "requests-oauthlib"
+version = "1.3.0"
+description = "OAuthlib authentication support for Requests."
+category = "main"
+optional = false
+python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
+
+[package.dependencies]
+oauthlib = ">=3.0.0"
+requests = ">=2.0.0"
+
+[package.extras]
+rsa = ["oauthlib[signedtoken] (>=3.0.0)"]
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "restructuredtext-lint"
@@ -2064,9 +3073,14 @@ python-versions = "*"
 [package.dependencies]
 docutils = ">=0.11,<1.0"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "ruamel.yaml"
-version = "0.17.4"
+version = "0.17.17"
 description = "ruamel.yaml is a YAML parser/emitter that supports roundtrip preservation of comments, seq/map flow style, and map key order"
 category = "main"
 optional = false
@@ -2079,13 +3093,23 @@ python-versions = ">=3"
 docs = ["ryd"]
 jinja2 = ["ruamel.yaml.jinja2 (>=0.2)"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "ruamel.yaml.clib"
-version = "0.2.2"
+version = "0.2.6"
 description = "C version of reader, parser and emitter for ruamel.yaml derived from libyaml"
 category = "main"
 optional = false
-python-versions = "*"
+python-versions = ">=3.5"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "rules"
@@ -2095,17 +3119,30 @@ category = "main"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "s3transfer"
-version = "0.3.7"
+version = "0.5.0"
 description = "An Amazon S3 Transfer Manager"
 category = "main"
 optional = true
-python-versions = "*"
+python-versions = ">= 3.6"
 
 [package.dependencies]
 botocore = ">=1.12.36,<2.0a.0"
 
+[package.extras]
+crt = ["botocore[crt] (>=1.20.29,<2.0a.0)"]
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "safety"
 version = "1.10.3"
@@ -2120,16 +3157,26 @@ dparse = ">=0.5.1"
 packaging = "*"
 requests = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "scramp"
-version = "1.4.0"
+version = "1.4.1"
 description = "An implementation of the SCRAM protocol."
 category = "dev"
 optional = false
 python-versions = ">=3.6"
 
 [package.dependencies]
-asn1crypto = "1.4.0"
+asn1crypto = ">=1.4.0"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "selenium"
@@ -2142,21 +3189,70 @@ python-versions = "*"
 [package.dependencies]
 urllib3 = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
+[[package]]
+name = "sentry-sdk"
+version = "1.4.3"
+description = "Python client for Sentry (https://sentry.io)"
+category = "main"
+optional = true
+python-versions = "*"
+
+[package.dependencies]
+certifi = "*"
+urllib3 = ">=1.10.0"
+
+[package.extras]
+aiohttp = ["aiohttp (>=3.5)"]
+beam = ["apache-beam (>=2.12)"]
+bottle = ["bottle (>=0.12.13)"]
+celery = ["celery (>=3)"]
+chalice = ["chalice (>=1.16.0)"]
+django = ["django (>=1.8)"]
+falcon = ["falcon (>=1.4)"]
+flask = ["flask (>=0.11)", "blinker (>=1.1)"]
+httpx = ["httpx (>=0.16.0)"]
+pure_eval = ["pure-eval", "executing", "asttokens"]
+pyspark = ["pyspark (>=2.4.4)"]
+rq = ["rq (>=0.6)"]
+sanic = ["sanic (>=0.8)"]
+sqlalchemy = ["sqlalchemy (>=1.2)"]
+tornado = ["tornado (>=5)"]
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "six"
-version = "1.15.0"
+version = "1.16.0"
 description = "Python 2 and 3 compatibility utilities"
 category = "main"
 optional = false
 python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "smmap"
-version = "4.0.0"
+version = "5.0.0"
 description = "A pure Python implementation of a sliding window memory map manager"
 category = "dev"
 optional = false
-python-versions = ">=3.5"
+python-versions = ">=3.6"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "snowballstemmer"
@@ -2166,14 +3262,24 @@ category = "dev"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "soupsieve"
-version = "2.2.1"
+version = "2.3"
 description = "A modern CSS selector implementation for Beautiful Soup."
 category = "main"
 optional = false
 python-versions = ">=3.6"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "spdx-license-list"
 version = "0.5.2"
@@ -2182,6 +3288,11 @@ category = "main"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "sphinx"
 version = "3.5.4"
@@ -2213,6 +3324,11 @@ docs = ["sphinxcontrib-websupport"]
 lint = ["flake8 (>=3.5.0)", "isort", "mypy (>=0.800)", "docutils-stubs"]
 test = ["pytest", "pytest-cov", "html5lib", "cython", "typed-ast"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "sphinx-autodoc-typehints"
 version = "1.12.0"
@@ -2228,6 +3344,24 @@ Sphinx = ">=3.0"
 test = ["pytest (>=3.1.0)", "typing-extensions (>=3.5)", "sphobjinv (>=2.0)", "Sphinx (>=3.2.0)", "dataclasses"]
 type_comments = ["typed-ast (>=1.4.0)"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
+[[package]]
+name = "sphinx-materialdesign-theme"
+version = "0.1.11"
+description = "Sphinx Material Design Theme"
+category = "dev"
+optional = false
+python-versions = "*"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "sphinxcontrib-applehelp"
 version = "1.0.2"
@@ -2240,6 +3374,11 @@ python-versions = ">=3.5"
 lint = ["flake8", "mypy", "docutils-stubs"]
 test = ["pytest"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "sphinxcontrib-devhelp"
 version = "1.0.2"
@@ -2252,6 +3391,11 @@ python-versions = ">=3.5"
 lint = ["flake8", "mypy", "docutils-stubs"]
 test = ["pytest"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "sphinxcontrib-django"
 version = "0.5.1"
@@ -2260,18 +3404,28 @@ category = "dev"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "sphinxcontrib-htmlhelp"
-version = "1.0.3"
+version = "2.0.0"
 description = "sphinxcontrib-htmlhelp is a sphinx extension which renders HTML help files"
 category = "dev"
 optional = false
-python-versions = ">=3.5"
+python-versions = ">=3.6"
 
 [package.extras]
 lint = ["flake8", "mypy", "docutils-stubs"]
 test = ["pytest", "html5lib"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "sphinxcontrib-jsmath"
 version = "1.0.1"
@@ -2283,6 +3437,11 @@ python-versions = ">=3.5"
 [package.extras]
 test = ["pytest", "flake8", "mypy"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "sphinxcontrib-qthelp"
 version = "1.0.3"
@@ -2295,9 +3454,14 @@ python-versions = ">=3.5"
 lint = ["flake8", "mypy", "docutils-stubs"]
 test = ["pytest"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "sphinxcontrib-serializinghtml"
-version = "1.1.4"
+version = "1.1.5"
 description = "sphinxcontrib-serializinghtml is a sphinx extension which outputs \"serialized\" HTML files (json and pickle)."
 category = "dev"
 optional = false
@@ -2307,26 +3471,40 @@ python-versions = ">=3.5"
 lint = ["flake8", "mypy", "docutils-stubs"]
 test = ["pytest"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "sqlparse"
-version = "0.4.1"
+version = "0.4.2"
 description = "A non-validating SQL parser."
 category = "main"
 optional = false
 python-versions = ">=3.5"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "stevedore"
-version = "3.3.0"
+version = "3.5.0"
 description = "Manage dynamic plugins for Python applications"
 category = "dev"
 optional = false
 python-versions = ">=3.6"
 
 [package.dependencies]
-importlib-metadata = {version = ">=1.7.0", markers = "python_version < \"3.8\""}
 pbr = ">=2.0.0,<2.1.0 || >2.1.0"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "termcolor"
 version = "1.1.0"
@@ -2335,9 +3513,14 @@ category = "dev"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "testfixtures"
-version = "6.17.1"
+version = "6.18.3"
 description = "A collection of helpers and mock objects for unit tests and doc tests."
 category = "dev"
 optional = false
@@ -2348,6 +3531,11 @@ build = ["setuptools-git", "wheel", "twine"]
 docs = ["sphinx", "zope.component", "sybil", "twisted", "mock", "django (<2)", "django"]
 test = ["pytest (>=3.6)", "pytest-cov", "pytest-django", "zope.component", "sybil", "twisted", "mock", "django (<2)", "django"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "testing.common.database"
 version = "2.0.3"
@@ -2359,6 +3547,11 @@ python-versions = "*"
 [package.extras]
 testing = ["nose"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "testing.postgresql"
 version = "1.3.0"
@@ -2374,13 +3567,10 @@ pg8000 = ">=1.10"
 [package.extras]
 testing = ["sqlalchemy", "nose", "psycopg2"]
 
-[[package]]
-name = "text-unidecode"
-version = "1.3"
-description = "The most basic Text::Unidecode port"
-category = "main"
-optional = false
-python-versions = "*"
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "toml"
@@ -2390,46 +3580,44 @@ category = "main"
 optional = false
 python-versions = ">=2.6, !=3.0.*, !=3.1.*, !=3.2.*"
 
-[[package]]
-name = "tqdm"
-version = "4.60.0"
-description = "Fast, Extensible Progress Meter"
-category = "main"
-optional = false
-python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,>=2.7"
-
-[package.extras]
-dev = ["py-make (>=0.1.0)", "twine", "wheel"]
-notebook = ["ipywidgets (>=6)"]
-telegram = ["requests"]
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "traitlets"
-version = "5.0.5"
+version = "5.1.1"
 description = "Traitlets Python configuration system"
 category = "main"
 optional = false
 python-versions = ">=3.7"
 
-[package.dependencies]
-ipython-genutils = "*"
-
 [package.extras]
 test = ["pytest"]
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "twilio"
-version = "6.56.0"
+version = "7.3.0"
 description = "Twilio API client and TwiML generator"
 category = "main"
 optional = false
-python-versions = "*"
+python-versions = ">=3.6.0"
 
 [package.dependencies]
-PyJWT = "1.7.1"
+PyJWT = ">=2.0.0,<3.0.0"
 pytz = "*"
-requests = {version = ">=2.0.0", markers = "python_version >= \"3.0\""}
-six = "*"
+requests = ">=2.0.0"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "typed-ast"
@@ -2439,35 +3627,81 @@ category = "dev"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
+[[package]]
+name = "types-pytz"
+version = "2021.3.0"
+description = "Typing stubs for pytz"
+category = "dev"
+optional = false
+python-versions = "*"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
+[[package]]
+name = "types-pyyaml"
+version = "6.0.0"
+description = "Typing stubs for PyYAML"
+category = "dev"
+optional = false
+python-versions = "*"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "typing-extensions"
-version = "3.7.4.3"
+version = "3.10.0.2"
 description = "Backported and Experimental Type Hints for Python 3.5+"
-category = "main"
+category = "dev"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "urllib3"
-version = "1.26.4"
+version = "1.26.7"
 description = "HTTP library with thread-safe connection pooling, file post, and more."
 category = "main"
 optional = false
 python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, <4"
 
 [package.extras]
+brotli = ["brotlipy (>=0.6.0)"]
 secure = ["pyOpenSSL (>=0.14)", "cryptography (>=1.3.4)", "idna (>=2.0.0)", "certifi", "ipaddress"]
 socks = ["PySocks (>=1.5.6,!=1.5.7,<2.0)"]
-brotli = ["brotlipy (>=0.6.0)"]
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [[package]]
 name = "uwsgi"
-version = "2.0.19.1"
+version = "2.0.20"
 description = "The uWSGI server"
 category = "dev"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "vine"
 version = "5.0.0"
@@ -2476,6 +3710,11 @@ category = "main"
 optional = false
 python-versions = ">=3.6"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "wcwidth"
 version = "0.2.5"
@@ -2484,6 +3723,11 @@ category = "main"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "webencodings"
 version = "0.5.1"
@@ -2492,6 +3736,37 @@ category = "main"
 optional = false
 python-versions = "*"
 
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
+[[package]]
+name = "whoosh"
+version = "2.7.4"
+description = "Fast, pure-Python full text indexing, search, and spell checking library."
+category = "main"
+optional = false
+python-versions = "*"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
+[[package]]
+name = "wrapt"
+version = "1.13.3"
+description = "Module for decorators, wrappers and monkey patching."
+category = "main"
+optional = false
+python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,>=2.7"
+
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
+
 [[package]]
 name = "yubiotp"
 version = "1.0.0.post1"
@@ -2503,26 +3778,20 @@ python-versions = "*"
 [package.dependencies]
 pycryptodome = "*"
 
-[[package]]
-name = "zipp"
-version = "3.4.1"
-description = "Backport of pathlib-compatible object wrapper for zip files"
-category = "main"
-optional = false
-python-versions = ">=3.6"
-
-[package.extras]
-docs = ["sphinx", "jaraco.packaging (>=8.2)", "rst.linker (>=1.9)"]
-testing = ["pytest (>=4.6)", "pytest-checkdocs (>=1.2.3)", "pytest-flake8", "pytest-cov", "pytest-enabler", "jaraco.itertools", "func-timeout", "pytest-black (>=0.3.7)", "pytest-mypy"]
+[package.source]
+type = "legacy"
+url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
+reference = "gitlab"
 
 [extras]
 ldap = ["django-auth-ldap"]
 s3 = ["boto3", "django-storages"]
+sentry = []
 
 [metadata]
 lock-version = "1.1"
-python-versions = "^3.7"
-content-hash = "af4f0501b8a9a33699a20493a4786ddf21abbbbebea9ca6cdc456c31f6fd7bcd"
+python-versions = "^3.9"
+content-hash = "28e627cd320dc660f1e12088709cc04f4097a53edc3283b57c1385ef19a118de"
 
 [metadata.files]
 alabaster = [
@@ -2530,7 +3799,7 @@ alabaster = [
     {file = "alabaster-0.7.12.tar.gz", hash = "sha256:a661d72d58e6ea8a57f7a86e37d86716863ee5e92788398526d58b26a4e4dc02"},
 ]
 aleksis-builddeps = [
-    {file = "AlekSIS-Builddeps-3.tar.gz", hash = "sha256:04597e29a861e576d78adc068c9f1bf85450c81dc43cb1f7db0ed43e975b64a2"},
+    {file = "AlekSIS-Builddeps-4.tar.gz", hash = "sha256:aaaa22965228b9b9b7de812e3e7fa9cbfdbf8635bb22d6f3a201dc0cc6d8d307"},
 ]
 amqp = [
     {file = "amqp-5.0.6-py3-none-any.whl", hash = "sha256:493a2ac6788ce270a2f6a765b017299f60c1998f5a8617908ee9be082f7300fb"},
@@ -2545,8 +3814,8 @@ appnope = [
     {file = "appnope-0.1.2.tar.gz", hash = "sha256:dd83cd4b5b460958838f6eb3000c660b1f9caf2a5b1de4264e941512f603258a"},
 ]
 asgiref = [
-    {file = "asgiref-3.3.4-py3-none-any.whl", hash = "sha256:92906c611ce6c967347bbfea733f13d6313901d54dcca88195eaeb52b2a8e8ee"},
-    {file = "asgiref-3.3.4.tar.gz", hash = "sha256:d1216dfbdfb63826470995d31caed36225dcaf34f182e0fa257a4dd9e86f1b78"},
+    {file = "asgiref-3.4.1-py3-none-any.whl", hash = "sha256:ffc141aa908e6f175673e7b1b3b7af4fdb0ecb738fc5c8b88f69f055c2415214"},
+    {file = "asgiref-3.4.1.tar.gz", hash = "sha256:4ef1ab46b484e3c706329cedeff284a5d40824200638503f5768edb6de7d58e9"},
 ]
 asn1crypto = [
     {file = "asn1crypto-1.4.0-py2.py3-none-any.whl", hash = "sha256:4bcdf33c861c7d40bdcd74d8e4dd7661aac320fcdf40b9a3f95b4ee12fde2fa8"},
@@ -2557,12 +3826,12 @@ atomicwrites = [
     {file = "atomicwrites-1.4.0.tar.gz", hash = "sha256:ae70396ad1a434f9c7046fd2dd196fc04b12f9e91ffb859164193be8b6168a7a"},
 ]
 attrs = [
-    {file = "attrs-20.3.0-py2.py3-none-any.whl", hash = "sha256:31b2eced602aa8423c2aea9c76a724617ed67cf9513173fd3a4f03e3a929c7e6"},
-    {file = "attrs-20.3.0.tar.gz", hash = "sha256:832aa3cde19744e49938b91fea06d69ecb9e649c93ba974535d08ad92164f700"},
+    {file = "attrs-21.2.0-py2.py3-none-any.whl", hash = "sha256:149e90d6d8ac20db7a955ad60cf0e6881a3f20d37096140088356da6c716b0b1"},
+    {file = "attrs-21.2.0.tar.gz", hash = "sha256:ef6aaac3ca6cd92904cdd0d83f629a15f18053ec84e6432106f7a4d04ae4f5fb"},
 ]
 babel = [
-    {file = "Babel-2.9.0-py2.py3-none-any.whl", hash = "sha256:9d35c22fcc79893c3ecc85ac4a56cde1ecf3f19c540bba0922308a6c06ca6fa5"},
-    {file = "Babel-2.9.0.tar.gz", hash = "sha256:da031ab54472314f210b0adcff1588ee5d1d1d0ba4dbd07b94dba82bde791e05"},
+    {file = "Babel-2.9.1-py2.py3-none-any.whl", hash = "sha256:ab49e12b91d937cd11f0b67cb259a57ab4ad2b59ac7a3b41d6c06c0ac5b0def9"},
+    {file = "Babel-2.9.1.tar.gz", hash = "sha256:bc0c176f9f6a994582230df350aa6e05ba2ebe4b3ac317eab29d9be5d2768da0"},
 ]
 backcall = [
     {file = "backcall-0.2.0-py2.py3-none-any.whl", hash = "sha256:fbbce6a29f263178a1f7915c1940bde0ec2b2a967566fe1c65c1dfb7422bd255"},
@@ -2573,9 +3842,8 @@ bandit = [
     {file = "bandit-1.7.0.tar.gz", hash = "sha256:8a4c7415254d75df8ff3c3b15cfe9042ecee628a1e40b44c15a98890fbfc2608"},
 ]
 beautifulsoup4 = [
-    {file = "beautifulsoup4-4.9.3-py2-none-any.whl", hash = "sha256:4c98143716ef1cb40bf7f39a8e3eec8f8b009509e74904ba3a7b315431577e35"},
-    {file = "beautifulsoup4-4.9.3-py3-none-any.whl", hash = "sha256:fff47e031e34ec82bf17e00da8f592fe7de69aeea38be00523c04623c04fb666"},
-    {file = "beautifulsoup4-4.9.3.tar.gz", hash = "sha256:84729e322ad1d5b4d25f805bfa05b902dd96450f43842c4e99067d5e1369eb25"},
+    {file = "beautifulsoup4-4.10.0-py3-none-any.whl", hash = "sha256:9a315ce70049920ea4572a4055bc4bd700c940521d36fc858205ad4fcde149bf"},
+    {file = "beautifulsoup4-4.10.0.tar.gz", hash = "sha256:c23ad23c521d818955a4151a67d81580319d4bf548d3d49f4223ae041ff98891"},
 ]
 billiard = [
     {file = "billiard-3.6.4.0-py3-none-any.whl", hash = "sha256:87103ea78fa6ab4d5c751c4909bcff74617d985de7fa8b672cf8618afd5a875b"},
@@ -2586,20 +3854,20 @@ black = [
     {file = "black-19.10b0.tar.gz", hash = "sha256:c2edb73a08e9e0e6f65a0e6af18b059b8b1cdd5bef997d7a0b181df93dc81539"},
 ]
 bleach = [
-    {file = "bleach-3.3.0-py2.py3-none-any.whl", hash = "sha256:6123ddc1052673e52bab52cdc955bcb57a015264a1c57d37bea2f6b817af0125"},
-    {file = "bleach-3.3.0.tar.gz", hash = "sha256:98b3170739e5e83dd9dc19633f074727ad848cbedb6026708c8ac2d3b697a433"},
+    {file = "bleach-4.1.0-py2.py3-none-any.whl", hash = "sha256:4d2651ab93271d1129ac9cbc679f524565cc8a1b791909c4a51eac4446a15994"},
+    {file = "bleach-4.1.0.tar.gz", hash = "sha256:0900d8b37eba61a802ee40ac0061f8c2b5dee29c1927dd1d233e075ebf5a71da"},
 ]
 "boolean.py" = [
     {file = "boolean.py-3.8-py2.py3-none-any.whl", hash = "sha256:d75da0fd0354425fa64f6bbc6cec6ae1485d0eec3447b73187ff8cbf9b572e26"},
     {file = "boolean.py-3.8.tar.gz", hash = "sha256:cc24e20f985d60cd4a3a5a1c0956dd12611159d32a75081dabd0c9ab981acaa4"},
 ]
 boto3 = [
-    {file = "boto3-1.17.53-py2.py3-none-any.whl", hash = "sha256:3bf3305571f3c8b738a53e9e7dcff59137dffe94670046c084a17f9fa4599ff3"},
-    {file = "boto3-1.17.53.tar.gz", hash = "sha256:1d26f6e7ae3c940cb07119077ac42485dcf99164350da0ab50d0f5ad345800cd"},
+    {file = "boto3-1.19.12-py3-none-any.whl", hash = "sha256:b9105554477978e80fda1103ff21ecf07502080667730e45383e1d3951c87954"},
+    {file = "boto3-1.19.12.tar.gz", hash = "sha256:182a2b756a2c2180b473bc8452227062394a24e3701548be23ebc30d85976c64"},
 ]
 botocore = [
-    {file = "botocore-1.20.53-py2.py3-none-any.whl", hash = "sha256:d5e70d17b91c9b5867be7d6de0caa7dde9ed789bed62f03ea9b60718dc9350bf"},
-    {file = "botocore-1.20.53.tar.gz", hash = "sha256:e303500c4e80f6a706602da53daa6f751cfa8f491665c99a24ee732ab6321573"},
+    {file = "botocore-1.22.12-py3-none-any.whl", hash = "sha256:1d1094fb53ebe4535d8840fbd7c14aadb65bde7ff03a65f9a4f1d76bd03e16ff"},
+    {file = "botocore-1.22.12.tar.gz", hash = "sha256:fc59b55e8c5dde64b017b2f114c25f8cce397b667e812aea7eafb4b59b49d7cb"},
 ]
 bs4 = [
     {file = "bs4-0.0.1.tar.gz", hash = "sha256:36ecea1fd7cc5c0c6e4a1ff075df26d50da647b75376626cc186e2212886dd3a"},
@@ -2609,38 +3877,92 @@ calendarweek = [
     {file = "calendarweek-0.5.0.tar.gz", hash = "sha256:32f5c8663799a2f5a0b8909976c7a3ae77397acd7e7c31d1456ece5b452988a5"},
 ]
 celery = [
-    {file = "celery-5.0.5-py3-none-any.whl", hash = "sha256:5e8d364e058554e83bbb116e8377d90c79be254785f357cb2cec026e79febe13"},
-    {file = "celery-5.0.5.tar.gz", hash = "sha256:f4efebe6f8629b0da2b8e529424de376494f5b7a743c321c8a2ddc2b1414921c"},
+    {file = "celery-5.1.2-py3-none-any.whl", hash = "sha256:9dab2170b4038f7bf10ef2861dbf486ddf1d20592290a1040f7b7a1259705d42"},
+    {file = "celery-5.1.2.tar.gz", hash = "sha256:8d9a3de9162965e97f8e8cc584c67aad83b3f7a267584fa47701ed11c3e0d4b0"},
 ]
 celery-haystack-ng = [
     {file = "celery-haystack-ng-0.20.post2.tar.gz", hash = "sha256:d2e077851f13dddc36fc86134c7c8a937e46ae75e576eb8e77e03b03977fc7bb"},
     {file = "celery_haystack_ng-0.20.post2-py2.py3-none-any.whl", hash = "sha256:a13e00f2c29411b06c6cdf59ad6a90b6c158e3384e7ec6d6d64f6a69e8ff299a"},
 ]
 celery-progress = [
-    {file = "celery_progress-0.1.0-py3-none-any.whl", hash = "sha256:01bc7ecb2483ed7085b957413a392f85b7e1002fc8ce6d24f3d1ff264173002d"},
+    {file = "celery-progress-0.1.1.tar.gz", hash = "sha256:b2622d1b410a763412810f0293153c984f4a0220b76769bd701b5b45e583ddad"},
+    {file = "celery_progress-0.1.1-py3-none-any.whl", hash = "sha256:36a1e58b4408c9bf6aa63908204b50960b005db8e13f3c94ce6f8d6a2a4d4a6c"},
 ]
 certifi = [
-    {file = "certifi-2020.12.5-py2.py3-none-any.whl", hash = "sha256:719a74fb9e33b9bd44cc7f3a8d94bc35e4049deebe19ba7d8e108280cfd59830"},
-    {file = "certifi-2020.12.5.tar.gz", hash = "sha256:1a4995114262bffbc2413b159f2a1a480c969de6e6eb13ee966d470af86af59c"},
-]
-chardet = [
-    {file = "chardet-4.0.0-py2.py3-none-any.whl", hash = "sha256:f864054d66fd9118f2e67044ac8981a54775ec5b67aed0441892edb553d21da5"},
-    {file = "chardet-4.0.0.tar.gz", hash = "sha256:0d6f53a15db4120f2b08c94f11e7d93d2c911ee118b6b30a04ec3ee8310179fa"},
+    {file = "certifi-2021.10.8-py2.py3-none-any.whl", hash = "sha256:d62a0163eb4c2344ac042ab2bdf75399a71a2d8c7d47eac2e2ee91b9d6339569"},
+    {file = "certifi-2021.10.8.tar.gz", hash = "sha256:78884e7c1d4b00ce3cea67b44566851c4343c120abd683433ce934a68ea58872"},
+]
+cffi = [
+    {file = "cffi-1.15.0-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:c2502a1a03b6312837279c8c1bd3ebedf6c12c4228ddbad40912d671ccc8a962"},
+    {file = "cffi-1.15.0-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:23cfe892bd5dd8941608f93348c0737e369e51c100d03718f108bf1add7bd6d0"},
+    {file = "cffi-1.15.0-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:41d45de54cd277a7878919867c0f08b0cf817605e4eb94093e7516505d3c8d14"},
+    {file = "cffi-1.15.0-cp27-cp27m-win32.whl", hash = "sha256:4a306fa632e8f0928956a41fa8e1d6243c71e7eb59ffbd165fc0b41e316b2474"},
+    {file = "cffi-1.15.0-cp27-cp27m-win_amd64.whl", hash = "sha256:e7022a66d9b55e93e1a845d8c9eba2a1bebd4966cd8bfc25d9cd07d515b33fa6"},
+    {file = "cffi-1.15.0-cp27-cp27mu-manylinux1_i686.whl", hash = "sha256:14cd121ea63ecdae71efa69c15c5543a4b5fbcd0bbe2aad864baca0063cecf27"},
+    {file = "cffi-1.15.0-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:d4d692a89c5cf08a8557fdeb329b82e7bf609aadfaed6c0d79f5a449a3c7c023"},
+    {file = "cffi-1.15.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:0104fb5ae2391d46a4cb082abdd5c69ea4eab79d8d44eaaf79f1b1fd806ee4c2"},
+    {file = "cffi-1.15.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:91ec59c33514b7c7559a6acda53bbfe1b283949c34fe7440bcf917f96ac0723e"},
+    {file = "cffi-1.15.0-cp310-cp310-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:f5c7150ad32ba43a07c4479f40241756145a1f03b43480e058cfd862bf5041c7"},
+    {file = "cffi-1.15.0-cp310-cp310-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:00c878c90cb53ccfaae6b8bc18ad05d2036553e6d9d1d9dbcf323bbe83854ca3"},
+    {file = "cffi-1.15.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:abb9a20a72ac4e0fdb50dae135ba5e77880518e742077ced47eb1499e29a443c"},
+    {file = "cffi-1.15.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a5263e363c27b653a90078143adb3d076c1a748ec9ecc78ea2fb916f9b861962"},
+    {file = "cffi-1.15.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:f54a64f8b0c8ff0b64d18aa76675262e1700f3995182267998c31ae974fbc382"},
+    {file = "cffi-1.15.0-cp310-cp310-win32.whl", hash = "sha256:c21c9e3896c23007803a875460fb786118f0cdd4434359577ea25eb556e34c55"},
+    {file = "cffi-1.15.0-cp310-cp310-win_amd64.whl", hash = "sha256:5e069f72d497312b24fcc02073d70cb989045d1c91cbd53979366077959933e0"},
+    {file = "cffi-1.15.0-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:64d4ec9f448dfe041705426000cc13e34e6e5bb13736e9fd62e34a0b0c41566e"},
+    {file = "cffi-1.15.0-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2756c88cbb94231c7a147402476be2c4df2f6078099a6f4a480d239a8817ae39"},
+    {file = "cffi-1.15.0-cp36-cp36m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3b96a311ac60a3f6be21d2572e46ce67f09abcf4d09344c49274eb9e0bf345fc"},
+    {file = "cffi-1.15.0-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:75e4024375654472cc27e91cbe9eaa08567f7fbdf822638be2814ce059f58032"},
+    {file = "cffi-1.15.0-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:59888172256cac5629e60e72e86598027aca6bf01fa2465bdb676d37636573e8"},
+    {file = "cffi-1.15.0-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:27c219baf94952ae9d50ec19651a687b826792055353d07648a5695413e0c605"},
+    {file = "cffi-1.15.0-cp36-cp36m-win32.whl", hash = "sha256:4958391dbd6249d7ad855b9ca88fae690783a6be9e86df65865058ed81fc860e"},
+    {file = "cffi-1.15.0-cp36-cp36m-win_amd64.whl", hash = "sha256:f6f824dc3bce0edab5f427efcfb1d63ee75b6fcb7282900ccaf925be84efb0fc"},
+    {file = "cffi-1.15.0-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:06c48159c1abed75c2e721b1715c379fa3200c7784271b3c46df01383b593636"},
+    {file = "cffi-1.15.0-cp37-cp37m-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:c2051981a968d7de9dd2d7b87bcb9c939c74a34626a6e2f8181455dd49ed69e4"},
+    {file = "cffi-1.15.0-cp37-cp37m-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:fd8a250edc26254fe5b33be00402e6d287f562b6a5b2152dec302fa15bb3e997"},
+    {file = "cffi-1.15.0-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:91d77d2a782be4274da750752bb1650a97bfd8f291022b379bb8e01c66b4e96b"},
+    {file = "cffi-1.15.0-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:45db3a33139e9c8f7c09234b5784a5e33d31fd6907800b316decad50af323ff2"},
+    {file = "cffi-1.15.0-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:263cc3d821c4ab2213cbe8cd8b355a7f72a8324577dc865ef98487c1aeee2bc7"},
+    {file = "cffi-1.15.0-cp37-cp37m-win32.whl", hash = "sha256:17771976e82e9f94976180f76468546834d22a7cc404b17c22df2a2c81db0c66"},
+    {file = "cffi-1.15.0-cp37-cp37m-win_amd64.whl", hash = "sha256:3415c89f9204ee60cd09b235810be700e993e343a408693e80ce7f6a40108029"},
+    {file = "cffi-1.15.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:4238e6dab5d6a8ba812de994bbb0a79bddbdf80994e4ce802b6f6f3142fcc880"},
+    {file = "cffi-1.15.0-cp38-cp38-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:0808014eb713677ec1292301ea4c81ad277b6cdf2fdd90fd540af98c0b101d20"},
+    {file = "cffi-1.15.0-cp38-cp38-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:57e9ac9ccc3101fac9d6014fba037473e4358ef4e89f8e181f8951a2c0162024"},
+    {file = "cffi-1.15.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8b6c2ea03845c9f501ed1313e78de148cd3f6cad741a75d43a29b43da27f2e1e"},
+    {file = "cffi-1.15.0-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:10dffb601ccfb65262a27233ac273d552ddc4d8ae1bf93b21c94b8511bffe728"},
+    {file = "cffi-1.15.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:786902fb9ba7433aae840e0ed609f45c7bcd4e225ebb9c753aa39725bb3e6ad6"},
+    {file = "cffi-1.15.0-cp38-cp38-win32.whl", hash = "sha256:da5db4e883f1ce37f55c667e5c0de439df76ac4cb55964655906306918e7363c"},
+    {file = "cffi-1.15.0-cp38-cp38-win_amd64.whl", hash = "sha256:181dee03b1170ff1969489acf1c26533710231c58f95534e3edac87fff06c443"},
+    {file = "cffi-1.15.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:45e8636704eacc432a206ac7345a5d3d2c62d95a507ec70d62f23cd91770482a"},
+    {file = "cffi-1.15.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:31fb708d9d7c3f49a60f04cf5b119aeefe5644daba1cd2a0fe389b674fd1de37"},
+    {file = "cffi-1.15.0-cp39-cp39-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:6dc2737a3674b3e344847c8686cf29e500584ccad76204efea14f451d4cc669a"},
+    {file = "cffi-1.15.0-cp39-cp39-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:74fdfdbfdc48d3f47148976f49fab3251e550a8720bebc99bf1483f5bfb5db3e"},
+    {file = "cffi-1.15.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ffaa5c925128e29efbde7301d8ecaf35c8c60ffbcd6a1ffd3a552177c8e5e796"},
+    {file = "cffi-1.15.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3f7d084648d77af029acb79a0ff49a0ad7e9d09057a9bf46596dac9514dc07df"},
+    {file = "cffi-1.15.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:ef1f279350da2c586a69d32fc8733092fd32cc8ac95139a00377841f59a3f8d8"},
+    {file = "cffi-1.15.0-cp39-cp39-win32.whl", hash = "sha256:2a23af14f408d53d5e6cd4e3d9a24ff9e05906ad574822a10563efcef137979a"},
+    {file = "cffi-1.15.0-cp39-cp39-win_amd64.whl", hash = "sha256:3773c4d81e6e818df2efbc7dd77325ca0dcb688116050fb2b3011218eda36139"},
+    {file = "cffi-1.15.0.tar.gz", hash = "sha256:920f0d66a896c2d99f0adbb391f990a84091179542c205fa53ce5787aff87954"},
+]
+charset-normalizer = [
+    {file = "charset-normalizer-2.0.7.tar.gz", hash = "sha256:e019de665e2bcf9c2b64e2e5aa025fa991da8720daa3c1138cadd2fd1856aed0"},
+    {file = "charset_normalizer-2.0.7-py3-none-any.whl", hash = "sha256:f7af805c321bfa1ce6714c51f254e0d5bb5e5834039bc17db7ebe3a4cec9492b"},
 ]
 click = [
     {file = "click-7.1.2-py2.py3-none-any.whl", hash = "sha256:dacca89f4bfadd5de3d7489b7c8a566eee0d3676333fbb50030263894c38c0dc"},
     {file = "click-7.1.2.tar.gz", hash = "sha256:d2b5255c7c6349bc1bd1e59e08cd12acbbd63ce649f2588755783aa94dfb6b1a"},
 ]
 click-didyoumean = [
-    {file = "click-didyoumean-0.0.3.tar.gz", hash = "sha256:112229485c9704ff51362fe34b2d4f0b12fc71cc20f6d2b3afabed4b8bfa6aeb"},
+    {file = "click-didyoumean-0.3.0.tar.gz", hash = "sha256:f184f0d851d96b6d29297354ed981b7dd71df7ff500d82fa6d11f0856bee8035"},
+    {file = "click_didyoumean-0.3.0-py3-none-any.whl", hash = "sha256:a0713dc7a1de3f06bc0df5a9567ad19ead2d3d5689b434768a6145bff77c0667"},
 ]
 click-plugins = [
     {file = "click-plugins-1.1.1.tar.gz", hash = "sha256:46ab999744a9d831159c3411bb0c79346d94a444df9a3a3742e9ed63645f264b"},
     {file = "click_plugins-1.1.1-py2.py3-none-any.whl", hash = "sha256:5d262006d3222f5057fd81e1623d4443e41dcda5dc815c06b442aa3c02889fc8"},
 ]
 click-repl = [
-    {file = "click-repl-0.1.6.tar.gz", hash = "sha256:b9f29d52abc4d6059f8e276132a111ab8d94980afe6a5432b9d996544afa95d5"},
-    {file = "click_repl-0.1.6-py3-none-any.whl", hash = "sha256:9c4c3d022789cae912aad8a3f5e1d7c2cdd016ee1225b5212ad3e8691563cda5"},
+    {file = "click-repl-0.2.0.tar.gz", hash = "sha256:cd12f68d745bf6151210790540b4cb064c7b13e571bc64b6957d98d120dacfd8"},
+    {file = "click_repl-0.2.0-py3-none-any.whl", hash = "sha256:94b3fbbc9406a236f176e0506524b2937e4b23b6f4c0c0b2a0a83f8a64e9194b"},
 ]
 colorama = [
     {file = "colorama-0.4.4-py2.py3-none-any.whl", hash = "sha256:9f47eda37229f68eee03b24b9748937c7dc3868f906e8ba69fbcbdd3bc5dc3e2"},
@@ -2654,174 +3976,195 @@ configobj = [
     {file = "configobj-5.0.6.tar.gz", hash = "sha256:a2f5650770e1c87fb335af19a9b7eb73fc05ccf22144eb68db7d00cd2bcb0902"},
 ]
 coverage = [
-    {file = "coverage-5.5-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:b6d534e4b2ab35c9f93f46229363e17f63c53ad01330df9f2d6bd1187e5eaacf"},
-    {file = "coverage-5.5-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:b7895207b4c843c76a25ab8c1e866261bcfe27bfaa20c192de5190121770672b"},
-    {file = "coverage-5.5-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:c2723d347ab06e7ddad1a58b2a821218239249a9e4365eaff6649d31180c1669"},
-    {file = "coverage-5.5-cp27-cp27m-manylinux2010_i686.whl", hash = "sha256:900fbf7759501bc7807fd6638c947d7a831fc9fdf742dc10f02956ff7220fa90"},
-    {file = "coverage-5.5-cp27-cp27m-manylinux2010_x86_64.whl", hash = "sha256:004d1880bed2d97151facef49f08e255a20ceb6f9432df75f4eef018fdd5a78c"},
-    {file = "coverage-5.5-cp27-cp27m-win32.whl", hash = "sha256:06191eb60f8d8a5bc046f3799f8a07a2d7aefb9504b0209aff0b47298333302a"},
-    {file = "coverage-5.5-cp27-cp27m-win_amd64.whl", hash = "sha256:7501140f755b725495941b43347ba8a2777407fc7f250d4f5a7d2a1050ba8e82"},
-    {file = "coverage-5.5-cp27-cp27mu-manylinux1_i686.whl", hash = "sha256:372da284cfd642d8e08ef606917846fa2ee350f64994bebfbd3afb0040436905"},
-    {file = "coverage-5.5-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:8963a499849a1fc54b35b1c9f162f4108017b2e6db2c46c1bed93a72262ed083"},
-    {file = "coverage-5.5-cp27-cp27mu-manylinux2010_i686.whl", hash = "sha256:869a64f53488f40fa5b5b9dcb9e9b2962a66a87dab37790f3fcfb5144b996ef5"},
-    {file = "coverage-5.5-cp27-cp27mu-manylinux2010_x86_64.whl", hash = "sha256:4a7697d8cb0f27399b0e393c0b90f0f1e40c82023ea4d45d22bce7032a5d7b81"},
-    {file = "coverage-5.5-cp310-cp310-macosx_10_14_x86_64.whl", hash = "sha256:8d0a0725ad7c1a0bcd8d1b437e191107d457e2ec1084b9f190630a4fb1af78e6"},
-    {file = "coverage-5.5-cp310-cp310-manylinux1_x86_64.whl", hash = "sha256:51cb9476a3987c8967ebab3f0fe144819781fca264f57f89760037a2ea191cb0"},
-    {file = "coverage-5.5-cp310-cp310-win_amd64.whl", hash = "sha256:c0891a6a97b09c1f3e073a890514d5012eb256845c451bd48f7968ef939bf4ae"},
-    {file = "coverage-5.5-cp35-cp35m-macosx_10_9_x86_64.whl", hash = "sha256:3487286bc29a5aa4b93a072e9592f22254291ce96a9fbc5251f566b6b7343cdb"},
-    {file = "coverage-5.5-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:deee1077aae10d8fa88cb02c845cfba9b62c55e1183f52f6ae6a2df6a2187160"},
-    {file = "coverage-5.5-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:f11642dddbb0253cc8853254301b51390ba0081750a8ac03f20ea8103f0c56b6"},
-    {file = "coverage-5.5-cp35-cp35m-manylinux2010_i686.whl", hash = "sha256:6c90e11318f0d3c436a42409f2749ee1a115cd8b067d7f14c148f1ce5574d701"},
-    {file = "coverage-5.5-cp35-cp35m-manylinux2010_x86_64.whl", hash = "sha256:30c77c1dc9f253283e34c27935fded5015f7d1abe83bc7821680ac444eaf7793"},
-    {file = "coverage-5.5-cp35-cp35m-win32.whl", hash = "sha256:9a1ef3b66e38ef8618ce5fdc7bea3d9f45f3624e2a66295eea5e57966c85909e"},
-    {file = "coverage-5.5-cp35-cp35m-win_amd64.whl", hash = "sha256:972c85d205b51e30e59525694670de6a8a89691186012535f9d7dbaa230e42c3"},
-    {file = "coverage-5.5-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:af0e781009aaf59e25c5a678122391cb0f345ac0ec272c7961dc5455e1c40066"},
-    {file = "coverage-5.5-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:74d881fc777ebb11c63736622b60cb9e4aee5cace591ce274fb69e582a12a61a"},
-    {file = "coverage-5.5-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:92b017ce34b68a7d67bd6d117e6d443a9bf63a2ecf8567bb3d8c6c7bc5014465"},
-    {file = "coverage-5.5-cp36-cp36m-manylinux2010_i686.whl", hash = "sha256:d636598c8305e1f90b439dbf4f66437de4a5e3c31fdf47ad29542478c8508bbb"},
-    {file = "coverage-5.5-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:41179b8a845742d1eb60449bdb2992196e211341818565abded11cfa90efb821"},
-    {file = "coverage-5.5-cp36-cp36m-win32.whl", hash = "sha256:040af6c32813fa3eae5305d53f18875bedd079960822ef8ec067a66dd8afcd45"},
-    {file = "coverage-5.5-cp36-cp36m-win_amd64.whl", hash = "sha256:5fec2d43a2cc6965edc0bb9e83e1e4b557f76f843a77a2496cbe719583ce8184"},
-    {file = "coverage-5.5-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:18ba8bbede96a2c3dde7b868de9dcbd55670690af0988713f0603f037848418a"},
-    {file = "coverage-5.5-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:2910f4d36a6a9b4214bb7038d537f015346f413a975d57ca6b43bf23d6563b53"},
-    {file = "coverage-5.5-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:f0b278ce10936db1a37e6954e15a3730bea96a0997c26d7fee88e6c396c2086d"},
-    {file = "coverage-5.5-cp37-cp37m-manylinux2010_i686.whl", hash = "sha256:796c9c3c79747146ebd278dbe1e5c5c05dd6b10cc3bcb8389dfdf844f3ead638"},
-    {file = "coverage-5.5-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:53194af30d5bad77fcba80e23a1441c71abfb3e01192034f8246e0d8f99528f3"},
-    {file = "coverage-5.5-cp37-cp37m-win32.whl", hash = "sha256:184a47bbe0aa6400ed2d41d8e9ed868b8205046518c52464fde713ea06e3a74a"},
-    {file = "coverage-5.5-cp37-cp37m-win_amd64.whl", hash = "sha256:2949cad1c5208b8298d5686d5a85b66aae46d73eec2c3e08c817dd3513e5848a"},
-    {file = "coverage-5.5-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:217658ec7187497e3f3ebd901afdca1af062b42cfe3e0dafea4cced3983739f6"},
-    {file = "coverage-5.5-cp38-cp38-manylinux1_i686.whl", hash = "sha256:1aa846f56c3d49205c952d8318e76ccc2ae23303351d9270ab220004c580cfe2"},
-    {file = "coverage-5.5-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:24d4a7de75446be83244eabbff746d66b9240ae020ced65d060815fac3423759"},
-    {file = "coverage-5.5-cp38-cp38-manylinux2010_i686.whl", hash = "sha256:d1f8bf7b90ba55699b3a5e44930e93ff0189aa27186e96071fac7dd0d06a1873"},
-    {file = "coverage-5.5-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:970284a88b99673ccb2e4e334cfb38a10aab7cd44f7457564d11898a74b62d0a"},
-    {file = "coverage-5.5-cp38-cp38-win32.whl", hash = "sha256:01d84219b5cdbfc8122223b39a954820929497a1cb1422824bb86b07b74594b6"},
-    {file = "coverage-5.5-cp38-cp38-win_amd64.whl", hash = "sha256:2e0d881ad471768bf6e6c2bf905d183543f10098e3b3640fc029509530091502"},
-    {file = "coverage-5.5-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:d1f9ce122f83b2305592c11d64f181b87153fc2c2bbd3bb4a3dde8303cfb1a6b"},
-    {file = "coverage-5.5-cp39-cp39-manylinux1_i686.whl", hash = "sha256:13c4ee887eca0f4c5a247b75398d4114c37882658300e153113dafb1d76de529"},
-    {file = "coverage-5.5-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:52596d3d0e8bdf3af43db3e9ba8dcdaac724ba7b5ca3f6358529d56f7a166f8b"},
-    {file = "coverage-5.5-cp39-cp39-manylinux2010_i686.whl", hash = "sha256:2cafbbb3af0733db200c9b5f798d18953b1a304d3f86a938367de1567f4b5bff"},
-    {file = "coverage-5.5-cp39-cp39-manylinux2010_x86_64.whl", hash = "sha256:44d654437b8ddd9eee7d1eaee28b7219bec228520ff809af170488fd2fed3e2b"},
-    {file = "coverage-5.5-cp39-cp39-win32.whl", hash = "sha256:d314ed732c25d29775e84a960c3c60808b682c08d86602ec2c3008e1202e3bb6"},
-    {file = "coverage-5.5-cp39-cp39-win_amd64.whl", hash = "sha256:13034c4409db851670bc9acd836243aeee299949bd5673e11844befcb0149f03"},
-    {file = "coverage-5.5-pp36-none-any.whl", hash = "sha256:f030f8873312a16414c0d8e1a1ddff2d3235655a2174e3648b4fa66b3f2f1079"},
-    {file = "coverage-5.5-pp37-none-any.whl", hash = "sha256:2a3859cb82dcbda1cfd3e6f71c27081d18aa251d20a17d87d26d4cd216fb0af4"},
-    {file = "coverage-5.5.tar.gz", hash = "sha256:ebe78fe9a0e874362175b02371bdfbee64d8edc42a044253ddf4ee7d3c15212c"},
+    {file = "coverage-6.1.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:42a1fb5dee3355df90b635906bb99126faa7936d87dfc97eacc5293397618cb7"},
+    {file = "coverage-6.1.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a00284dbfb53b42e35c7dd99fc0e26ef89b4a34efff68078ed29d03ccb28402a"},
+    {file = "coverage-6.1.1-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:51a441011a30d693e71dea198b2a6f53ba029afc39f8e2aeb5b77245c1b282ef"},
+    {file = "coverage-6.1.1-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:e76f017b6d4140a038c5ff12be1581183d7874e41f1c0af58ecf07748d36a336"},
+    {file = "coverage-6.1.1-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:7833c872718dc913f18e51ee97ea0dece61d9930893a58b20b3daf09bb1af6b6"},
+    {file = "coverage-6.1.1-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:8186b5a4730c896cbe1e4b645bdc524e62d874351ae50e1db7c3e9f5dc81dc26"},
+    {file = "coverage-6.1.1-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:bbca34dca5a2d60f81326d908d77313816fad23d11b6069031a3d6b8c97a54f9"},
+    {file = "coverage-6.1.1-cp310-cp310-win32.whl", hash = "sha256:72bf437d54186d104388cbae73c9f2b0f8a3e11b6e8d7deb593bd14625c96026"},
+    {file = "coverage-6.1.1-cp310-cp310-win_amd64.whl", hash = "sha256:994ce5a7b3d20981b81d83618aa4882f955bfa573efdbef033d5632b58597ba9"},
+    {file = "coverage-6.1.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:ab6a0fe4c96f8058d41948ddf134420d3ef8c42d5508b5a341a440cce7a37a1d"},
+    {file = "coverage-6.1.1-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:10ab138b153e4cc408b43792cb7f518f9ee02f4ff55cd1ab67ad6fd7e9905c7e"},
+    {file = "coverage-6.1.1-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:7e083d32965d2eb6638a77e65b622be32a094fdc0250f28ce6039b0732fbcaa8"},
+    {file = "coverage-6.1.1-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:359a32515e94e398a5c0fa057e5887a42e647a9502d8e41165cf5cb8d3d1ca67"},
+    {file = "coverage-6.1.1-cp36-cp36m-musllinux_1_1_aarch64.whl", hash = "sha256:bf656cd74ff7b4ed7006cdb2a6728150aaad69c7242b42a2a532f77b63ea233f"},
+    {file = "coverage-6.1.1-cp36-cp36m-musllinux_1_1_i686.whl", hash = "sha256:dc5023be1c2a8b0a0ab5e31389e62c28b2453eb31dd069f4b8d1a0f9814d951a"},
+    {file = "coverage-6.1.1-cp36-cp36m-musllinux_1_1_x86_64.whl", hash = "sha256:557594a50bfe3fb0b1b57460f6789affe8850ad19c1acf2d14a3e12b2757d489"},
+    {file = "coverage-6.1.1-cp36-cp36m-win32.whl", hash = "sha256:9eb0a1923354e0fdd1c8a6f53f5db2e6180d670e2b587914bf2e79fa8acfd003"},
+    {file = "coverage-6.1.1-cp36-cp36m-win_amd64.whl", hash = "sha256:04a92a6cf9afd99f9979c61348ec79725a9f9342fb45e63c889e33c04610d97b"},
+    {file = "coverage-6.1.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:479228e1b798d3c246ac89b09897ee706c51b3e5f8f8d778067f38db73ccc717"},
+    {file = "coverage-6.1.1-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:78287731e3601ea5ce9d6468c82d88a12ef8fe625d6b7bdec9b45d96c1ad6533"},
+    {file = "coverage-6.1.1-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:c95257aa2ccf75d3d91d772060538d5fea7f625e48157f8ca44594f94d41cb33"},
+    {file = "coverage-6.1.1-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:9ad5895938a894c368d49d8470fe9f519909e5ebc6b8f8ea5190bd0df6aa4271"},
+    {file = "coverage-6.1.1-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:326d944aad0189603733d646e8d4a7d952f7145684da973c463ec2eefe1387c2"},
+    {file = "coverage-6.1.1-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:e7d5606b9240ed4def9cbdf35be4308047d11e858b9c88a6c26974758d6225ce"},
+    {file = "coverage-6.1.1-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:572f917267f363101eec375c109c9c1118037c7cc98041440b5eabda3185ac7b"},
+    {file = "coverage-6.1.1-cp37-cp37m-win32.whl", hash = "sha256:35cd2230e1ed76df7d0081a997f0fe705be1f7d8696264eb508076e0d0b5a685"},
+    {file = "coverage-6.1.1-cp37-cp37m-win_amd64.whl", hash = "sha256:65ad3ff837c89a229d626b8004f0ee32110f9bfdb6a88b76a80df36ccc60d926"},
+    {file = "coverage-6.1.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:977ce557d79577a3dd510844904d5d968bfef9489f512be65e2882e1c6eed7d8"},
+    {file = "coverage-6.1.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:62512c0ec5d307f56d86504c58eace11c1bc2afcdf44e3ff20de8ca427ca1d0e"},
+    {file = "coverage-6.1.1-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:2e5b9c17a56b8bf0c0a9477fcd30d357deb486e4e1b389ed154f608f18556c8a"},
+    {file = "coverage-6.1.1-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:666c6b32b69e56221ad1551d377f718ed00e6167c7a1b9257f780b105a101271"},
+    {file = "coverage-6.1.1-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:fb2fa2f6506c03c48ca42e3fe5a692d7470d290c047ee6de7c0f3e5fa7639ac9"},
+    {file = "coverage-6.1.1-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:f0f80e323a17af63eac6a9db0c9188c10f1fd815c3ab299727150cc0eb92c7a4"},
+    {file = "coverage-6.1.1-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:738e823a746841248b56f0f3bd6abf3b73af191d1fd65e4c723b9c456216f0ad"},
+    {file = "coverage-6.1.1-cp38-cp38-win32.whl", hash = "sha256:8605add58e6a960729aa40c0fd9a20a55909dd9b586d3e8104cc7f45869e4c6b"},
+    {file = "coverage-6.1.1-cp38-cp38-win_amd64.whl", hash = "sha256:6e994003e719458420e14ffb43c08f4c14990e20d9e077cb5cad7a3e419bbb54"},
+    {file = "coverage-6.1.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:e3c4f5211394cd0bf6874ac5d29684a495f9c374919833dcfff0bd6d37f96201"},
+    {file = "coverage-6.1.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e14bceb1f3ae8a14374be2b2d7bc12a59226872285f91d66d301e5f41705d4d6"},
+    {file = "coverage-6.1.1-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:0147f7833c41927d84f5af9219d9b32f875c0689e5e74ac8ca3cb61e73a698f9"},
+    {file = "coverage-6.1.1-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:b1d0a1bce919de0dd8da5cff4e616b2d9e6ebf3bd1410ff645318c3dd615010a"},
+    {file = "coverage-6.1.1-cp39-cp39-win32.whl", hash = "sha256:a11a2c019324fc111485e79d55907e7289e53d0031275a6c8daed30690bc50c0"},
+    {file = "coverage-6.1.1-cp39-cp39-win_amd64.whl", hash = "sha256:4d8b453764b9b26b0dd2afb83086a7c3f9379134e340288d2a52f8a91592394b"},
+    {file = "coverage-6.1.1-pp36-none-any.whl", hash = "sha256:3b270c6b48d3ff5a35deb3648028ba2643ad8434b07836782b1139cf9c66313f"},
+    {file = "coverage-6.1.1-pp37-none-any.whl", hash = "sha256:ffa8fee2b1b9e60b531c4c27cf528d6b5d5da46b1730db1f4d6eee56ff282e07"},
+    {file = "coverage-6.1.1-pp38-none-any.whl", hash = "sha256:4cd919057636f63ab299ccb86ea0e78b87812400c76abab245ca385f17d19fb5"},
+    {file = "coverage-6.1.1.tar.gz", hash = "sha256:b8e4f15b672c9156c1154249a9c5746e86ac9ae9edc3799ee3afebc323d9d9e0"},
+]
+cryptography = [
+    {file = "cryptography-35.0.0-cp36-abi3-macosx_10_10_x86_64.whl", hash = "sha256:d57e0cdc1b44b6cdf8af1d01807db06886f10177469312fbde8f44ccbb284bc9"},
+    {file = "cryptography-35.0.0-cp36-abi3-macosx_11_0_arm64.whl", hash = "sha256:ced40344e811d6abba00295ced98c01aecf0c2de39481792d87af4fa58b7b4d6"},
+    {file = "cryptography-35.0.0-cp36-abi3-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:54b2605e5475944e2213258e0ab8696f4f357a31371e538ef21e8d61c843c28d"},
+    {file = "cryptography-35.0.0-cp36-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_24_aarch64.whl", hash = "sha256:7b7ceeff114c31f285528ba8b390d3e9cfa2da17b56f11d366769a807f17cbaa"},
+    {file = "cryptography-35.0.0-cp36-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2d69645f535f4b2c722cfb07a8eab916265545b3475fdb34e0be2f4ee8b0b15e"},
+    {file = "cryptography-35.0.0-cp36-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4a2d0e0acc20ede0f06ef7aa58546eee96d2592c00f450c9acb89c5879b61992"},
+    {file = "cryptography-35.0.0-cp36-abi3-manylinux_2_24_x86_64.whl", hash = "sha256:07bb7fbfb5de0980590ddfc7f13081520def06dc9ed214000ad4372fb4e3c7f6"},
+    {file = "cryptography-35.0.0-cp36-abi3-musllinux_1_1_aarch64.whl", hash = "sha256:7eba2cebca600a7806b893cb1d541a6e910afa87e97acf2021a22b32da1df52d"},
+    {file = "cryptography-35.0.0-cp36-abi3-musllinux_1_1_x86_64.whl", hash = "sha256:18d90f4711bf63e2fb21e8c8e51ed8189438e6b35a6d996201ebd98a26abbbe6"},
+    {file = "cryptography-35.0.0-cp36-abi3-win32.whl", hash = "sha256:c10c797ac89c746e488d2ee92bd4abd593615694ee17b2500578b63cad6b93a8"},
+    {file = "cryptography-35.0.0-cp36-abi3-win_amd64.whl", hash = "sha256:7075b304cd567694dc692ffc9747f3e9cb393cc4aa4fb7b9f3abd6f5c4e43588"},
+    {file = "cryptography-35.0.0-pp36-pypy36_pp73-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:a688ebcd08250eab5bb5bca318cc05a8c66de5e4171a65ca51db6bd753ff8953"},
+    {file = "cryptography-35.0.0-pp36-pypy36_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d99915d6ab265c22873f1b4d6ea5ef462ef797b4140be4c9d8b179915e0985c6"},
+    {file = "cryptography-35.0.0-pp36-pypy36_pp73-manylinux_2_24_x86_64.whl", hash = "sha256:928185a6d1ccdb816e883f56ebe92e975a262d31cc536429041921f8cb5a62fd"},
+    {file = "cryptography-35.0.0-pp37-pypy37_pp73-macosx_10_10_x86_64.whl", hash = "sha256:ebeddd119f526bcf323a89f853afb12e225902a24d29b55fe18dd6fcb2838a76"},
+    {file = "cryptography-35.0.0-pp37-pypy37_pp73-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:22a38e96118a4ce3b97509443feace1d1011d0571fae81fc3ad35f25ba3ea999"},
+    {file = "cryptography-35.0.0-pp37-pypy37_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:eb80e8a1f91e4b7ef8b33041591e6d89b2b8e122d787e87eeb2b08da71bb16ad"},
+    {file = "cryptography-35.0.0-pp37-pypy37_pp73-manylinux_2_24_x86_64.whl", hash = "sha256:abb5a361d2585bb95012a19ed9b2c8f412c5d723a9836418fab7aaa0243e67d2"},
+    {file = "cryptography-35.0.0-pp37-pypy37_pp73-win_amd64.whl", hash = "sha256:1ed82abf16df40a60942a8c211251ae72858b25b7421ce2497c2eb7a1cee817c"},
+    {file = "cryptography-35.0.0.tar.gz", hash = "sha256:9933f28f70d0517686bd7de36166dda42094eac49415459d9bdf5e7df3e0086d"},
 ]
 curlylint = [
     {file = "curlylint-0.12.2-py3-none-any.whl", hash = "sha256:98bc15609ce858387dd70a28c7ddda96e82d0f1cb8bf51b8902532ce0fc1a97e"},
     {file = "curlylint-0.12.2.tar.gz", hash = "sha256:76b557cf8d007bd92df2dae61a02e65f8aa2ff3e05c6398b1314d92692fbb0d8"},
 ]
 decorator = [
-    {file = "decorator-5.0.7-py3-none-any.whl", hash = "sha256:945d84890bb20cc4a2f4a31fc4311c0c473af65ea318617f13a7257c9a58bc98"},
-    {file = "decorator-5.0.7.tar.gz", hash = "sha256:6f201a6c4dac3d187352661f508b9364ec8091217442c9478f1f83c003a0f060"},
+    {file = "decorator-5.1.0-py3-none-any.whl", hash = "sha256:7b12e7c3c6ab203a29e157335e9122cb03de9ab7264b137594103fd4a683b374"},
+    {file = "decorator-5.1.0.tar.gz", hash = "sha256:e59913af105b9860aa2c8d3272d9de5a56a4e608db9a2f167a8480b323d529a7"},
+]
+defusedxml = [
+    {file = "defusedxml-0.7.1-py2.py3-none-any.whl", hash = "sha256:a352e7e428770286cc899e2542b6cdaedb2b4953ff269a210103ec58f6198a61"},
+    {file = "defusedxml-0.7.1.tar.gz", hash = "sha256:1bb3032db185915b62d7c6209c5a8792be6a32ab2fedacc84e01b52c51aa3e69"},
+]
+deprecated = [
+    {file = "Deprecated-1.2.13-py2.py3-none-any.whl", hash = "sha256:64756e3e14c8c5eea9795d93c524551432a0be75629f8f29e67ab8caf076c76d"},
+    {file = "Deprecated-1.2.13.tar.gz", hash = "sha256:43ac5335da90c31c24ba028af536a91d41d53f9e6901ddb021bcc572ce44e38d"},
 ]
 dj-database-url = [
     {file = "dj-database-url-0.5.0.tar.gz", hash = "sha256:4aeaeb1f573c74835b0686a2b46b85990571159ffc21aa57ecd4d1e1cb334163"},
     {file = "dj_database_url-0.5.0-py2.py3-none-any.whl", hash = "sha256:851785365761ebe4994a921b433062309eb882fedd318e1b0fcecc607ed02da9"},
 ]
 django = [
-    {file = "Django-3.2-py3-none-any.whl", hash = "sha256:0604e84c4fb698a5e53e5857b5aea945b2f19a18f25f10b8748dbdf935788927"},
-    {file = "Django-3.2.tar.gz", hash = "sha256:21f0f9643722675976004eb683c55d33c05486f94506672df3d6a141546f389d"},
+    {file = "Django-3.2.9-py3-none-any.whl", hash = "sha256:e22c9266da3eec7827737cde57694d7db801fedac938d252bf27377cec06ed1b"},
+    {file = "Django-3.2.9.tar.gz", hash = "sha256:51284300f1522ffcdb07ccbdf676a307c6678659e1284f0618e5a774127a6a08"},
+]
+django-allauth = [
+    {file = "django-allauth-0.45.0.tar.gz", hash = "sha256:6d46be0e1480316ccd45476db3aefb39db70e038d2a543112d314b76bb999a4e"},
 ]
 django-any-js = [
     {file = "django-any-js-1.1.tar.gz", hash = "sha256:2972946902ba049f73bf8bb87e0a0118f77a8c9dca89438f193598bff758422f"},
     {file = "django_any_js-1.1-py3-none-any.whl", hash = "sha256:1499934e293bbcaad29b8edaaefca87dda79eb3df1faeaaea67b80e2866ae1f8"},
 ]
 django-appconf = [
-    {file = "django-appconf-1.0.4.tar.gz", hash = "sha256:be58deb54a43d77d2e1621fe59f787681376d3cd0b8bd8e4758ef6c3a6453380"},
-    {file = "django_appconf-1.0.4-py2.py3-none-any.whl", hash = "sha256:1b1d0e1069c843ebe8ae5aa48ec52403b1440402b320c3e3a206a0907e97bb06"},
+    {file = "django-appconf-1.0.5.tar.gz", hash = "sha256:be3db0be6c81fa84742000b89a81c016d70ae66a7ccb620cdef592b1f1a6aaa4"},
+    {file = "django_appconf-1.0.5-py3-none-any.whl", hash = "sha256:ae9f864ee1958c815a965ed63b3fba4874eec13de10236ba063a788f9a17389d"},
 ]
 django-auth-ldap = [
-    {file = "django-auth-ldap-2.4.0.tar.gz", hash = "sha256:60fcbfc3141c99c3c49d3ccd7311a3992a231c319d94b6d2c143968f63676676"},
-    {file = "django_auth_ldap-2.4.0-py3-none-any.whl", hash = "sha256:2d869955da8a0c9a4448671bd9826b9f87458f6a9fc20278e84de8a81200a2be"},
+    {file = "django-auth-ldap-3.0.0.tar.gz", hash = "sha256:1f2d5c562d9ba9a5e9a64099ae9798e1a63840a11afe4d1c4a9c74121f066eaa"},
+    {file = "django_auth_ldap-3.0.0-py3-none-any.whl", hash = "sha256:19ee19034f344d9efd07ed88d3187e256ec33ae39d6a47222083b89f7d35c5f6"},
 ]
 django-bleach = [
-    {file = "django-bleach-0.6.1.tar.gz", hash = "sha256:674709c26040618aff0741ce8261fd151e5ead405bd50568c2034662d69daac3"},
-    {file = "django_bleach-0.6.1-py2.py3-none-any.whl", hash = "sha256:59de95cd98f924992313821ab7f94cd64a03aa900ca980bd3b062d8aef1a7954"},
-]
-django-bulk-update = [
-    {file = "django-bulk-update-2.2.0.tar.gz", hash = "sha256:5ab7ce8a65eac26d19143cc189c0f041d5c03b9d1b290ca240dc4f3d6aaeb337"},
-    {file = "django_bulk_update-2.2.0-py2.py3-none-any.whl", hash = "sha256:49a403392ae05ea872494d74fb3dfa3515f8df5c07cc277c3dc94724c0ee6985"},
+    {file = "django-bleach-0.9.0.tar.gz", hash = "sha256:d76534566987187da35c60d9097bc18e6d8aadc2d8743cee46b820940e45cea7"},
+    {file = "django_bleach-0.9.0-py2.py3-none-any.whl", hash = "sha256:12aeffd1b9f96d91d92bb04e9ce46dba29294c7fc1c6678d111e769d279bad3e"},
 ]
 django-cachalot = [
-    {file = "django-cachalot-2.3.5.tar.gz", hash = "sha256:02afabb6e83f5f06c87a7e6f01ebcdbc52a4156ec849da8e68b14498bc474d3e"},
-    {file = "django_cachalot-2.3.5-py3-none-any.whl", hash = "sha256:ed0782f9702ead95337692f0fae8bbb9352a106490f272d9b76e86b1da81c7e3"},
+    {file = "django-cachalot-2.4.4.tar.gz", hash = "sha256:8489a74e9a689437f81fa61705b154a4be3cac1b62b61f68215d79668c9ac037"},
+    {file = "django_cachalot-2.4.4-py3-none-any.whl", hash = "sha256:f8995a85197ac290e63a2f97c577cc9825000fe081b08d5c7d35079e7f1daa43"},
 ]
 django-cache-memoize = [
-    {file = "django-cache-memoize-0.1.8.tar.gz", hash = "sha256:f85ca71ddfe3d61d561d5a382736f83148fb75e542585e7028b65d6d3681ec85"},
-    {file = "django_cache_memoize-0.1.8-py3-none-any.whl", hash = "sha256:81b00714b50917431ce12a4544e0630a70c86fed27755a82186efc2945b8f8b3"},
+    {file = "django-cache-memoize-0.1.10.tar.gz", hash = "sha256:63e8faa245a41c0dbad843807e9f21a6e59eba8e6e50df310fdf6485a6749843"},
+    {file = "django_cache_memoize-0.1.10-py3-none-any.whl", hash = "sha256:676299313079cde9242ae84db0160e80b1d44e8dd6bc9b1f4f1247e11b30c9e0"},
 ]
 django-celery-beat = [
-    {file = "django-celery-beat-2.2.0.tar.gz", hash = "sha256:b8a13afb15e7c53fc04f4f847ac71a6d32088959aba701eb7c4a59f0c28ba543"},
-    {file = "django_celery_beat-2.2.0-py2.py3-none-any.whl", hash = "sha256:c4c72a9579f20eff4c4ccf1b58ebdca5ef940f4210065057db1754ea5f8dffdc"},
+    {file = "django-celery-beat-2.2.1.tar.gz", hash = "sha256:97ae5eb309541551bdb07bf60cc57cadacf42a74287560ced2d2c06298620234"},
+    {file = "django_celery_beat-2.2.1-py2.py3-none-any.whl", hash = "sha256:ab43049634fd18dc037927d7c2c7d5f67f95283a20ebbda55f42f8606412e66c"},
 ]
 django-celery-email = [
     {file = "django-celery-email-3.0.0.tar.gz", hash = "sha256:5546cbba80952cc3b8a0ffa4206ce90a4a996a7ffd1c385a2bdb65903ca18ece"},
     {file = "django_celery_email-3.0.0-py2.py3-none-any.whl", hash = "sha256:0f72da39cb2ea83c69440566e87f27cd72f68f247f98ce99fb29889fcf329406"},
 ]
 django-celery-results = [
-    {file = "django_celery_results-2.0.1-py2.py3-none-any.whl", hash = "sha256:a2f7d172f7f57dd972538acc6e80a5bf50c673fb4d82fe027189c8659c60dfce"},
-    {file = "django_celery_results-2.0.1.tar.gz", hash = "sha256:d625e324138e5b2ef46ffa9e89fa353c16d619420066ac8b240ef9247b293a84"},
+    {file = "django_celery_results-2.2.0-py2.py3-none-any.whl", hash = "sha256:d5f83fad9091e52cd6dbb3ca80632153ad14b6cdac4d73258e040f92717237cb"},
+    {file = "django_celery_results-2.2.0.tar.gz", hash = "sha256:cc0285090a306f97f1d4b7929ed98af0475bf6db2568976b3387de4fbe812edc"},
 ]
 django-ckeditor = [
-    {file = "django-ckeditor-6.0.0.tar.gz", hash = "sha256:29fd1a333cb9741ac2c3fd4e427a5c00115ed33a2389716a09af7656022dcdde"},
-    {file = "django_ckeditor-6.0.0-py2.py3-none-any.whl", hash = "sha256:cc2d377f1bdcd4ca1540caeebe85f7e2cd006198d57328ef6c718d3eaa5a0846"},
+    {file = "django-ckeditor-6.1.0.tar.gz", hash = "sha256:f0d108f67a81a04e26d8de11255fe314f51026eaf8eb0534a807512ae3c21620"},
+    {file = "django_ckeditor-6.1.0-py2.py3-none-any.whl", hash = "sha256:346b26b9d60dc8a88524d0eaaf406f4e91a4b3c22d208ae87aa032bf500b251c"},
 ]
 django-cleanup = [
-    {file = "django-cleanup-5.1.0.tar.gz", hash = "sha256:8976aec12a22913afb3d1fcb541b1aedde2f5ec243e4260c5ff78bb6aa75a089"},
-    {file = "django_cleanup-5.1.0-py2.py3-none-any.whl", hash = "sha256:71e098c7d9ac3f3da40b95cff9c0bc51218d40ef419261232f46ba3141c50acc"},
+    {file = "django-cleanup-5.2.0.tar.gz", hash = "sha256:909d10ff574f5ce1a40fa63bd5c94c9ed866fd7ae770994c46cdf66c3db3e846"},
+    {file = "django_cleanup-5.2.0-py2.py3-none-any.whl", hash = "sha256:193cf69de54b9fc0a0f4547edbb3a63bbe01728cb029f9f4b7912098cc1bced7"},
 ]
 django-colorfield = [
-    {file = "django-colorfield-0.4.1.tar.gz", hash = "sha256:63a542c417b72d0dac898a0f61a2a00aed3c9aabc2f5057c926efccf421f7887"},
-    {file = "django_colorfield-0.4.1-py3-none-any.whl", hash = "sha256:e38f8b9dabbab48a6dab3d1eb5bd802decb92970d56a28128c9a70cdbf383e30"},
+    {file = "django-colorfield-0.4.5.tar.gz", hash = "sha256:66d7cb628d05c0eb09e25b0923e36bf6fbd3e339c568a199e66b34463119ca13"},
+    {file = "django_colorfield-0.4.5-py3-none-any.whl", hash = "sha256:579128d008b2d15e4df64f102b1ad58432de1b6f1017221c316d630e5c62960a"},
 ]
 django-dbbackup = [
     {file = "django-dbbackup-3.3.0.tar.gz", hash = "sha256:bb109735cae98b64ad084e5b461b7aca2d7b39992f10c9ed9435e3ebb6fb76c8"},
 ]
 django-debug-toolbar = [
-    {file = "django-debug-toolbar-3.2.1.tar.gz", hash = "sha256:a5ff2a54f24bf88286f9872836081078f4baa843dc3735ee88524e89f8821e33"},
-    {file = "django_debug_toolbar-3.2.1-py3-none-any.whl", hash = "sha256:e759e63e3fe2d3110e0e519639c166816368701eab4a47fed75d7de7018467b9"},
+    {file = "django-debug-toolbar-3.2.2.tar.gz", hash = "sha256:8c5b13795d4040008ee69ba82dcdd259c49db346cf7d0de6e561a49d191f0860"},
+    {file = "django_debug_toolbar-3.2.2-py3-none-any.whl", hash = "sha256:d7bab7573fab35b0fd029163371b7182f5826c13da69734beb675c761d06a4d3"},
 ]
 django-dynamic-preferences = [
-    {file = "django-dynamic-preferences-1.10.1.tar.gz", hash = "sha256:e4b2bb7b2563c5064ba56dd76441c77e06b850ff1466a386a1cd308909a6c7de"},
-    {file = "django_dynamic_preferences-1.10.1-py2.py3-none-any.whl", hash = "sha256:9419fa925fd2cbb665269ae72059eb3058bf080913d853419b827e4e7a141902"},
+    {file = "django-dynamic-preferences-1.11.0.tar.gz", hash = "sha256:f214c938b5872a17647e2b2ccfd9ad00a90a3c6c4aa83fa65d3c5c446e7a66c7"},
+    {file = "django_dynamic_preferences-1.11.0-py2.py3-none-any.whl", hash = "sha256:31aecebcbfcfb970b78cfa3e5f8cc9f77638efe8e7c90f205a48b01c45ee5002"},
 ]
 django-extensions = [
-    {file = "django-extensions-3.1.2.tar.gz", hash = "sha256:081828e985485662f62a22340c1506e37989d14b927652079a5b7cd84a82368b"},
-    {file = "django_extensions-3.1.2-py3-none-any.whl", hash = "sha256:17f85f4dcdd5eea09b8c4f0bad8f0370bf2db6d03e61b431fa7103fee29888de"},
+    {file = "django-extensions-3.1.3.tar.gz", hash = "sha256:5f0fea7bf131ca303090352577a9e7f8bfbf5489bd9d9c8aea9401db28db34a0"},
+    {file = "django_extensions-3.1.3-py3-none-any.whl", hash = "sha256:50de8977794a66a91575dd40f87d5053608f679561731845edbd325ceeb387e3"},
 ]
 django-favicon-plus-reloaded = [
-    {file = "django-favicon-plus-reloaded-1.0.4.tar.gz", hash = "sha256:90c761c636a338e6e9fb1d086649d82095085f92cff816c9cf074607f28c85a5"},
-    {file = "django_favicon_plus_reloaded-1.0.4-py3-none-any.whl", hash = "sha256:26e4316d41328a61ced52c7fc0ead795f0eb194d6a30311c34a9833c6fe30a7c"},
+    {file = "django-favicon-plus-reloaded-1.1.3.tar.gz", hash = "sha256:36c2a1cefc201df8bd132492e2440ccdc3d9ceb8e421b2ca181a4704ebacd190"},
+    {file = "django_favicon_plus_reloaded-1.1.3-py3-none-any.whl", hash = "sha256:a60b438360e82bf7075b856ff6a80bae20c825373a58deac627810e478c42be3"},
 ]
 django-filter = [
     {file = "django-filter-2.4.0.tar.gz", hash = "sha256:84e9d5bb93f237e451db814ed422a3a625751cbc9968b484ecc74964a8696b06"},
     {file = "django_filter-2.4.0-py3-none-any.whl", hash = "sha256:e00d32cebdb3d54273c48f4f878f898dced8d5dfaad009438fe61ebdf535ace1"},
 ]
 django-formtools = [
-    {file = "django-formtools-2.2.tar.gz", hash = "sha256:c5272c03c1cd51b2375abf7397a199a3148a9fbbf2f100e186467a84025d13b2"},
-    {file = "django_formtools-2.2-py2.py3-none-any.whl", hash = "sha256:304fa777b8ef9e0693ce7833f885cb89ba46b0e46fc23b01176900a93f46742f"},
+    {file = "django-formtools-2.3.tar.gz", hash = "sha256:9663b6eca64777b68d6d4142efad8597fe9a685924673b25aa8a1dcff4db00c3"},
+    {file = "django_formtools-2.3-py3-none-any.whl", hash = "sha256:4699937e19ee041d803943714fe0c1c7ad4cab802600eb64bbf4cdd0a1bfe7d9"},
 ]
 django-guardian = [
-    {file = "django-guardian-2.3.0.tar.gz", hash = "sha256:ed2de26e4defb800919c5749fb1bbe370d72829fbd72895b6cf4f7f1a7607e1b"},
-    {file = "django_guardian-2.3.0-py3-none-any.whl", hash = "sha256:0e70706c6cda88ddaf8849bddb525b8df49de05ba0798d4b3506049f0d95cbc8"},
-]
-django-hattori = [
-    {file = "django-hattori-0.2.1.tar.gz", hash = "sha256:6953d40881317252f19f62c4e7fe8058924b852c7498bc42beb7bc4d268c252c"},
-    {file = "django_hattori-0.2.1-py2.py3-none-any.whl", hash = "sha256:e529ed7af8fc34a0169c797c477672b687a205a56f3f5206f90c260acb83b7ac"},
+    {file = "django-guardian-2.4.0.tar.gz", hash = "sha256:c58a68ae76922d33e6bdc0e69af1892097838de56e93e78a8361090bcd9f89a0"},
+    {file = "django_guardian-2.4.0-py3-none-any.whl", hash = "sha256:440ca61358427e575323648b25f8384739e54c38b3d655c81d75e0cd0d61b697"},
 ]
 django-haystack = [
-    {file = "django-haystack-3.0.tar.gz", hash = "sha256:d490f920afa85471dd1fa5000bc8eff4b704daacbe09aee1a64e75cbc426f3be"},
+    {file = "django-haystack-3.1.1.tar.gz", hash = "sha256:6d05756b95d7d5ec1dbd4668eb999ced1504b47f588e2e54be53b1404c516a82"},
+    {file = "django_haystack-3.1.1-py3-none-any.whl", hash = "sha256:970ebc6362f3f84861209154ec34b57f3e87d2d7922f948df80177f0e2e3ced7"},
 ]
 django-health-check = [
-    {file = "django-health-check-3.16.3.tar.gz", hash = "sha256:a6aa6ea423eae4fd0665f6372b826af1ed20dfc3e88cf52789d0b49cfb64969c"},
-    {file = "django_health_check-3.16.3-py2.py3-none-any.whl", hash = "sha256:d0628ffc11aee7e62e73b58ff39179ea2a9ca5abfbc92cb345ceca268593dd71"},
+    {file = "django-health-check-3.16.4.tar.gz", hash = "sha256:334bcbbb9273a6dbd9c928e78474306e623dfb38cc442281cb9fd230a20a7fdb"},
+    {file = "django_health_check-3.16.4-py2.py3-none-any.whl", hash = "sha256:86a8869d67e72394a1dd73e37819a7d2cfd915588b96927fda611d7451fd4735"},
 ]
 django-impersonate = [
     {file = "django-impersonate-1.7.3.tar.gz", hash = "sha256:282003957577c7143fe31e5861f8fffdf6fe0c25557aedb28fcf8b11474eaa23"},
 ]
 django-ipware = [
-    {file = "django-ipware-3.0.2.tar.gz", hash = "sha256:c7df8e1410a8e5d6b1fbae58728402ea59950f043c3582e033e866f0f0cf5e94"},
+    {file = "django-ipware-4.0.0.tar.gz", hash = "sha256:1294f916f3b3475e40e1b0ec1bd320aa2397978eae672721c81cbc2ed517e9ee"},
+    {file = "django_ipware-4.0.0-py2.py3-none-any.whl", hash = "sha256:116bd0d7940f09bf7ffd465943992e23d87e772a9d6c0d3a57b74040589a383b"},
 ]
 django-js-asset = [
     {file = "django-js-asset-1.2.2.tar.gz", hash = "sha256:c163ae80d2e0b22d8fb598047cd0dcef31f81830e127cfecae278ad574167260"},
@@ -2836,34 +4179,35 @@ django-jsonstore = [
     {file = "django_jsonstore-0.5.0-py2-none-any.whl", hash = "sha256:9630c1fb43ae9f8e32733c5cf7d4c3775ba6f08532f517c64025053352d72844"},
 ]
 django-maintenance-mode = [
-    {file = "django-maintenance-mode-0.16.0.tar.gz", hash = "sha256:57595795062156d5f3f712c885acc18b77a303425bf78b5de80e7fd47d9ab433"},
-    {file = "django_maintenance_mode-0.16.0-py3-none-any.whl", hash = "sha256:88287573b4e95285052f664d4f08e15ac4c350c1a6c77bc743ca3fc6e1f6410c"},
+    {file = "django-maintenance-mode-0.16.1.tar.gz", hash = "sha256:da1f77f431ab5c55913459adb3c259e091f783ffc72de701690826aaaccce4ca"},
+    {file = "django_maintenance_mode-0.16.1-py3-none-any.whl", hash = "sha256:1bfac0b34429a9f6dbb0db169fb753f49f24ca155ae148d7526ff395303c158d"},
 ]
 django-material = [
-    {file = "django-material-1.7.6.tar.gz", hash = "sha256:5488e8fe24069cc6682801692ad05293a4b60a637a87a31e0ebd9f3319cd371d"},
-    {file = "django_material-1.7.6-py2.py3-none-any.whl", hash = "sha256:b5496505da7dd92f23ca694bc411c6bf0ff584fc30f4239d890ab29f9260160c"},
+    {file = "django-material-1.9.0.tar.gz", hash = "sha256:5a7144d1029b4a2bfee2e5d0d8d00f30742dd7e3f868b3787d8cd61e54f26437"},
+    {file = "django_material-1.9.0-py2.py3-none-any.whl", hash = "sha256:816513170771bcb2540b5ce314fbef1a906906220587a9cb9521e61092a6f610"},
 ]
 django-menu-generator-ng = [
     {file = "django-menu-generator-ng-1.2.3.tar.gz", hash = "sha256:0c21a094b094add909655728b6b2d4a8baa5a2047da8f649be52589bea0e3ba2"},
 ]
-django-middleware-global-request = [
-    {file = "django-middleware-global-request-0.1.2.tar.gz", hash = "sha256:f6490759bc9f7dbde4001709554e29ca715daf847f2222914b4e47117dca9313"},
-]
 django-model-utils = [
-    {file = "django-model-utils-4.1.1.tar.gz", hash = "sha256:eb5dd05ef7d7ce6bc79cae54ea7c4a221f6f81e2aad7722933aee66489e7264b"},
-    {file = "django_model_utils-4.1.1-py3-none-any.whl", hash = "sha256:ef7c440024e797796a3811432abdd2be8b5225ae64ef346f8bfc6de7d8e5d73c"},
+    {file = "django-model-utils-4.2.0.tar.gz", hash = "sha256:e7a95e102f9c9653427eadab980d5d59e1dea972913b9c9e01ac37f86bba0ddf"},
+    {file = "django_model_utils-4.2.0-py3-none-any.whl", hash = "sha256:a768a25c80514e0ad4e4a6f9c02c44498985f36c5dfdea47b5b1e8cf994beba6"},
+]
+django-oauth-toolkit = [
+    {file = "django-oauth-toolkit-1.5.0.tar.gz", hash = "sha256:650e5ef2244d1d8db8f507137e0d1e8b8aad1f4086a4a610526e8851f9a38308"},
+    {file = "django_oauth_toolkit-1.5.0-py3-none-any.whl", hash = "sha256:b2e346a7c1e222774bfb370f21b556b92b408395b4c23914e2d1b241b2e5376a"},
 ]
 django-otp = [
-    {file = "django-otp-1.0.3.tar.gz", hash = "sha256:381a15e65293b8b06d47b7d6b306e0b7af2e104137ac92f6c566d3b9b90b6244"},
-    {file = "django_otp-1.0.3-py3-none-any.whl", hash = "sha256:f4ab096b424c33ffe69453620356e1b7517f30dfb9ba13bfeaa1d1f20faddc13"},
+    {file = "django-otp-1.1.1.tar.gz", hash = "sha256:4c90cdaed683d736b0efafc034a3c6b410e1be2a53c24da287165b1f371d8776"},
+    {file = "django_otp-1.1.1-py3-none-any.whl", hash = "sha256:0c03a471db9e876f3671314bc9a65bd56a5c3c108ee0562c473701310bba4a77"},
 ]
 django-otp-yubikey = [
     {file = "django-otp-yubikey-1.0.0.post1.tar.gz", hash = "sha256:1da060257611d06e681848b7923fd788d878a79e8c358a373374deab13a085af"},
     {file = "django_otp_yubikey-1.0.0.post1-py2.py3-none-any.whl", hash = "sha256:613c96be211c1267400a5a78ae63f212c722f82dffb9daef3c8b1df370abb9be"},
 ]
 django-phonenumber-field = [
-    {file = "django-phonenumber-field-5.1.0.tar.gz", hash = "sha256:9eda963ac15b363393f677cc084efd45c3bd97bb5a0cdb4a06409ac99e05dd4b"},
-    {file = "django_phonenumber_field-5.1.0-py3-none-any.whl", hash = "sha256:48724ba235ee8248a474204faa0934c5baf9536f429859d05cb131fbd6b1c695"},
+    {file = "django-phonenumber-field-5.2.0.tar.gz", hash = "sha256:52b2e5970133ec5ab701218b802f7ab237229854dc95fd239b7e9e77dc43731d"},
+    {file = "django_phonenumber_field-5.2.0-py3-none-any.whl", hash = "sha256:5547fb2b2cc690a306ba77a5038419afc8fa8298a486fb7895008e9067cc7e75"},
 ]
 django-polymorphic = [
     {file = "django-polymorphic-3.0.0.tar.gz", hash = "sha256:9d886f19f031d26bb1391c055ed9be06fb226a04a4cec1842b372c58873b3caa"},
@@ -2873,70 +4217,72 @@ django-prometheus = [
     {file = "django-prometheus-2.1.0.tar.gz", hash = "sha256:dd3f8da1399140fbef5c00d1526a23d1ade286b144281c325f8e409a781643f2"},
     {file = "django_prometheus-2.1.0-py2.py3-none-any.whl", hash = "sha256:c338d6efde1ca336e90c540b5e87afe9287d7bcc82d651a778f302b0be17a933"},
 ]
-django-pwa = [
-    {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-redis = [
-    {file = "django-redis-4.12.1.tar.gz", hash = "sha256:306589c7021e6468b2656edc89f62b8ba67e8d5a1c8877e2688042263daa7a63"},
-    {file = "django_redis-4.12.1-py3-none-any.whl", hash = "sha256:1133b26b75baa3664164c3f44b9d5d133d1b8de45d94d79f38d1adc5b1d502e5"},
+    {file = "django-redis-5.0.0.tar.gz", hash = "sha256:048f665bbe27f8ff2edebae6aa9c534ab137f1e8fa7234147ef470df3f3aa9b8"},
+    {file = "django_redis-5.0.0-py3-none-any.whl", hash = "sha256:97739ca9de3f964c51412d1d7d8aecdfd86737bb197fce6e1ff12620c63c97ee"},
 ]
 django-render-block = [
     {file = "django-render-block-0.8.1.tar.gz", hash = "sha256:edbc5d444cc50f3eb3387cf17f6f1014bf19d6018f680861cdeae9e0306003fa"},
     {file = "django_render_block-0.8.1-py3-none-any.whl", hash = "sha256:903969efd0949f750c5fe71affe6e6b1ea66d03005c102a67fda36d5b9f4e1e1"},
 ]
 django-reversion = [
-    {file = "django-reversion-3.0.9.tar.gz", hash = "sha256:a5af55f086a3f9c38be2f049c251e06005b9ed48ba7a109473736b1fc95a066f"},
-    {file = "django_reversion-3.0.9-py3-none-any.whl", hash = "sha256:1b57127a136b969f4b843a915c72af271febe7f336469db6c27121f8adcad35c"},
+    {file = "django-reversion-4.0.1.tar.gz", hash = "sha256:6991f16e5d3a972912db3d56e3a714d10b07becd566ab87f85f2e9b671981339"},
+    {file = "django_reversion-4.0.1-py3-none-any.whl", hash = "sha256:2e40ed41e08cdd83a05dc70a1974feface52a61ba7d289727117163052081ae6"},
 ]
 django-sass-processor = [
-    {file = "django-sass-processor-1.0.1.tar.gz", hash = "sha256:dcaad47c591a2d52689c1bd209259e922e902d886293f0d5c9e0d1a4eb85eda2"},
-    {file = "django_sass_processor-1.0.1-py3-none-any.whl", hash = "sha256:1f043180c47754018e803a77da003377f5ea6558de57cd6946eb27a32e9c16a2"},
+    {file = "django-sass-processor-1.0.0.tar.gz", hash = "sha256:cb90efee38cd7b0fe727c78d8993ad7804de33f40328200dfc1a481307ef0466"},
 ]
 django-select2 = [
-    {file = "django-select2-7.7.1.tar.gz", hash = "sha256:dd091342e99436818b3fa98783ae6c24fb2a0cbc37ebd3faa0aef68422b6e416"},
-    {file = "django_select2-7.7.1-py2.py3-none-any.whl", hash = "sha256:8c54984bb931d842eab6a46d1b427c6883e5f5347529cda27dcd942fb37d87b9"},
-]
-django-settings-context-processor = [
-    {file = "django-settings-context-processor-0.2.tar.gz", hash = "sha256:d37c853d69a3069f5abbf94c7f4f6fc0fac38bbd0524190cd5a250ba800e496a"},
+    {file = "django-select2-7.9.0.tar.gz", hash = "sha256:4494cf5fa13333ddbd34e4dbeca4b44a22ed6367342cccd24f6f122f6ef4a20b"},
+    {file = "django_select2-7.9.0-py2.py3-none-any.whl", hash = "sha256:40d42643b459f79c31598fa5645f5f7e8edda8350a78761ab6c06105ca588872"},
 ]
 django-storages = [
-    {file = "django-storages-1.11.1.tar.gz", hash = "sha256:c823dbf56c9e35b0999a13d7e05062b837bae36c518a40255d522fbe3750fbb4"},
-    {file = "django_storages-1.11.1-py3-none-any.whl", hash = "sha256:f28765826d507a0309cfaa849bd084894bc71d81bf0d09479168d44785396f80"},
+    {file = "django-storages-1.12.3.tar.gz", hash = "sha256:a475edb2f0f04c4f7e548919a751ecd50117270833956ed5bd585c0575d2a5e7"},
+    {file = "django_storages-1.12.3-py3-none-any.whl", hash = "sha256:204a99f218b747c46edbfeeb1310d357f83f90fa6a6024d8d0a3f422570cee84"},
 ]
 django-stubs = [
-    {file = "django-stubs-1.8.0.tar.gz", hash = "sha256:717967d7fee0a6af0746724a0be80d72831a982a40fa8f245a6a46f4cafd157b"},
-    {file = "django_stubs-1.8.0-py3-none-any.whl", hash = "sha256:bde9e44e3c4574c2454e74a3e607cc3bc23b0441bb7d1312cd677d5e30984b74"},
+    {file = "django-stubs-1.9.0.tar.gz", hash = "sha256:664843091636a917faf5256d028476559dc360fdef9050b6df87ab61b21607bf"},
+    {file = "django_stubs-1.9.0-py3-none-any.whl", hash = "sha256:59c9f81af64d214b1954eaf90f037778c8d2b9c2de946a3cda177fefcf588fbd"},
 ]
 django-stubs-ext = [
-    {file = "django-stubs-ext-0.2.0.tar.gz", hash = "sha256:c14f297835a42c1122421ec7e2d06579996b29d33b8016002762afa5d78863af"},
-    {file = "django_stubs_ext-0.2.0-py3-none-any.whl", hash = "sha256:bd4a1e36ef2ba0ef15801933c85c68e59b383302c873795c6ecfc25950c7ecdb"},
+    {file = "django-stubs-ext-0.3.1.tar.gz", hash = "sha256:783c198d7e39a41be0b90fd843fa2770243a642922af679be4b19e03b82c8c28"},
+    {file = "django_stubs_ext-0.3.1-py3-none-any.whl", hash = "sha256:a51a3e9e844d4e1cacaaedbb33bf3def78a3956eed5d9575a640bd97ccd99cec"},
 ]
 django-tables2 = [
-    {file = "django-tables2-2.3.4.tar.gz", hash = "sha256:50ccadbd13740a996d8a4d4f144ef80134745cd0b5ec278061537e341f5ef7a2"},
-    {file = "django_tables2-2.3.4-py2.py3-none-any.whl", hash = "sha256:af5f70a9845fd8690c6b44120b065e55b9771cae99c1dcd0eb4f1cfa3f0a71e4"},
+    {file = "django-tables2-2.4.1.tar.gz", hash = "sha256:6c72dd208358539e789e4c0efd7d151e43283a4aa4093a35f44c43489e7ddeaa"},
+    {file = "django_tables2-2.4.1-py2.py3-none-any.whl", hash = "sha256:50762bf3d7c61a4eb70e763c3e278650d7266bb78d0497fc8fafcf4e507c9a64"},
 ]
 django-templated-email = [
-    {file = "django-templated-email-2.3.0.tar.gz", hash = "sha256:536c4e5ae099eabfb9aab36087d4d7799948c654e73da55a744213d086d5bb33"},
+    {file = "django-templated-email-3.0.0.tar.gz", hash = "sha256:49d61840ec551e640adaf341146e94d6f9058ae01df964480850bf988046e5eb"},
+    {file = "django_templated_email-3.0.0-py3-none-any.whl", hash = "sha256:bf1b68ffe6c8794c0c50e2ce20e3a166c6d511b3879abbd3cf059a3fc2fe2e60"},
 ]
 django-timezone-field = [
-    {file = "django-timezone-field-4.1.2.tar.gz", hash = "sha256:cffac62452d060e365938aa9c9f7b72d70d8b26b9c60243bce227b35abd1b9df"},
-    {file = "django_timezone_field-4.1.2-py3-none-any.whl", hash = "sha256:897c06e40b619cf5731a30d6c156886a7c64cba3a90364832148da7ef32ccf36"},
+    {file = "django-timezone-field-4.2.1.tar.gz", hash = "sha256:97780cde658daa5094ae515bb55ca97c1352928ab554041207ad515dee3fe971"},
+    {file = "django_timezone_field-4.2.1-py3-none-any.whl", hash = "sha256:6dc782e31036a58da35b553bd00c70f112d794700025270d8a6a4c1d2e5b26c6"},
+]
+django-titofisto = [
+    {file = "django-titofisto-0.1.2.post1.tar.gz", hash = "sha256:e3b0783142d075aadda1c041061f84affdbe767ffaeebd0f615359723339c208"},
+    {file = "django_titofisto-0.1.2.post1-py3-none-any.whl", hash = "sha256:abebb5db39562bde9999ffd4afe67d43eefae37c8a7ec7abbe611420e6f8aca0"},
 ]
 django-two-factor-auth = [
     {file = "django-two-factor-auth-1.13.1.tar.gz", hash = "sha256:a20e03d256fd9fd668988545f052cedcc47e5a981888562e5e27d0bb83deae89"},
     {file = "django_two_factor_auth-1.13.1-py2.py3-none-any.whl", hash = "sha256:d270d4288731233621a9462a89a8dfed2dcb86fa354125c816a89772d55f9e29"},
 ]
 django-uwsgi-ng = [
-    {file = "django-uwsgi-ng-1.1.1.tar.gz", hash = "sha256:777023fd291c5408f18e2ac4922faf25f161075699e11bf40f86dd90c9b9f1d4"},
+    {file = "django-uwsgi-ng-1.1.2.tar.gz", hash = "sha256:1df2ffa642f7a831bd8d7f7e459f7b0821113b37174ccf4bf977e4467d45d9b3"},
+    {file = "django_uwsgi_ng-1.1.2-py3-none-any.whl", hash = "sha256:8b1a489a1ed9e56da0efadfa86ec306b532e5cd953fe34b234aaefc26898c649"},
 ]
 django-widget-tweaks = [
-    {file = "django-widget-tweaks-1.4.8.tar.gz", hash = "sha256:9f91ca4217199b7671971d3c1f323a2bec71a0c27dec6260b3c006fa541bc489"},
-    {file = "django_widget_tweaks-1.4.8-py2.py3-none-any.whl", hash = "sha256:f80bff4a8a59b278bb277a405a76a8b9a884e4bae7a6c70e78a39c626cd1c836"},
+    {file = "django-widget-tweaks-1.4.9.tar.gz", hash = "sha256:19bcb66a4a9e68493ced04e7124882d753c5be517ed001556f9e35a40147f545"},
+    {file = "django_widget_tweaks-1.4.9-py2.py3-none-any.whl", hash = "sha256:d6c64fbf92cd2df9031f597c1374982233c05a1190d295c39d1c57ce007569c7"},
 ]
 django-yarnpkg = [
     {file = "django-yarnpkg-6.0.1.tar.gz", hash = "sha256:aa059347b246c6f242401581d2c129bdcb45aa726be59fe2f288762a9843348a"},
 ]
+djangorestframework = [
+    {file = "djangorestframework-3.12.4-py3-none-any.whl", hash = "sha256:6d1d59f623a5ad0509fe0d6bfe93cbdfe17b8116ebc8eda86d45f6e16e819aaf"},
+    {file = "djangorestframework-3.12.4.tar.gz", hash = "sha256:f747949a8ddac876e879190df194b925c177cdeb725a099db1460872f7c0a7f2"},
+]
 docutils = [
     {file = "docutils-0.16-py2.py3-none-any.whl", hash = "sha256:0c5b78adfbf7762415433f5515cd5c9e762339e23369dbe8000d84a4bf4ab3af"},
     {file = "docutils-0.16.tar.gz", hash = "sha256:c2de3a60e9e7d07be26b7f2b00ca0309c207e06c100f9cc2a94931fc75a478fc"},
@@ -2946,31 +4292,27 @@ dparse = [
     {file = "dparse-0.5.1.tar.gz", hash = "sha256:a1b5f169102e1c894f9a7d5ccf6f9402a836a5d24be80a986c7ce9eaed78f367"},
 ]
 dynaconf = [
-    {file = "dynaconf-3.1.4-py2.py3-none-any.whl", hash = "sha256:e6f383b84150b70fc439c8b2757581a38a58d07962aa14517292dcce1a77e160"},
-    {file = "dynaconf-3.1.4.tar.gz", hash = "sha256:b2f472d83052f809c5925565b8a2ba76a103d5dc1dbb9748b693ed67212781b9"},
-]
-faker = [
-    {file = "Faker-8.1.0-py3-none-any.whl", hash = "sha256:44eb060fad3015690ff3fec6564d7171be393021e820ad1851d96cb968fbfcd4"},
-    {file = "Faker-8.1.0.tar.gz", hash = "sha256:26c7c3df8d46f1db595a34962f8967021dd90bbd38cc6e27461a3fb16cd413ae"},
+    {file = "dynaconf-3.1.7-py2.py3-none-any.whl", hash = "sha256:f52fe5db7622da56a552275e8f64e4df46e3b4ae11158831b042e8ba2f6d1c96"},
+    {file = "dynaconf-3.1.7.tar.gz", hash = "sha256:e9d80b46ba4d9372f2f40c812594c963f74178140c0b596e57f2881001fc4d35"},
 ]
 flake8 = [
-    {file = "flake8-3.9.1-py2.py3-none-any.whl", hash = "sha256:3b9f848952dddccf635be78098ca75010f073bfe14d2c6bda867154bea728d2a"},
-    {file = "flake8-3.9.1.tar.gz", hash = "sha256:1aa8990be1e689d96c745c5682b687ea49f2e05a443aff1f8251092b0014e378"},
+    {file = "flake8-3.9.2-py2.py3-none-any.whl", hash = "sha256:bf8fd333346d844f616e8d47905ef3a3384edae6b4e9beb0c5101e25e3110907"},
+    {file = "flake8-3.9.2.tar.gz", hash = "sha256:07528381786f2a6237b061f6e96610a4167b226cb926e2aa2b6b1d78057c576b"},
 ]
 flake8-bandit = [
     {file = "flake8_bandit-2.1.2.tar.gz", hash = "sha256:687fc8da2e4a239b206af2e54a90093572a60d0954f3054e23690739b0b0de3b"},
 ]
 flake8-black = [
-    {file = "flake8-black-0.2.1.tar.gz", hash = "sha256:f26651bc10db786c03f4093414f7c9ea982ed8a244cec323c984feeffdf4c118"},
-    {file = "flake8_black-0.2.1-py3-none-any.whl", hash = "sha256:941514149cb8b489cb17a4bb1cf18d84375db3b34381bb018de83509437931a0"},
+    {file = "flake8-black-0.2.3.tar.gz", hash = "sha256:c199844bc1b559d91195ebe8620216f21ed67f2cc1ff6884294c91a0d2492684"},
+    {file = "flake8_black-0.2.3-py3-none-any.whl", hash = "sha256:cc080ba5b3773b69ba102b6617a00cc4ecbad8914109690cfda4d565ea435d96"},
 ]
 flake8-builtins = [
     {file = "flake8-builtins-1.5.3.tar.gz", hash = "sha256:09998853b2405e98e61d2ff3027c47033adbdc17f9fe44ca58443d876eb00f3b"},
     {file = "flake8_builtins-1.5.3-py2.py3-none-any.whl", hash = "sha256:7706babee43879320376861897e5d1468e396a40b8918ed7bccf70e5f90b8687"},
 ]
 flake8-django = [
-    {file = "flake8-django-1.1.1.tar.gz", hash = "sha256:fb4e8f669d3cf44297bb6e1c5d0a358ab0aba373cd4c69268cf2798de6bcbd9b"},
-    {file = "flake8_django-1.1.1-py3-none-any.whl", hash = "sha256:c71da0e61b6119dae91cbffdbdb00f1d6ebe3f5d0c43f5bf136929997ab0b72d"},
+    {file = "flake8-django-1.1.2.tar.gz", hash = "sha256:b4314abb5bacda450d2eae564a0604447111b1b98188e46bca41682ad2ab59d6"},
+    {file = "flake8_django-1.1.2-py3-none-any.whl", hash = "sha256:f8bfdbe8352c2c5f3788c2a2f6652dd2604af24af07a5aa112206d63ae228fdc"},
 ]
 flake8-docstrings = [
     {file = "flake8-docstrings-1.6.0.tar.gz", hash = "sha256:9fe7c6a306064af8e62a055c2f61e9eb1da55f84bb39caef2b84ce53708ac34b"},
@@ -2981,8 +4323,8 @@ flake8-fixme = [
     {file = "flake8_fixme-1.1.1-py2.py3-none-any.whl", hash = "sha256:226a6f2ef916730899f29ac140bed5d4a17e5aba79f00a0e3ae1eff1997cb1ac"},
 ]
 flake8-isort = [
-    {file = "flake8-isort-4.0.0.tar.gz", hash = "sha256:2b91300f4f1926b396c2c90185844eb1a3d5ec39ea6138832d119da0a208f4d9"},
-    {file = "flake8_isort-4.0.0-py2.py3-none-any.whl", hash = "sha256:729cd6ef9ba3659512dee337687c05d79c78e1215fdf921ed67e5fe46cce2f3c"},
+    {file = "flake8-isort-4.1.1.tar.gz", hash = "sha256:d814304ab70e6e58859bc5c3e221e2e6e71c958e7005239202fee19c24f82717"},
+    {file = "flake8_isort-4.1.1-py3-none-any.whl", hash = "sha256:c4e8b6dcb7be9b71a02e6e5d4196cefcef0f3447be51e82730fb336fff164949"},
 ]
 flake8-mypy = [
     {file = "flake8-mypy-17.8.0.tar.gz", hash = "sha256:47120db63aff631ee1f84bac6fe8e64731dc66da3efc1c51f85e15ade4a3ba18"},
@@ -2993,177 +4335,184 @@ flake8-polyfill = [
     {file = "flake8_polyfill-1.0.2-py2.py3-none-any.whl", hash = "sha256:12be6a34ee3ab795b19ca73505e7b55826d5f6ad7230d31b18e106400169b9e9"},
 ]
 flake8-rst-docstrings = [
-    {file = "flake8-rst-docstrings-0.1.2.tar.gz", hash = "sha256:7d34d2175a0cd92aba0872ade74268b2f2c12582c7267d4a0e6ef1c32a828ce3"},
-    {file = "flake8_rst_docstrings-0.1.2-py3-none-any.whl", hash = "sha256:73b5db2fd9d4d7c7e6b7767931730c4570e5a89f34a1501b837afb3b6d31537a"},
+    {file = "flake8-rst-docstrings-0.2.3.tar.gz", hash = "sha256:3045794e1c8467fba33aaea5c246b8369efc9c44ef8b0b20199bb6df7a4bd47b"},
+    {file = "flake8_rst_docstrings-0.2.3-py3-none-any.whl", hash = "sha256:565bbb391d7e4d0042924102221e9857ad72929cdd305b26501736ec22c1451a"},
 ]
 freezegun = [
     {file = "freezegun-1.1.0-py2.py3-none-any.whl", hash = "sha256:2ae695f7eb96c62529f03a038461afe3c692db3465e215355e1bb4b0ab408712"},
     {file = "freezegun-1.1.0.tar.gz", hash = "sha256:177f9dd59861d871e27a484c3332f35a6e3f5d14626f2bf91be37891f18927f3"},
 ]
 gitdb = [
-    {file = "gitdb-4.0.7-py3-none-any.whl", hash = "sha256:6c4cc71933456991da20917998acbe6cf4fb41eeaab7d6d67fbc05ecd4c865b0"},
-    {file = "gitdb-4.0.7.tar.gz", hash = "sha256:96bf5c08b157a666fec41129e6d327235284cca4c81e92109260f353ba138005"},
+    {file = "gitdb-4.0.9-py3-none-any.whl", hash = "sha256:8033ad4e853066ba6ca92050b9df2f89301b8fc8bf7e9324d412a63f8bf1a8fd"},
+    {file = "gitdb-4.0.9.tar.gz", hash = "sha256:bac2fd45c0a1c9cf619e63a90d62bdc63892ef92387424b855792a6cabe789aa"},
 ]
 gitpython = [
-    {file = "GitPython-3.1.14-py3-none-any.whl", hash = "sha256:3283ae2fba31c913d857e12e5ba5f9a7772bbc064ae2bb09efafa71b0dd4939b"},
-    {file = "GitPython-3.1.14.tar.gz", hash = "sha256:be27633e7509e58391f10207cd32b2a6cf5b908f92d9cd30da2e514e1137af61"},
+    {file = "GitPython-3.1.24-py3-none-any.whl", hash = "sha256:dc0a7f2f697657acc8d7f89033e8b1ea94dd90356b2983bca89dc8d2ab3cc647"},
+    {file = "GitPython-3.1.24.tar.gz", hash = "sha256:df83fdf5e684fef7c6ee2c02fc68a5ceb7e7e759d08b694088d0cacb4eba59e5"},
+]
+haystack-redis = [
+    {file = "haystack-redis-0.0.1.tar.gz", hash = "sha256:ccfea88bdc1387c9f7f6f19e9bc062a3612039ef94cfd3e78cf59a96ddd269b2"},
+    {file = "haystack_redis-0.0.1-py3-none-any.whl", hash = "sha256:4fdeee5a9d8daadb1fed4584fd2ffbb25b1ed2315dacb97b53093756d6b54467"},
 ]
 html2text = [
     {file = "html2text-2020.1.16-py3-none-any.whl", hash = "sha256:c7c629882da0cf377d66f073329ccf34a12ed2adf0169b9285ae4e63ef54c82b"},
     {file = "html2text-2020.1.16.tar.gz", hash = "sha256:e296318e16b059ddb97f7a8a1d6a5c1d7af4544049a01e261731d2d5cc277bbb"},
 ]
 idna = [
-    {file = "idna-2.10-py2.py3-none-any.whl", hash = "sha256:b97d804b1e9b523befed77c48dacec60e6dcb0b5391d57af6a65a312a90648c0"},
-    {file = "idna-2.10.tar.gz", hash = "sha256:b307872f855b18632ce0c21c5e45be78c0ea7ae4c15c828c20788b26921eb3f6"},
+    {file = "idna-3.3-py3-none-any.whl", hash = "sha256:84d9dd047ffa80596e0f246e2eab0b391788b0503584e8945f2368256d2735ff"},
+    {file = "idna-3.3.tar.gz", hash = "sha256:9d643ff0a55b762d5cdb124b8eaa99c66322e2157b69160bc32796e824360e6d"},
 ]
 imagesize = [
     {file = "imagesize-1.2.0-py2.py3-none-any.whl", hash = "sha256:6965f19a6a2039c7d48bca7dba2473069ff854c36ae6f19d2cde309d998228a1"},
     {file = "imagesize-1.2.0.tar.gz", hash = "sha256:b1f6b5a4eab1f73479a50fb79fcf729514a900c341d8503d62a62dbc4127a2b1"},
 ]
-importlib-metadata = [
-    {file = "importlib_metadata-3.10.1-py3-none-any.whl", hash = "sha256:2ec0faae539743ae6aaa84b49a169670a465f7f5d64e6add98388cc29fd1f2f6"},
-    {file = "importlib_metadata-3.10.1.tar.gz", hash = "sha256:c9356b657de65c53744046fa8f7358afe0714a1af7d570c00c3835c2d724a7c1"},
-]
 iniconfig = [
     {file = "iniconfig-1.1.1-py2.py3-none-any.whl", hash = "sha256:011e24c64b7f47f6ebd835bb12a743f2fbe9a26d4cecaa7f53bc4f35ee9da8b3"},
     {file = "iniconfig-1.1.1.tar.gz", hash = "sha256:bc3af051d7d14b2ee5ef9969666def0cd1a000e121eaea580d4a313df4b37f32"},
 ]
 ipython = [
-    {file = "ipython-7.22.0-py3-none-any.whl", hash = "sha256:c0ce02dfaa5f854809ab7413c601c4543846d9da81010258ecdab299b542d199"},
-    {file = "ipython-7.22.0.tar.gz", hash = "sha256:9c900332d4c5a6de534b4befeeb7de44ad0cc42e8327fa41b7685abde58cec74"},
-]
-ipython-genutils = [
-    {file = "ipython_genutils-0.2.0-py2.py3-none-any.whl", hash = "sha256:72dd37233799e619666c9f639a9da83c34013a73e8bbc79a7a6348d93c61fab8"},
-    {file = "ipython_genutils-0.2.0.tar.gz", hash = "sha256:eb2e116e75ecef9d4d228fdc66af54269afa26ab4463042e33785b887c628ba8"},
+    {file = "ipython-7.29.0-py3-none-any.whl", hash = "sha256:a658beaf856ce46bc453366d5dc6b2ddc6c481efd3540cb28aa3943819caac9f"},
+    {file = "ipython-7.29.0.tar.gz", hash = "sha256:4f69d7423a5a1972f6347ff233e38bbf4df6a150ef20fbb00c635442ac3060aa"},
 ]
 isort = [
-    {file = "isort-5.8.0-py3-none-any.whl", hash = "sha256:2bb1680aad211e3c9944dbce1d4ba09a989f04e238296c87fe2139faa26d655d"},
-    {file = "isort-5.8.0.tar.gz", hash = "sha256:0a943902919f65c5684ac4e0154b1ad4fac6dcaa5d9f3426b732f1c8b5419be6"},
+    {file = "isort-5.10.0-py3-none-any.whl", hash = "sha256:1a18ccace2ed8910bd9458b74a3ecbafd7b2f581301b0ab65cfdd4338272d76f"},
+    {file = "isort-5.10.0.tar.gz", hash = "sha256:e52ff6d38012b131628cf0f26c51e7bd3a7c81592eefe3ac71411e692f1b9345"},
 ]
 jedi = [
     {file = "jedi-0.18.0-py2.py3-none-any.whl", hash = "sha256:18456d83f65f400ab0c2d3319e48520420ef43b23a086fdc05dff34132f0fb93"},
     {file = "jedi-0.18.0.tar.gz", hash = "sha256:92550a404bad8afed881a137ec9a461fed49eca661414be45059329614ed0707"},
 ]
 jinja2 = [
-    {file = "Jinja2-2.11.3-py2.py3-none-any.whl", hash = "sha256:03e47ad063331dd6a3f04a43eddca8a966a26ba0c5b7207a9a9e4e08f1b29419"},
-    {file = "Jinja2-2.11.3.tar.gz", hash = "sha256:a6d58433de0ae800347cab1fa3043cebbabe8baa9d29e668f1c768cb87a333c6"},
+    {file = "Jinja2-3.0.2-py3-none-any.whl", hash = "sha256:8569982d3f0889eed11dd620c706d39b60c36d6d25843961f33f77fb6bc6b20c"},
+    {file = "Jinja2-3.0.2.tar.gz", hash = "sha256:827a0e32839ab1600d4eb1c4c33ec5a8edfbc5cb42dafa13b81f182f97784b45"},
 ]
 jmespath = [
     {file = "jmespath-0.10.0-py2.py3-none-any.whl", hash = "sha256:cdf6525904cc597730141d61b36f2e4b8ecc257c420fa2f4549bac2c2d0cb72f"},
     {file = "jmespath-0.10.0.tar.gz", hash = "sha256:b85d0567b8666149a93172712e68920734333c0ce7e89b78b3e987f71e5ed4f9"},
 ]
+jwcrypto = [
+    {file = "jwcrypto-1.0-py2.py3-none-any.whl", hash = "sha256:db93a656d9a7a35dda5a68deb5c9f301f4e60507d8aef1559e0637b9ac497137"},
+    {file = "jwcrypto-1.0.tar.gz", hash = "sha256:f88816eb0a41b8f006af978ced5f171f33782525006cdb055b536a40f4d46ac9"},
+]
 kombu = [
-    {file = "kombu-5.0.2-py2.py3-none-any.whl", hash = "sha256:6dc509178ac4269b0e66ab4881f70a2035c33d3a622e20585f965986a5182006"},
-    {file = "kombu-5.0.2.tar.gz", hash = "sha256:f4965fba0a4718d47d470beeb5d6446e3357a62402b16c510b6a2f251e05ac3c"},
+    {file = "kombu-5.2.0-py3-none-any.whl", hash = "sha256:8eefdb136d15449743f5db3d7af53cf0f1eb283c011ed670986c745bef268b81"},
+    {file = "kombu-5.2.0.tar.gz", hash = "sha256:af17a0ab0c7508dd50cc9d1378d0cf68566a953fca8aaae4e8dab107a79aa440"},
 ]
 libsass = [
-    {file = "libsass-0.20.1-cp27-cp27m-macosx_10_14_x86_64.whl", hash = "sha256:4a246e4b88fd279abef8b669206228c92534d96ddcd0770d7012088c408dff23"},
-    {file = "libsass-0.20.1-cp27-cp27m-win32.whl", hash = "sha256:697f0f9fa8a1367ca9ec6869437cb235b1c537fc8519983d1d890178614a8903"},
-    {file = "libsass-0.20.1-cp27-cp27m-win_amd64.whl", hash = "sha256:1b2d415bbf6fa7da33ef46e549db1418498267b459978eff8357e5e823962d35"},
-    {file = "libsass-0.20.1-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:1521d2a8d4b397c6ec90640a1f6b5529077035efc48ef1c2e53095544e713d1b"},
-    {file = "libsass-0.20.1-cp36-abi3-macosx_10_14_x86_64.whl", hash = "sha256:2ae806427b28bc1bb7cb0258666d854fcf92ba52a04656b0b17ba5e190fb48a9"},
-    {file = "libsass-0.20.1-cp36-abi3-manylinux1_x86_64.whl", hash = "sha256:25ebc2085f5eee574761ccc8d9cd29a9b436fc970546d5ef08c6fa41eb57dff1"},
-    {file = "libsass-0.20.1-cp36-cp36m-win32.whl", hash = "sha256:553e5096414a8d4fb48d0a48f5a038d3411abe254d79deac5e008516c019e63a"},
-    {file = "libsass-0.20.1-cp36-cp36m-win_amd64.whl", hash = "sha256:e64ae2587f1a683e831409aad03ba547c245ef997e1329fffadf7a866d2510b8"},
-    {file = "libsass-0.20.1-cp37-cp37m-win32.whl", hash = "sha256:c9411fec76f480ffbacc97d8188322e02a5abca6fc78e70b86a2a2b421eae8a2"},
-    {file = "libsass-0.20.1-cp37-cp37m-win_amd64.whl", hash = "sha256:a8fd4af9f853e8bf42b1425c5e48dd90b504fa2e70d7dac5ac80b8c0a5a5fe85"},
-    {file = "libsass-0.20.1-cp38-cp38-win32.whl", hash = "sha256:f6852828e9e104d2ce0358b73c550d26dd86cc3a69439438c3b618811b9584f5"},
-    {file = "libsass-0.20.1-cp38-cp38-win_amd64.whl", hash = "sha256:daa98a51086d92aa7e9c8871cf1a8258124b90e2abf4697852a3dca619838618"},
-    {file = "libsass-0.20.1.tar.gz", hash = "sha256:e0e60836eccbf2d9e24ec978a805cd6642fa92515fbd95e3493fee276af76f8a"},
+    {file = "libsass-0.21.0-cp27-cp27m-macosx_10_14_x86_64.whl", hash = "sha256:06c8776417fe930714bdc930a3d7e795ae3d72be6ac883ff72a1b8f7c49e5ffb"},
+    {file = "libsass-0.21.0-cp27-cp27m-win32.whl", hash = "sha256:a005f298f64624f313a3ac618ab03f844c71d84ae4f4a4aec4b68d2a4ffe75eb"},
+    {file = "libsass-0.21.0-cp27-cp27m-win_amd64.whl", hash = "sha256:6b984510ed94993708c0d697b4fef2d118929bbfffc3b90037be0f5ccadf55e7"},
+    {file = "libsass-0.21.0-cp27-cp27mu-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:1e25dd9047a9392d3c59a0b869e0404f2b325a03871ee45285ee33b3664f5613"},
+    {file = "libsass-0.21.0-cp36-abi3-macosx_10_14_x86_64.whl", hash = "sha256:12f39712de38689a8b785b7db41d3ba2ea1d46f9379d81ea4595802d91fa6529"},
+    {file = "libsass-0.21.0-cp36-abi3-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:e2b1a7d093f2e76dc694c17c0c285e846d0b0deb0e8b21dc852ba1a3a4e2f1d6"},
+    {file = "libsass-0.21.0-cp36-abi3-win32.whl", hash = "sha256:abc29357ee540849faf1383e1746d40d69ed5cb6d4c346df276b258f5aa8977a"},
+    {file = "libsass-0.21.0-cp36-abi3-win_amd64.whl", hash = "sha256:659ae41af8708681fa3ec73f47b9735a6725e71c3b66ff570bfce78952f2314e"},
+    {file = "libsass-0.21.0.tar.gz", hash = "sha256:d5ba529d9ce668be9380563279f3ffe988f27bc5b299c5a28453df2e0b0fbaf2"},
 ]
 license-expression = [
     {file = "license-expression-1.2.tar.gz", hash = "sha256:7960e1dfdf20d127e75ead931476f2b5c7556df05b117a73880b22ade17d1abc"},
     {file = "license_expression-1.2-py2.py3-none-any.whl", hash = "sha256:6d97906380cecfc758a77f6d38c6760f2afade7e83d2b8295e234fe21f486fb8"},
 ]
 markupsafe = [
-    {file = "MarkupSafe-1.1.1-cp27-cp27m-macosx_10_6_intel.whl", hash = "sha256:09027a7803a62ca78792ad89403b1b7a73a01c8cb65909cd876f7fcebd79b161"},
-    {file = "MarkupSafe-1.1.1-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:e249096428b3ae81b08327a63a485ad0878de3fb939049038579ac0ef61e17e7"},
-    {file = "MarkupSafe-1.1.1-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:500d4957e52ddc3351cabf489e79c91c17f6e0899158447047588650b5e69183"},
-    {file = "MarkupSafe-1.1.1-cp27-cp27m-win32.whl", hash = "sha256:b2051432115498d3562c084a49bba65d97cf251f5a331c64a12ee7e04dacc51b"},
-    {file = "MarkupSafe-1.1.1-cp27-cp27m-win_amd64.whl", hash = "sha256:98c7086708b163d425c67c7a91bad6e466bb99d797aa64f965e9d25c12111a5e"},
-    {file = "MarkupSafe-1.1.1-cp27-cp27mu-manylinux1_i686.whl", hash = "sha256:cd5df75523866410809ca100dc9681e301e3c27567cf498077e8551b6d20e42f"},
-    {file = "MarkupSafe-1.1.1-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:43a55c2930bbc139570ac2452adf3d70cdbb3cfe5912c71cdce1c2c6bbd9c5d1"},
-    {file = "MarkupSafe-1.1.1-cp34-cp34m-macosx_10_6_intel.whl", hash = "sha256:1027c282dad077d0bae18be6794e6b6b8c91d58ed8a8d89a89d59693b9131db5"},
-    {file = "MarkupSafe-1.1.1-cp34-cp34m-manylinux1_i686.whl", hash = "sha256:62fe6c95e3ec8a7fad637b7f3d372c15ec1caa01ab47926cfdf7a75b40e0eac1"},
-    {file = "MarkupSafe-1.1.1-cp34-cp34m-manylinux1_x86_64.whl", hash = "sha256:88e5fcfb52ee7b911e8bb6d6aa2fd21fbecc674eadd44118a9cc3863f938e735"},
-    {file = "MarkupSafe-1.1.1-cp34-cp34m-win32.whl", hash = "sha256:ade5e387d2ad0d7ebf59146cc00c8044acbd863725f887353a10df825fc8ae21"},
-    {file = "MarkupSafe-1.1.1-cp34-cp34m-win_amd64.whl", hash = "sha256:09c4b7f37d6c648cb13f9230d847adf22f8171b1ccc4d5682398e77f40309235"},
-    {file = "MarkupSafe-1.1.1-cp35-cp35m-macosx_10_6_intel.whl", hash = "sha256:79855e1c5b8da654cf486b830bd42c06e8780cea587384cf6545b7d9ac013a0b"},
-    {file = "MarkupSafe-1.1.1-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:c8716a48d94b06bb3b2524c2b77e055fb313aeb4ea620c8dd03a105574ba704f"},
-    {file = "MarkupSafe-1.1.1-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:7c1699dfe0cf8ff607dbdcc1e9b9af1755371f92a68f706051cc8c37d447c905"},
-    {file = "MarkupSafe-1.1.1-cp35-cp35m-win32.whl", hash = "sha256:6dd73240d2af64df90aa7c4e7481e23825ea70af4b4922f8ede5b9e35f78a3b1"},
-    {file = "MarkupSafe-1.1.1-cp35-cp35m-win_amd64.whl", hash = "sha256:9add70b36c5666a2ed02b43b335fe19002ee5235efd4b8a89bfcf9005bebac0d"},
-    {file = "MarkupSafe-1.1.1-cp36-cp36m-macosx_10_6_intel.whl", hash = "sha256:24982cc2533820871eba85ba648cd53d8623687ff11cbb805be4ff7b4c971aff"},
-    {file = "MarkupSafe-1.1.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:d53bc011414228441014aa71dbec320c66468c1030aae3a6e29778a3382d96e5"},
-    {file = "MarkupSafe-1.1.1-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:00bc623926325b26bb9605ae9eae8a215691f33cae5df11ca5424f06f2d1f473"},
-    {file = "MarkupSafe-1.1.1-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:717ba8fe3ae9cc0006d7c451f0bb265ee07739daf76355d06366154ee68d221e"},
-    {file = "MarkupSafe-1.1.1-cp36-cp36m-manylinux2010_i686.whl", hash = "sha256:3b8a6499709d29c2e2399569d96719a1b21dcd94410a586a18526b143ec8470f"},
-    {file = "MarkupSafe-1.1.1-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:84dee80c15f1b560d55bcfe6d47b27d070b4681c699c572af2e3c7cc90a3b8e0"},
-    {file = "MarkupSafe-1.1.1-cp36-cp36m-manylinux2014_aarch64.whl", hash = "sha256:b1dba4527182c95a0db8b6060cc98ac49b9e2f5e64320e2b56e47cb2831978c7"},
-    {file = "MarkupSafe-1.1.1-cp36-cp36m-win32.whl", hash = "sha256:535f6fc4d397c1563d08b88e485c3496cf5784e927af890fb3c3aac7f933ec66"},
-    {file = "MarkupSafe-1.1.1-cp36-cp36m-win_amd64.whl", hash = "sha256:b1282f8c00509d99fef04d8ba936b156d419be841854fe901d8ae224c59f0be5"},
-    {file = "MarkupSafe-1.1.1-cp37-cp37m-macosx_10_6_intel.whl", hash = "sha256:8defac2f2ccd6805ebf65f5eeb132adcf2ab57aa11fdf4c0dd5169a004710e7d"},
-    {file = "MarkupSafe-1.1.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:bf5aa3cbcfdf57fa2ee9cd1822c862ef23037f5c832ad09cfea57fa846dec193"},
-    {file = "MarkupSafe-1.1.1-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:46c99d2de99945ec5cb54f23c8cd5689f6d7177305ebff350a58ce5f8de1669e"},
-    {file = "MarkupSafe-1.1.1-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:ba59edeaa2fc6114428f1637ffff42da1e311e29382d81b339c1817d37ec93c6"},
-    {file = "MarkupSafe-1.1.1-cp37-cp37m-manylinux2010_i686.whl", hash = "sha256:6fffc775d90dcc9aed1b89219549b329a9250d918fd0b8fa8d93d154918422e1"},
-    {file = "MarkupSafe-1.1.1-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:a6a744282b7718a2a62d2ed9d993cad6f5f585605ad352c11de459f4108df0a1"},
-    {file = "MarkupSafe-1.1.1-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:195d7d2c4fbb0ee8139a6cf67194f3973a6b3042d742ebe0a9ed36d8b6f0c07f"},
-    {file = "MarkupSafe-1.1.1-cp37-cp37m-win32.whl", hash = "sha256:b00c1de48212e4cc9603895652c5c410df699856a2853135b3967591e4beebc2"},
-    {file = "MarkupSafe-1.1.1-cp37-cp37m-win_amd64.whl", hash = "sha256:9bf40443012702a1d2070043cb6291650a0841ece432556f784f004937f0f32c"},
-    {file = "MarkupSafe-1.1.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:6788b695d50a51edb699cb55e35487e430fa21f1ed838122d722e0ff0ac5ba15"},
-    {file = "MarkupSafe-1.1.1-cp38-cp38-manylinux1_i686.whl", hash = "sha256:cdb132fc825c38e1aeec2c8aa9338310d29d337bebbd7baa06889d09a60a1fa2"},
-    {file = "MarkupSafe-1.1.1-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:13d3144e1e340870b25e7b10b98d779608c02016d5184cfb9927a9f10c689f42"},
-    {file = "MarkupSafe-1.1.1-cp38-cp38-manylinux2010_i686.whl", hash = "sha256:acf08ac40292838b3cbbb06cfe9b2cb9ec78fce8baca31ddb87aaac2e2dc3bc2"},
-    {file = "MarkupSafe-1.1.1-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:d9be0ba6c527163cbed5e0857c451fcd092ce83947944d6c14bc95441203f032"},
-    {file = "MarkupSafe-1.1.1-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:caabedc8323f1e93231b52fc32bdcde6db817623d33e100708d9a68e1f53b26b"},
-    {file = "MarkupSafe-1.1.1-cp38-cp38-win32.whl", hash = "sha256:596510de112c685489095da617b5bcbbac7dd6384aeebeda4df6025d0256a81b"},
-    {file = "MarkupSafe-1.1.1-cp38-cp38-win_amd64.whl", hash = "sha256:e8313f01ba26fbbe36c7be1966a7b7424942f670f38e666995b88d012765b9be"},
-    {file = "MarkupSafe-1.1.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:d73a845f227b0bfe8a7455ee623525ee656a9e2e749e4742706d80a6065d5e2c"},
-    {file = "MarkupSafe-1.1.1-cp39-cp39-manylinux1_i686.whl", hash = "sha256:98bae9582248d6cf62321dcb52aaf5d9adf0bad3b40582925ef7c7f0ed85fceb"},
-    {file = "MarkupSafe-1.1.1-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:2beec1e0de6924ea551859edb9e7679da6e4870d32cb766240ce17e0a0ba2014"},
-    {file = "MarkupSafe-1.1.1-cp39-cp39-manylinux2010_i686.whl", hash = "sha256:7fed13866cf14bba33e7176717346713881f56d9d2bcebab207f7a036f41b850"},
-    {file = "MarkupSafe-1.1.1-cp39-cp39-manylinux2010_x86_64.whl", hash = "sha256:6f1e273a344928347c1290119b493a1f0303c52f5a5eae5f16d74f48c15d4a85"},
-    {file = "MarkupSafe-1.1.1-cp39-cp39-manylinux2014_aarch64.whl", hash = "sha256:feb7b34d6325451ef96bc0e36e1a6c0c1c64bc1fbec4b854f4529e51887b1621"},
-    {file = "MarkupSafe-1.1.1-cp39-cp39-win32.whl", hash = "sha256:22c178a091fc6630d0d045bdb5992d2dfe14e3259760e713c490da5323866c39"},
-    {file = "MarkupSafe-1.1.1-cp39-cp39-win_amd64.whl", hash = "sha256:b7d644ddb4dbd407d31ffb699f1d140bc35478da613b441c582aeb7c43838dd8"},
-    {file = "MarkupSafe-1.1.1.tar.gz", hash = "sha256:29872e92839765e546828bb7754a68c418d927cd064fd4708fab9fe9c8bb116b"},
+    {file = "MarkupSafe-2.0.1-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:d8446c54dc28c01e5a2dbac5a25f071f6653e6e40f3a8818e8b45d790fe6ef53"},
+    {file = "MarkupSafe-2.0.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:36bc903cbb393720fad60fc28c10de6acf10dc6cc883f3e24ee4012371399a38"},
+    {file = "MarkupSafe-2.0.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2d7d807855b419fc2ed3e631034685db6079889a1f01d5d9dac950f764da3dad"},
+    {file = "MarkupSafe-2.0.1-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:add36cb2dbb8b736611303cd3bfcee00afd96471b09cda130da3581cbdc56a6d"},
+    {file = "MarkupSafe-2.0.1-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:168cd0a3642de83558a5153c8bd34f175a9a6e7f6dc6384b9655d2697312a646"},
+    {file = "MarkupSafe-2.0.1-cp310-cp310-win32.whl", hash = "sha256:99df47edb6bda1249d3e80fdabb1dab8c08ef3975f69aed437cb69d0a5de1e28"},
+    {file = "MarkupSafe-2.0.1-cp310-cp310-win_amd64.whl", hash = "sha256:e0f138900af21926a02425cf736db95be9f4af72ba1bb21453432a07f6082134"},
+    {file = "MarkupSafe-2.0.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:f9081981fe268bd86831e5c75f7de206ef275defcb82bc70740ae6dc507aee51"},
+    {file = "MarkupSafe-2.0.1-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:0955295dd5eec6cb6cc2fe1698f4c6d84af2e92de33fbcac4111913cd100a6ff"},
+    {file = "MarkupSafe-2.0.1-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:0446679737af14f45767963a1a9ef7620189912317d095f2d9ffa183a4d25d2b"},
+    {file = "MarkupSafe-2.0.1-cp36-cp36m-manylinux2010_i686.whl", hash = "sha256:f826e31d18b516f653fe296d967d700fddad5901ae07c622bb3705955e1faa94"},
+    {file = "MarkupSafe-2.0.1-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:fa130dd50c57d53368c9d59395cb5526eda596d3ffe36666cd81a44d56e48872"},
+    {file = "MarkupSafe-2.0.1-cp36-cp36m-manylinux2014_aarch64.whl", hash = "sha256:905fec760bd2fa1388bb5b489ee8ee5f7291d692638ea5f67982d968366bef9f"},
+    {file = "MarkupSafe-2.0.1-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:bf5d821ffabf0ef3533c39c518f3357b171a1651c1ff6827325e4489b0e46c3c"},
+    {file = "MarkupSafe-2.0.1-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:0d4b31cc67ab36e3392bbf3862cfbadac3db12bdd8b02a2731f509ed5b829724"},
+    {file = "MarkupSafe-2.0.1-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:baa1a4e8f868845af802979fcdbf0bb11f94f1cb7ced4c4b8a351bb60d108145"},
+    {file = "MarkupSafe-2.0.1-cp36-cp36m-win32.whl", hash = "sha256:6c4ca60fa24e85fe25b912b01e62cb969d69a23a5d5867682dd3e80b5b02581d"},
+    {file = "MarkupSafe-2.0.1-cp36-cp36m-win_amd64.whl", hash = "sha256:b2f4bf27480f5e5e8ce285a8c8fd176c0b03e93dcc6646477d4630e83440c6a9"},
+    {file = "MarkupSafe-2.0.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:0717a7390a68be14b8c793ba258e075c6f4ca819f15edfc2a3a027c823718567"},
+    {file = "MarkupSafe-2.0.1-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:6557b31b5e2c9ddf0de32a691f2312a32f77cd7681d8af66c2692efdbef84c18"},
+    {file = "MarkupSafe-2.0.1-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:49e3ceeabbfb9d66c3aef5af3a60cc43b85c33df25ce03d0031a608b0a8b2e3f"},
+    {file = "MarkupSafe-2.0.1-cp37-cp37m-manylinux2010_i686.whl", hash = "sha256:d7f9850398e85aba693bb640262d3611788b1f29a79f0c93c565694658f4071f"},
+    {file = "MarkupSafe-2.0.1-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:6a7fae0dd14cf60ad5ff42baa2e95727c3d81ded453457771d02b7d2b3f9c0c2"},
+    {file = "MarkupSafe-2.0.1-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:b7f2d075102dc8c794cbde1947378051c4e5180d52d276987b8d28a3bd58c17d"},
+    {file = "MarkupSafe-2.0.1-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e9936f0b261d4df76ad22f8fee3ae83b60d7c3e871292cd42f40b81b70afae85"},
+    {file = "MarkupSafe-2.0.1-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:2a7d351cbd8cfeb19ca00de495e224dea7e7d919659c2841bbb7f420ad03e2d6"},
+    {file = "MarkupSafe-2.0.1-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:60bf42e36abfaf9aff1f50f52644b336d4f0a3fd6d8a60ca0d054ac9f713a864"},
+    {file = "MarkupSafe-2.0.1-cp37-cp37m-win32.whl", hash = "sha256:a30e67a65b53ea0a5e62fe23682cfe22712e01f453b95233b25502f7c61cb415"},
+    {file = "MarkupSafe-2.0.1-cp37-cp37m-win_amd64.whl", hash = "sha256:611d1ad9a4288cf3e3c16014564df047fe08410e628f89805e475368bd304914"},
+    {file = "MarkupSafe-2.0.1-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:5bb28c636d87e840583ee3adeb78172efc47c8b26127267f54a9c0ec251d41a9"},
+    {file = "MarkupSafe-2.0.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:be98f628055368795d818ebf93da628541e10b75b41c559fdf36d104c5787066"},
+    {file = "MarkupSafe-2.0.1-cp38-cp38-manylinux1_i686.whl", hash = "sha256:1d609f577dc6e1aa17d746f8bd3c31aa4d258f4070d61b2aa5c4166c1539de35"},
+    {file = "MarkupSafe-2.0.1-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:7d91275b0245b1da4d4cfa07e0faedd5b0812efc15b702576d103293e252af1b"},
+    {file = "MarkupSafe-2.0.1-cp38-cp38-manylinux2010_i686.whl", hash = "sha256:01a9b8ea66f1658938f65b93a85ebe8bc016e6769611be228d797c9d998dd298"},
+    {file = "MarkupSafe-2.0.1-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:47ab1e7b91c098ab893b828deafa1203de86d0bc6ab587b160f78fe6c4011f75"},
+    {file = "MarkupSafe-2.0.1-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:97383d78eb34da7e1fa37dd273c20ad4320929af65d156e35a5e2d89566d9dfb"},
+    {file = "MarkupSafe-2.0.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6fcf051089389abe060c9cd7caa212c707e58153afa2c649f00346ce6d260f1b"},
+    {file = "MarkupSafe-2.0.1-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:5855f8438a7d1d458206a2466bf82b0f104a3724bf96a1c781ab731e4201731a"},
+    {file = "MarkupSafe-2.0.1-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:3dd007d54ee88b46be476e293f48c85048603f5f516008bee124ddd891398ed6"},
+    {file = "MarkupSafe-2.0.1-cp38-cp38-win32.whl", hash = "sha256:023cb26ec21ece8dc3907c0e8320058b2e0cb3c55cf9564da612bc325bed5e64"},
+    {file = "MarkupSafe-2.0.1-cp38-cp38-win_amd64.whl", hash = "sha256:984d76483eb32f1bcb536dc27e4ad56bba4baa70be32fa87152832cdd9db0833"},
+    {file = "MarkupSafe-2.0.1-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:2ef54abee730b502252bcdf31b10dacb0a416229b72c18b19e24a4509f273d26"},
+    {file = "MarkupSafe-2.0.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:3c112550557578c26af18a1ccc9e090bfe03832ae994343cfdacd287db6a6ae7"},
+    {file = "MarkupSafe-2.0.1-cp39-cp39-manylinux1_i686.whl", hash = "sha256:53edb4da6925ad13c07b6d26c2a852bd81e364f95301c66e930ab2aef5b5ddd8"},
+    {file = "MarkupSafe-2.0.1-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:f5653a225f31e113b152e56f154ccbe59eeb1c7487b39b9d9f9cdb58e6c79dc5"},
+    {file = "MarkupSafe-2.0.1-cp39-cp39-manylinux2010_i686.whl", hash = "sha256:4efca8f86c54b22348a5467704e3fec767b2db12fc39c6d963168ab1d3fc9135"},
+    {file = "MarkupSafe-2.0.1-cp39-cp39-manylinux2010_x86_64.whl", hash = "sha256:ab3ef638ace319fa26553db0624c4699e31a28bb2a835c5faca8f8acf6a5a902"},
+    {file = "MarkupSafe-2.0.1-cp39-cp39-manylinux2014_aarch64.whl", hash = "sha256:f8ba0e8349a38d3001fae7eadded3f6606f0da5d748ee53cc1dab1d6527b9509"},
+    {file = "MarkupSafe-2.0.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c47adbc92fc1bb2b3274c4b3a43ae0e4573d9fbff4f54cd484555edbf030baf1"},
+    {file = "MarkupSafe-2.0.1-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:37205cac2a79194e3750b0af2a5720d95f786a55ce7df90c3af697bfa100eaac"},
+    {file = "MarkupSafe-2.0.1-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:1f2ade76b9903f39aa442b4aadd2177decb66525062db244b35d71d0ee8599b6"},
+    {file = "MarkupSafe-2.0.1-cp39-cp39-win32.whl", hash = "sha256:10f82115e21dc0dfec9ab5c0223652f7197feb168c940f3ef61563fc2d6beb74"},
+    {file = "MarkupSafe-2.0.1-cp39-cp39-win_amd64.whl", hash = "sha256:693ce3f9e70a6cf7d2fb9e6c9d8b204b6b39897a2c4a1aa65728d5ac97dcc1d8"},
+    {file = "MarkupSafe-2.0.1.tar.gz", hash = "sha256:594c67807fb16238b30c44bdf74f36c02cdf22d1c8cda91ef8a0ed8dabf5620a"},
+]
+matplotlib-inline = [
+    {file = "matplotlib-inline-0.1.3.tar.gz", hash = "sha256:a04bfba22e0d1395479f866853ec1ee28eea1485c1d69a6faf00dc3e24ff34ee"},
+    {file = "matplotlib_inline-0.1.3-py3-none-any.whl", hash = "sha256:aed605ba3b72462d64d475a21a9296f400a19c4f74a31b59103d2a99ffd5aa5c"},
 ]
 mccabe = [
     {file = "mccabe-0.6.1-py2.py3-none-any.whl", hash = "sha256:ab8a6258860da4b6677da4bd2fe5dc2c659cff31b3ee4f7f5d64e79735b80d42"},
     {file = "mccabe-0.6.1.tar.gz", hash = "sha256:dd8d182285a0fe56bace7f45b5e7d1a6ebcbf524e8f3bd87eb0f125271b8831f"},
 ]
 mypy = [
-    {file = "mypy-0.812-cp35-cp35m-macosx_10_9_x86_64.whl", hash = "sha256:a26f8ec704e5a7423c8824d425086705e381b4f1dfdef6e3a1edab7ba174ec49"},
-    {file = "mypy-0.812-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:28fb5479c494b1bab244620685e2eb3c3f988d71fd5d64cc753195e8ed53df7c"},
-    {file = "mypy-0.812-cp35-cp35m-manylinux2010_x86_64.whl", hash = "sha256:9743c91088d396c1a5a3c9978354b61b0382b4e3c440ce83cf77994a43e8c521"},
-    {file = "mypy-0.812-cp35-cp35m-win_amd64.whl", hash = "sha256:d7da2e1d5f558c37d6e8c1246f1aec1e7349e4913d8fb3cb289a35de573fe2eb"},
-    {file = "mypy-0.812-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:4eec37370483331d13514c3f55f446fc5248d6373e7029a29ecb7b7494851e7a"},
-    {file = "mypy-0.812-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:d65cc1df038ef55a99e617431f0553cd77763869eebdf9042403e16089fe746c"},
-    {file = "mypy-0.812-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:61a3d5b97955422964be6b3baf05ff2ce7f26f52c85dd88db11d5e03e146a3a6"},
-    {file = "mypy-0.812-cp36-cp36m-win_amd64.whl", hash = "sha256:25adde9b862f8f9aac9d2d11971f226bd4c8fbaa89fb76bdadb267ef22d10064"},
-    {file = "mypy-0.812-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:552a815579aa1e995f39fd05dde6cd378e191b063f031f2acfe73ce9fb7f9e56"},
-    {file = "mypy-0.812-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:499c798053cdebcaa916eef8cd733e5584b5909f789de856b482cd7d069bdad8"},
-    {file = "mypy-0.812-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:5873888fff1c7cf5b71efbe80e0e73153fe9212fafdf8e44adfe4c20ec9f82d7"},
-    {file = "mypy-0.812-cp37-cp37m-win_amd64.whl", hash = "sha256:9f94aac67a2045ec719ffe6111df543bac7874cee01f41928f6969756e030564"},
-    {file = "mypy-0.812-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:d23e0ea196702d918b60c8288561e722bf437d82cb7ef2edcd98cfa38905d506"},
-    {file = "mypy-0.812-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:674e822aa665b9fd75130c6c5f5ed9564a38c6cea6a6432ce47eafb68ee578c5"},
-    {file = "mypy-0.812-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:abf7e0c3cf117c44d9285cc6128856106183938c68fd4944763003decdcfeb66"},
-    {file = "mypy-0.812-cp38-cp38-win_amd64.whl", hash = "sha256:0d0a87c0e7e3a9becdfbe936c981d32e5ee0ccda3e0f07e1ef2c3d1a817cf73e"},
-    {file = "mypy-0.812-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:7ce3175801d0ae5fdfa79b4f0cfed08807af4d075b402b7e294e6aa72af9aa2a"},
-    {file = "mypy-0.812-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:b09669bcda124e83708f34a94606e01b614fa71931d356c1f1a5297ba11f110a"},
-    {file = "mypy-0.812-cp39-cp39-manylinux2010_x86_64.whl", hash = "sha256:33f159443db0829d16f0a8d83d94df3109bb6dd801975fe86bacb9bf71628e97"},
-    {file = "mypy-0.812-cp39-cp39-win_amd64.whl", hash = "sha256:3f2aca7f68580dc2508289c729bd49ee929a436208d2b2b6aab15745a70a57df"},
-    {file = "mypy-0.812-py3-none-any.whl", hash = "sha256:2f9b3407c58347a452fc0736861593e105139b905cca7d097e413453a1d650b4"},
-    {file = "mypy-0.812.tar.gz", hash = "sha256:cd07039aa5df222037005b08fbbfd69b3ab0b0bd7a07d7906de75ae52c4e3119"},
+    {file = "mypy-0.910-cp35-cp35m-macosx_10_9_x86_64.whl", hash = "sha256:a155d80ea6cee511a3694b108c4494a39f42de11ee4e61e72bc424c490e46457"},
+    {file = "mypy-0.910-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:b94e4b785e304a04ea0828759172a15add27088520dc7e49ceade7834275bedb"},
+    {file = "mypy-0.910-cp35-cp35m-manylinux2010_x86_64.whl", hash = "sha256:088cd9c7904b4ad80bec811053272986611b84221835e079be5bcad029e79dd9"},
+    {file = "mypy-0.910-cp35-cp35m-win_amd64.whl", hash = "sha256:adaeee09bfde366d2c13fe6093a7df5df83c9a2ba98638c7d76b010694db760e"},
+    {file = "mypy-0.910-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:ecd2c3fe726758037234c93df7e98deb257fd15c24c9180dacf1ef829da5f921"},
+    {file = "mypy-0.910-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:d9dd839eb0dc1bbe866a288ba3c1afc33a202015d2ad83b31e875b5905a079b6"},
+    {file = "mypy-0.910-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:3e382b29f8e0ccf19a2df2b29a167591245df90c0b5a2542249873b5c1d78212"},
+    {file = "mypy-0.910-cp36-cp36m-win_amd64.whl", hash = "sha256:53fd2eb27a8ee2892614370896956af2ff61254c275aaee4c230ae771cadd885"},
+    {file = "mypy-0.910-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:b6fb13123aeef4a3abbcfd7e71773ff3ff1526a7d3dc538f3929a49b42be03f0"},
+    {file = "mypy-0.910-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:e4dab234478e3bd3ce83bac4193b2ecd9cf94e720ddd95ce69840273bf44f6de"},
+    {file = "mypy-0.910-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:7df1ead20c81371ccd6091fa3e2878559b5c4d4caadaf1a484cf88d93ca06703"},
+    {file = "mypy-0.910-cp37-cp37m-win_amd64.whl", hash = "sha256:0aadfb2d3935988ec3815952e44058a3100499f5be5b28c34ac9d79f002a4a9a"},
+    {file = "mypy-0.910-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:ec4e0cd079db280b6bdabdc807047ff3e199f334050db5cbb91ba3e959a67504"},
+    {file = "mypy-0.910-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:119bed3832d961f3a880787bf621634ba042cb8dc850a7429f643508eeac97b9"},
+    {file = "mypy-0.910-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:866c41f28cee548475f146aa4d39a51cf3b6a84246969f3759cb3e9c742fc072"},
+    {file = "mypy-0.910-cp38-cp38-win_amd64.whl", hash = "sha256:ceb6e0a6e27fb364fb3853389607cf7eb3a126ad335790fa1e14ed02fba50811"},
+    {file = "mypy-0.910-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:1a85e280d4d217150ce8cb1a6dddffd14e753a4e0c3cf90baabb32cefa41b59e"},
+    {file = "mypy-0.910-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:42c266ced41b65ed40a282c575705325fa7991af370036d3f134518336636f5b"},
+    {file = "mypy-0.910-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:3c4b8ca36877fc75339253721f69603a9c7fdb5d4d5a95a1a1b899d8b86a4de2"},
+    {file = "mypy-0.910-cp39-cp39-manylinux2010_x86_64.whl", hash = "sha256:c0df2d30ed496a08de5daed2a9ea807d07c21ae0ab23acf541ab88c24b26ab97"},
+    {file = "mypy-0.910-cp39-cp39-win_amd64.whl", hash = "sha256:c6c2602dffb74867498f86e6129fd52a2770c48b7cd3ece77ada4fa38f94eba8"},
+    {file = "mypy-0.910-py3-none-any.whl", hash = "sha256:ef565033fa5a958e62796867b1df10c40263ea9ded87164d67572834e57a174d"},
+    {file = "mypy-0.910.tar.gz", hash = "sha256:704098302473cb31a218f1775a873b376b30b4c18229421e9e9dc8916fd16150"},
 ]
 mypy-extensions = [
     {file = "mypy_extensions-0.4.3-py2.py3-none-any.whl", hash = "sha256:090fedd75945a69ae91ce1303b5824f428daf5a028d2f6ab8a299250a846f15d"},
     {file = "mypy_extensions-0.4.3.tar.gz", hash = "sha256:2d82818f5bb3e369420cb3c4060a7970edba416647068eb4c5343488a6c604a8"},
 ]
+oauthlib = [
+    {file = "oauthlib-3.1.1-py2.py3-none-any.whl", hash = "sha256:42bf6354c2ed8c6acb54d971fce6f88193d97297e18602a3a886603f9d7730cc"},
+    {file = "oauthlib-3.1.1.tar.gz", hash = "sha256:8f0215fcc533dd8dd1bee6f4c412d4f0cd7297307d43ac61666389e3bc3198a3"},
+]
 packaging = [
-    {file = "packaging-20.9-py2.py3-none-any.whl", hash = "sha256:67714da7f7bc052e064859c05c595155bd1ee9f69f76557e21f051443c20947a"},
-    {file = "packaging-20.9.tar.gz", hash = "sha256:5b327ac1320dc863dca72f4514ecc086f31186744b84a230374cc1fd776feae5"},
+    {file = "packaging-21.2-py3-none-any.whl", hash = "sha256:14317396d1e8cdb122989b916fa2c7e9ca8e2be9e8060a6eff75b6b7b4d8a7e0"},
+    {file = "packaging-21.2.tar.gz", hash = "sha256:096d689d78ca690e4cd8a89568ba06d07ca097e3306a4381635073ca91479966"},
 ]
 parso = [
     {file = "parso-0.8.2-py2.py3-none-any.whl", hash = "sha256:a8c4922db71e4fdb90e0d0bc6e50f9b273d3397925e5e60a717e719201778d22"},
@@ -3174,12 +4523,12 @@ parsy = [
     {file = "parsy-1.1.0.tar.gz", hash = "sha256:36173ba01a5372c7a1b32352cc73a279a49198f52252adf1c8c1ed41d1f94e8d"},
 ]
 pathspec = [
-    {file = "pathspec-0.8.1-py2.py3-none-any.whl", hash = "sha256:aa0cb481c4041bf52ffa7b0d8fa6cd3e88a2ca4879c533c9153882ee2556790d"},
-    {file = "pathspec-0.8.1.tar.gz", hash = "sha256:86379d6b86d75816baba717e64b1a3a3469deb93bb76d613c9ce79edc5cb68fd"},
+    {file = "pathspec-0.9.0-py2.py3-none-any.whl", hash = "sha256:7d15c4ddb0b5c802d161efc417ec1a2558ea2653c2e8ad9c19098201dc1c993a"},
+    {file = "pathspec-0.9.0.tar.gz", hash = "sha256:e564499435a2673d586f6b2130bb5b95f04a3ba06f81b8f895b651a3c76aabb1"},
 ]
 pbr = [
-    {file = "pbr-5.5.1-py2.py3-none-any.whl", hash = "sha256:b236cde0ac9a6aedd5e3c34517b423cd4fd97ef723849da6b0d2231142d89c00"},
-    {file = "pbr-5.5.1.tar.gz", hash = "sha256:5fad80b613c402d5b7df7bd84812548b2a61e9977387a80a5fc5c396492b13c9"},
+    {file = "pbr-5.7.0-py2.py3-none-any.whl", hash = "sha256:60002958e459b195e8dbe61bf22bcf344eedf1b4e03a321a5414feb15566100c"},
+    {file = "pbr-5.7.0.tar.gz", hash = "sha256:4651ca1445e80f2781827305de3d76b3ce53195f2227762684eb08f17bc473b7"},
 ]
 persisting-theory = [
     {file = "persisting-theory-0.2.1.tar.gz", hash = "sha256:00ff7dcc8f481ff75c770ca5797d968e8725b6df1f77fe0cf7d20fa1e5790c0a"},
@@ -3189,63 +4538,71 @@ pexpect = [
     {file = "pexpect-4.8.0.tar.gz", hash = "sha256:fc65a43959d153d0114afe13997d439c22823a27cefceb5ff35c2178c6784c0c"},
 ]
 pg8000 = [
-    {file = "pg8000-1.19.2-py3-none-any.whl", hash = "sha256:53924c8d7566cc66aaf4aecb3d8128703ef4054a510602290d62e607398e8d72"},
-    {file = "pg8000-1.19.2.tar.gz", hash = "sha256:44cbb4d3c912f2da167ca02bf98a0040f7e93260e4ef114c28d5d6152012ea07"},
+    {file = "pg8000-1.22.0-py3-none-any.whl", hash = "sha256:a0e82542f4a56b2139c41ff09c1aeff294c10b6500bb6c57890c0c1c551cbc03"},
+    {file = "pg8000-1.22.0.tar.gz", hash = "sha256:c5172252fc92142ec104cd5e7231be4580a1a0a814403707bafbf7bb8383a29a"},
 ]
 phonenumbers = [
-    {file = "phonenumbers-8.12.21-py2.py3-none-any.whl", hash = "sha256:7d66d6f09541568941e51fe2275de0f2d519b3234c80c8c166c56d01d098d3fc"},
-    {file = "phonenumbers-8.12.21.tar.gz", hash = "sha256:271f26271338616460eef8b878e5879401299adfc0cd0891c48b4744afbb45b2"},
+    {file = "phonenumbers-8.12.36-py2.py3-none-any.whl", hash = "sha256:fa91fff1cefee6873c78f08c767f341a7658c849ffb74b605cb38cefef0e76fa"},
+    {file = "phonenumbers-8.12.36.tar.gz", hash = "sha256:e29717fcf86d68082fc6e42ca07e52bff863b6e0b354edd1644ba15c35ef213d"},
 ]
 pickleshare = [
     {file = "pickleshare-0.7.5-py2.py3-none-any.whl", hash = "sha256:9649af414d74d4df115d5d718f82acb59c9d418196b7b4290ed47a12ce62df56"},
     {file = "pickleshare-0.7.5.tar.gz", hash = "sha256:87683d47965c1da65cdacaf31c8441d12b8044cdec9aca500cd78fc2c683afca"},
 ]
 pillow = [
-    {file = "Pillow-8.2.0-cp36-cp36m-macosx_10_10_x86_64.whl", hash = "sha256:dc38f57d8f20f06dd7c3161c59ca2c86893632623f33a42d592f097b00f720a9"},
-    {file = "Pillow-8.2.0-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:a013cbe25d20c2e0c4e85a9daf438f85121a4d0344ddc76e33fd7e3965d9af4b"},
-    {file = "Pillow-8.2.0-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:8bb1e155a74e1bfbacd84555ea62fa21c58e0b4e7e6b20e4447b8d07990ac78b"},
-    {file = "Pillow-8.2.0-cp36-cp36m-manylinux2014_aarch64.whl", hash = "sha256:c5236606e8570542ed424849f7852a0ff0bce2c4c8d0ba05cc202a5a9c97dee9"},
-    {file = "Pillow-8.2.0-cp36-cp36m-win32.whl", hash = "sha256:12e5e7471f9b637762453da74e390e56cc43e486a88289995c1f4c1dc0bfe727"},
-    {file = "Pillow-8.2.0-cp36-cp36m-win_amd64.whl", hash = "sha256:5afe6b237a0b81bd54b53f835a153770802f164c5570bab5e005aad693dab87f"},
-    {file = "Pillow-8.2.0-cp37-cp37m-macosx_10_10_x86_64.whl", hash = "sha256:cb7a09e173903541fa888ba010c345893cd9fc1b5891aaf060f6ca77b6a3722d"},
-    {file = "Pillow-8.2.0-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:0d19d70ee7c2ba97631bae1e7d4725cdb2ecf238178096e8c82ee481e189168a"},
-    {file = "Pillow-8.2.0-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:083781abd261bdabf090ad07bb69f8f5599943ddb539d64497ed021b2a67e5a9"},
-    {file = "Pillow-8.2.0-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:c6b39294464b03457f9064e98c124e09008b35a62e3189d3513e5148611c9388"},
-    {file = "Pillow-8.2.0-cp37-cp37m-win32.whl", hash = "sha256:01425106e4e8cee195a411f729cff2a7d61813b0b11737c12bd5991f5f14bcd5"},
-    {file = "Pillow-8.2.0-cp37-cp37m-win_amd64.whl", hash = "sha256:3b570f84a6161cf8865c4e08adf629441f56e32f180f7aa4ccbd2e0a5a02cba2"},
-    {file = "Pillow-8.2.0-cp38-cp38-macosx_10_10_x86_64.whl", hash = "sha256:031a6c88c77d08aab84fecc05c3cde8414cd6f8406f4d2b16fed1e97634cc8a4"},
-    {file = "Pillow-8.2.0-cp38-cp38-manylinux1_i686.whl", hash = "sha256:66cc56579fd91f517290ab02c51e3a80f581aba45fd924fcdee01fa06e635812"},
-    {file = "Pillow-8.2.0-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:6c32cc3145928c4305d142ebec682419a6c0a8ce9e33db900027ddca1ec39178"},
-    {file = "Pillow-8.2.0-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:624b977355cde8b065f6d51b98497d6cd5fbdd4f36405f7a8790e3376125e2bb"},
-    {file = "Pillow-8.2.0-cp38-cp38-win32.whl", hash = "sha256:5cbf3e3b1014dddc45496e8cf38b9f099c95a326275885199f427825c6522232"},
-    {file = "Pillow-8.2.0-cp38-cp38-win_amd64.whl", hash = "sha256:463822e2f0d81459e113372a168f2ff59723e78528f91f0bd25680ac185cf797"},
-    {file = "Pillow-8.2.0-cp39-cp39-macosx_10_10_x86_64.whl", hash = "sha256:95d5ef984eff897850f3a83883363da64aae1000e79cb3c321915468e8c6add5"},
-    {file = "Pillow-8.2.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:b91c36492a4bbb1ee855b7d16fe51379e5f96b85692dc8210831fbb24c43e484"},
-    {file = "Pillow-8.2.0-cp39-cp39-manylinux1_i686.whl", hash = "sha256:d68cb92c408261f806b15923834203f024110a2e2872ecb0bd2a110f89d3c602"},
-    {file = "Pillow-8.2.0-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:f217c3954ce5fd88303fc0c317af55d5e0204106d86dea17eb8205700d47dec2"},
-    {file = "Pillow-8.2.0-cp39-cp39-manylinux2014_aarch64.whl", hash = "sha256:5b70110acb39f3aff6b74cf09bb4169b167e2660dabc304c1e25b6555fa781ef"},
-    {file = "Pillow-8.2.0-cp39-cp39-win32.whl", hash = "sha256:a7d5e9fad90eff8f6f6106d3b98b553a88b6f976e51fce287192a5d2d5363713"},
-    {file = "Pillow-8.2.0-cp39-cp39-win_amd64.whl", hash = "sha256:238c197fc275b475e87c1453b05b467d2d02c2915fdfdd4af126145ff2e4610c"},
-    {file = "Pillow-8.2.0-pp36-pypy36_pp73-macosx_10_10_x86_64.whl", hash = "sha256:0e04d61f0064b545b989126197930807c86bcbd4534d39168f4aa5fda39bb8f9"},
-    {file = "Pillow-8.2.0-pp36-pypy36_pp73-manylinux2010_i686.whl", hash = "sha256:63728564c1410d99e6d1ae8e3b810fe012bc440952168af0a2877e8ff5ab96b9"},
-    {file = "Pillow-8.2.0-pp36-pypy36_pp73-manylinux2010_x86_64.whl", hash = "sha256:c03c07ed32c5324939b19e36ae5f75c660c81461e312a41aea30acdd46f93a7c"},
-    {file = "Pillow-8.2.0-pp37-pypy37_pp73-macosx_10_10_x86_64.whl", hash = "sha256:4d98abdd6b1e3bf1a1cbb14c3895226816e666749ac040c4e2554231068c639b"},
-    {file = "Pillow-8.2.0-pp37-pypy37_pp73-manylinux2010_i686.whl", hash = "sha256:aac00e4bc94d1b7813fe882c28990c1bc2f9d0e1aa765a5f2b516e8a6a16a9e4"},
-    {file = "Pillow-8.2.0-pp37-pypy37_pp73-manylinux2010_x86_64.whl", hash = "sha256:22fd0f42ad15dfdde6c581347eaa4adb9a6fc4b865f90b23378aa7914895e120"},
-    {file = "Pillow-8.2.0-pp37-pypy37_pp73-win32.whl", hash = "sha256:e98eca29a05913e82177b3ba3d198b1728e164869c613d76d0de4bde6768a50e"},
-    {file = "Pillow-8.2.0.tar.gz", hash = "sha256:a787ab10d7bb5494e5f76536ac460741788f1fbce851068d73a87ca7c35fc3e1"},
+    {file = "Pillow-8.4.0-cp310-cp310-macosx_10_10_universal2.whl", hash = "sha256:81f8d5c81e483a9442d72d182e1fb6dcb9723f289a57e8030811bac9ea3fef8d"},
+    {file = "Pillow-8.4.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:3f97cfb1e5a392d75dd8b9fd274d205404729923840ca94ca45a0af57e13dbe6"},
+    {file = "Pillow-8.4.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:eb9fc393f3c61f9054e1ed26e6fe912c7321af2f41ff49d3f83d05bacf22cc78"},
+    {file = "Pillow-8.4.0-cp310-cp310-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d82cdb63100ef5eedb8391732375e6d05993b765f72cb34311fab92103314649"},
+    {file = "Pillow-8.4.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:62cc1afda735a8d109007164714e73771b499768b9bb5afcbbee9d0ff374b43f"},
+    {file = "Pillow-8.4.0-cp310-cp310-win32.whl", hash = "sha256:e3dacecfbeec9a33e932f00c6cd7996e62f53ad46fbe677577394aaa90ee419a"},
+    {file = "Pillow-8.4.0-cp310-cp310-win_amd64.whl", hash = "sha256:620582db2a85b2df5f8a82ddeb52116560d7e5e6b055095f04ad828d1b0baa39"},
+    {file = "Pillow-8.4.0-cp36-cp36m-macosx_10_10_x86_64.whl", hash = "sha256:1bc723b434fbc4ab50bb68e11e93ce5fb69866ad621e3c2c9bdb0cd70e345f55"},
+    {file = "Pillow-8.4.0-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:72cbcfd54df6caf85cc35264c77ede902452d6df41166010262374155947460c"},
+    {file = "Pillow-8.4.0-cp36-cp36m-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:70ad9e5c6cb9b8487280a02c0ad8a51581dcbbe8484ce058477692a27c151c0a"},
+    {file = "Pillow-8.4.0-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:25a49dc2e2f74e65efaa32b153527fc5ac98508d502fa46e74fa4fd678ed6645"},
+    {file = "Pillow-8.4.0-cp36-cp36m-win32.whl", hash = "sha256:93ce9e955cc95959df98505e4608ad98281fff037350d8c2671c9aa86bcf10a9"},
+    {file = "Pillow-8.4.0-cp36-cp36m-win_amd64.whl", hash = "sha256:2e4440b8f00f504ee4b53fe30f4e381aae30b0568193be305256b1462216feff"},
+    {file = "Pillow-8.4.0-cp37-cp37m-macosx_10_10_x86_64.whl", hash = "sha256:8c803ac3c28bbc53763e6825746f05cc407b20e4a69d0122e526a582e3b5e153"},
+    {file = "Pillow-8.4.0-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c8a17b5d948f4ceeceb66384727dde11b240736fddeda54ca740b9b8b1556b29"},
+    {file = "Pillow-8.4.0-cp37-cp37m-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1394a6ad5abc838c5cd8a92c5a07535648cdf6d09e8e2d6df916dfa9ea86ead8"},
+    {file = "Pillow-8.4.0-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:792e5c12376594bfcb986ebf3855aa4b7c225754e9a9521298e460e92fb4a488"},
+    {file = "Pillow-8.4.0-cp37-cp37m-win32.whl", hash = "sha256:d99ec152570e4196772e7a8e4ba5320d2d27bf22fdf11743dd882936ed64305b"},
+    {file = "Pillow-8.4.0-cp37-cp37m-win_amd64.whl", hash = "sha256:7b7017b61bbcdd7f6363aeceb881e23c46583739cb69a3ab39cb384f6ec82e5b"},
+    {file = "Pillow-8.4.0-cp38-cp38-macosx_10_10_x86_64.whl", hash = "sha256:d89363f02658e253dbd171f7c3716a5d340a24ee82d38aab9183f7fdf0cdca49"},
+    {file = "Pillow-8.4.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:0a0956fdc5defc34462bb1c765ee88d933239f9a94bc37d132004775241a7585"},
+    {file = "Pillow-8.4.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5b7bb9de00197fb4261825c15551adf7605cf14a80badf1761d61e59da347779"},
+    {file = "Pillow-8.4.0-cp38-cp38-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:72b9e656e340447f827885b8d7a15fc8c4e68d410dc2297ef6787eec0f0ea409"},
+    {file = "Pillow-8.4.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a5a4532a12314149d8b4e4ad8ff09dde7427731fcfa5917ff16d0291f13609df"},
+    {file = "Pillow-8.4.0-cp38-cp38-win32.whl", hash = "sha256:82aafa8d5eb68c8463b6e9baeb4f19043bb31fefc03eb7b216b51e6a9981ae09"},
+    {file = "Pillow-8.4.0-cp38-cp38-win_amd64.whl", hash = "sha256:066f3999cb3b070a95c3652712cffa1a748cd02d60ad7b4e485c3748a04d9d76"},
+    {file = "Pillow-8.4.0-cp39-cp39-macosx_10_10_x86_64.whl", hash = "sha256:5503c86916d27c2e101b7f71c2ae2cddba01a2cf55b8395b0255fd33fa4d1f1a"},
+    {file = "Pillow-8.4.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:4acc0985ddf39d1bc969a9220b51d94ed51695d455c228d8ac29fcdb25810e6e"},
+    {file = "Pillow-8.4.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0b052a619a8bfcf26bd8b3f48f45283f9e977890263e4571f2393ed8898d331b"},
+    {file = "Pillow-8.4.0-cp39-cp39-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:493cb4e415f44cd601fcec11c99836f707bb714ab03f5ed46ac25713baf0ff20"},
+    {file = "Pillow-8.4.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b8831cb7332eda5dc89b21a7bce7ef6ad305548820595033a4b03cf3091235ed"},
+    {file = "Pillow-8.4.0-cp39-cp39-win32.whl", hash = "sha256:5e9ac5f66616b87d4da618a20ab0a38324dbe88d8a39b55be8964eb520021e02"},
+    {file = "Pillow-8.4.0-cp39-cp39-win_amd64.whl", hash = "sha256:3eb1ce5f65908556c2d8685a8f0a6e989d887ec4057326f6c22b24e8a172c66b"},
+    {file = "Pillow-8.4.0-pp36-pypy36_pp73-macosx_10_10_x86_64.whl", hash = "sha256:ddc4d832a0f0b4c52fff973a0d44b6c99839a9d016fe4e6a1cb8f3eea96479c2"},
+    {file = "Pillow-8.4.0-pp36-pypy36_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:9a3e5ddc44c14042f0844b8cf7d2cd455f6cc80fd7f5eefbe657292cf601d9ad"},
+    {file = "Pillow-8.4.0-pp36-pypy36_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c70e94281588ef053ae8998039610dbd71bc509e4acbc77ab59d7d2937b10698"},
+    {file = "Pillow-8.4.0-pp37-pypy37_pp73-macosx_10_10_x86_64.whl", hash = "sha256:3862b7256046fcd950618ed22d1d60b842e3a40a48236a5498746f21189afbbc"},
+    {file = "Pillow-8.4.0-pp37-pypy37_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a4901622493f88b1a29bd30ec1a2f683782e57c3c16a2dbc7f2595ba01f639df"},
+    {file = "Pillow-8.4.0-pp37-pypy37_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:84c471a734240653a0ec91dec0996696eea227eafe72a33bd06c92697728046b"},
+    {file = "Pillow-8.4.0-pp37-pypy37_pp73-win_amd64.whl", hash = "sha256:244cf3b97802c34c41905d22810846802a3329ddcb93ccc432870243211c79fc"},
+    {file = "Pillow-8.4.0.tar.gz", hash = "sha256:b8e2f83c56e141920c39464b852de3719dfbfb6e3c99a2d8da0edf4fb33176ed"},
 ]
 pluggy = [
-    {file = "pluggy-0.13.1-py2.py3-none-any.whl", hash = "sha256:966c145cd83c96502c3c3868f50408687b38434af77734af1e9ca461a4081d2d"},
-    {file = "pluggy-0.13.1.tar.gz", hash = "sha256:15b2acde666561e1298d71b523007ed7364de07029219b604cf808bfa1c765b0"},
+    {file = "pluggy-1.0.0-py2.py3-none-any.whl", hash = "sha256:74134bbf457f031a36d68416e1509f34bd5ccc019f0bcc952c7b909d06b37bd3"},
+    {file = "pluggy-1.0.0.tar.gz", hash = "sha256:4224373bacce55f955a878bf9cfa763c1e360858e330072059e10bad68531159"},
 ]
 prometheus-client = [
-    {file = "prometheus_client-0.10.1-py2.py3-none-any.whl", hash = "sha256:030e4f9df5f53db2292eec37c6255957eb76168c6f974e4176c711cf91ed34aa"},
-    {file = "prometheus_client-0.10.1.tar.gz", hash = "sha256:b6c5a9643e3545bcbfd9451766cbaa5d9c67e7303c7bc32c750b6fa70ecb107d"},
+    {file = "prometheus_client-0.12.0-py2.py3-none-any.whl", hash = "sha256:317453ebabff0a1b02df7f708efbab21e3489e7072b61cb6957230dd004a0af0"},
+    {file = "prometheus_client-0.12.0.tar.gz", hash = "sha256:1b12ba48cee33b9b0b9de64a1047cbd3c5f2d0ab6ebcead7ddda613a750ec3c5"},
 ]
 prompt-toolkit = [
-    {file = "prompt_toolkit-3.0.18-py3-none-any.whl", hash = "sha256:bf00f22079f5fadc949f42ae8ff7f05702826a97059ffcc6281036ad40ac6f04"},
-    {file = "prompt_toolkit-3.0.18.tar.gz", hash = "sha256:e1b4f11b9336a28fa11810bc623c357420f69dfdb6d2dac41ca2c21a55c033bc"},
+    {file = "prompt_toolkit-3.0.22-py3-none-any.whl", hash = "sha256:48d85cdca8b6c4f16480c7ce03fd193666b62b0a21667ca56b4bb5ad679d1170"},
+    {file = "prompt_toolkit-3.0.22.tar.gz", hash = "sha256:449f333dd120bd01f5d296a8ce1452114ba3a71fae7288d2f0ae2c918764fa72"},
 ]
 psutil = [
     {file = "psutil-5.8.0-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:0066a82f7b1b37d334e68697faba68e5ad5e858279fd6351c8ca6024e8d6ba64"},
@@ -3278,127 +4635,103 @@ psutil = [
     {file = "psutil-5.8.0.tar.gz", hash = "sha256:0c9ccb99ab76025f2f0bbecf341d4656e9c1351db8cc8a03ccd62e318ab4b5c6"},
 ]
 psycopg2 = [
-    {file = "psycopg2-2.8.6-cp27-cp27m-win32.whl", hash = "sha256:068115e13c70dc5982dfc00c5d70437fe37c014c808acce119b5448361c03725"},
-    {file = "psycopg2-2.8.6-cp27-cp27m-win_amd64.whl", hash = "sha256:d160744652e81c80627a909a0e808f3c6653a40af435744de037e3172cf277f5"},
-    {file = "psycopg2-2.8.6-cp34-cp34m-win32.whl", hash = "sha256:b8cae8b2f022efa1f011cc753adb9cbadfa5a184431d09b273fb49b4167561ad"},
-    {file = "psycopg2-2.8.6-cp34-cp34m-win_amd64.whl", hash = "sha256:f22ea9b67aea4f4a1718300908a2fb62b3e4276cf00bd829a97ab5894af42ea3"},
-    {file = "psycopg2-2.8.6-cp35-cp35m-win32.whl", hash = "sha256:26e7fd115a6db75267b325de0fba089b911a4a12ebd3d0b5e7acb7028bc46821"},
-    {file = "psycopg2-2.8.6-cp35-cp35m-win_amd64.whl", hash = "sha256:00195b5f6832dbf2876b8bf77f12bdce648224c89c880719c745b90515233301"},
-    {file = "psycopg2-2.8.6-cp36-cp36m-win32.whl", hash = "sha256:a49833abfdede8985ba3f3ec641f771cca215479f41523e99dace96d5b8cce2a"},
-    {file = "psycopg2-2.8.6-cp36-cp36m-win_amd64.whl", hash = "sha256:f974c96fca34ae9e4f49839ba6b78addf0346777b46c4da27a7bf54f48d3057d"},
-    {file = "psycopg2-2.8.6-cp37-cp37m-win32.whl", hash = "sha256:6a3d9efb6f36f1fe6aa8dbb5af55e067db802502c55a9defa47c5a1dad41df84"},
-    {file = "psycopg2-2.8.6-cp37-cp37m-win_amd64.whl", hash = "sha256:56fee7f818d032f802b8eed81ef0c1232b8b42390df189cab9cfa87573fe52c5"},
-    {file = "psycopg2-2.8.6-cp38-cp38-win32.whl", hash = "sha256:ad2fe8a37be669082e61fb001c185ffb58867fdbb3e7a6b0b0d2ffe232353a3e"},
-    {file = "psycopg2-2.8.6-cp38-cp38-win_amd64.whl", hash = "sha256:56007a226b8e95aa980ada7abdea6b40b75ce62a433bd27cec7a8178d57f4051"},
-    {file = "psycopg2-2.8.6-cp39-cp39-win32.whl", hash = "sha256:2c93d4d16933fea5bbacbe1aaf8fa8c1348740b2e50b3735d1b0bf8154cbf0f3"},
-    {file = "psycopg2-2.8.6-cp39-cp39-win_amd64.whl", hash = "sha256:d5062ae50b222da28253059880a871dc87e099c25cb68acf613d9d227413d6f7"},
-    {file = "psycopg2-2.8.6.tar.gz", hash = "sha256:fb23f6c71107c37fd667cb4ea363ddeb936b348bbd6449278eb92c189699f543"},
+    {file = "psycopg2-2.9.1-cp36-cp36m-win32.whl", hash = "sha256:7f91312f065df517187134cce8e395ab37f5b601a42446bdc0f0d51773621854"},
+    {file = "psycopg2-2.9.1-cp36-cp36m-win_amd64.whl", hash = "sha256:830c8e8dddab6b6716a4bf73a09910c7954a92f40cf1d1e702fb93c8a919cc56"},
+    {file = "psycopg2-2.9.1-cp37-cp37m-win32.whl", hash = "sha256:89409d369f4882c47f7ea20c42c5046879ce22c1e4ea20ef3b00a4dfc0a7f188"},
+    {file = "psycopg2-2.9.1-cp37-cp37m-win_amd64.whl", hash = "sha256:7640e1e4d72444ef012e275e7b53204d7fab341fb22bc76057ede22fe6860b25"},
+    {file = "psycopg2-2.9.1-cp38-cp38-win32.whl", hash = "sha256:079d97fc22de90da1d370c90583659a9f9a6ee4007355f5825e5f1c70dffc1fa"},
+    {file = "psycopg2-2.9.1-cp38-cp38-win_amd64.whl", hash = "sha256:2c992196719fadda59f72d44603ee1a2fdcc67de097eea38d41c7ad9ad246e62"},
+    {file = "psycopg2-2.9.1-cp39-cp39-win32.whl", hash = "sha256:2087013c159a73e09713294a44d0c8008204d06326006b7f652bef5ace66eebb"},
+    {file = "psycopg2-2.9.1-cp39-cp39-win_amd64.whl", hash = "sha256:bf35a25f1aaa8a3781195595577fcbb59934856ee46b4f252f56ad12b8043bcf"},
+    {file = "psycopg2-2.9.1.tar.gz", hash = "sha256:de5303a6f1d0a7a34b9d40e4d3bef684ccc44a49bbe3eb85e3c0bffb4a131b7c"},
 ]
 ptyprocess = [
     {file = "ptyprocess-0.7.0-py2.py3-none-any.whl", hash = "sha256:4b41f3967fce3af57cc7e94b888626c18bf37a083e3651ca8feeb66d492fef35"},
     {file = "ptyprocess-0.7.0.tar.gz", hash = "sha256:5c5d0a3b48ceee0b48485e0c26037c0acd7d29765ca3fbb5cb3831d347423220"},
 ]
 py = [
-    {file = "py-1.10.0-py2.py3-none-any.whl", hash = "sha256:3b80836aa6d1feeaa108e046da6423ab8f6ceda6468545ae8d02d9d58d18818a"},
-    {file = "py-1.10.0.tar.gz", hash = "sha256:21b81bda15b66ef5e1a777a21c4dcd9c20ad3efd0b3f817e7a809035269e1bd3"},
+    {file = "py-1.11.0-py2.py3-none-any.whl", hash = "sha256:607c53218732647dff4acdfcd50cb62615cedf612e72d1724fb1a0cc6405b378"},
+    {file = "py-1.11.0.tar.gz", hash = "sha256:51c75c4126074b472f746a24399ad32f6053d1b34b68d2fa41e558e6f4a98719"},
 ]
 pyasn1 = [
-    {file = "pyasn1-0.4.8-py2.4.egg", hash = "sha256:fec3e9d8e36808a28efb59b489e4528c10ad0f480e57dcc32b4de5c9d8c9fdf3"},
-    {file = "pyasn1-0.4.8-py2.5.egg", hash = "sha256:0458773cfe65b153891ac249bcf1b5f8f320b7c2ce462151f8fa74de8934becf"},
-    {file = "pyasn1-0.4.8-py2.6.egg", hash = "sha256:5c9414dcfede6e441f7e8f81b43b34e834731003427e5b09e4e00e3172a10f00"},
-    {file = "pyasn1-0.4.8-py2.7.egg", hash = "sha256:6e7545f1a61025a4e58bb336952c5061697da694db1cae97b116e9c46abcf7c8"},
     {file = "pyasn1-0.4.8-py2.py3-none-any.whl", hash = "sha256:39c7e2ec30515947ff4e87fb6f456dfc6e84857d34be479c9d4a4ba4bf46aa5d"},
-    {file = "pyasn1-0.4.8-py3.1.egg", hash = "sha256:78fa6da68ed2727915c4767bb386ab32cdba863caa7dbe473eaae45f9959da86"},
-    {file = "pyasn1-0.4.8-py3.2.egg", hash = "sha256:08c3c53b75eaa48d71cf8c710312316392ed40899cb34710d092e96745a358b7"},
-    {file = "pyasn1-0.4.8-py3.3.egg", hash = "sha256:03840c999ba71680a131cfaee6fab142e1ed9bbd9c693e285cc6aca0d555e576"},
-    {file = "pyasn1-0.4.8-py3.4.egg", hash = "sha256:7ab8a544af125fb704feadb008c99a88805126fb525280b2270bb25cc1d78a12"},
-    {file = "pyasn1-0.4.8-py3.5.egg", hash = "sha256:e89bf84b5437b532b0803ba5c9a5e054d21fec423a89952a74f87fa2c9b7bce2"},
-    {file = "pyasn1-0.4.8-py3.6.egg", hash = "sha256:014c0e9976956a08139dc0712ae195324a75e142284d5f87f1a87ee1b068a359"},
-    {file = "pyasn1-0.4.8-py3.7.egg", hash = "sha256:99fcc3c8d804d1bc6d9a099921e39d827026409a58f2a720dcdb89374ea0c776"},
     {file = "pyasn1-0.4.8.tar.gz", hash = "sha256:aef77c9fb94a3ac588e87841208bdec464471d9871bd5050a287cc9a475cd0ba"},
 ]
 pyasn1-modules = [
     {file = "pyasn1-modules-0.2.8.tar.gz", hash = "sha256:905f84c712230b2c592c19470d3ca8d552de726050d1d1716282a1f6146be65e"},
-    {file = "pyasn1_modules-0.2.8-py2.4.egg", hash = "sha256:0fe1b68d1e486a1ed5473f1302bd991c1611d319bba158e98b106ff86e1d7199"},
-    {file = "pyasn1_modules-0.2.8-py2.5.egg", hash = "sha256:fe0644d9ab041506b62782e92b06b8c68cca799e1a9636ec398675459e031405"},
-    {file = "pyasn1_modules-0.2.8-py2.6.egg", hash = "sha256:a99324196732f53093a84c4369c996713eb8c89d360a496b599fb1a9c47fc3eb"},
-    {file = "pyasn1_modules-0.2.8-py2.7.egg", hash = "sha256:0845a5582f6a02bb3e1bde9ecfc4bfcae6ec3210dd270522fee602365430c3f8"},
     {file = "pyasn1_modules-0.2.8-py2.py3-none-any.whl", hash = "sha256:a50b808ffeb97cb3601dd25981f6b016cbb3d31fbf57a8b8a87428e6158d0c74"},
-    {file = "pyasn1_modules-0.2.8-py3.1.egg", hash = "sha256:f39edd8c4ecaa4556e989147ebf219227e2cd2e8a43c7e7fcb1f1c18c5fd6a3d"},
-    {file = "pyasn1_modules-0.2.8-py3.2.egg", hash = "sha256:b80486a6c77252ea3a3e9b1e360bc9cf28eaac41263d173c032581ad2f20fe45"},
-    {file = "pyasn1_modules-0.2.8-py3.3.egg", hash = "sha256:65cebbaffc913f4fe9e4808735c95ea22d7a7775646ab690518c056784bc21b4"},
-    {file = "pyasn1_modules-0.2.8-py3.4.egg", hash = "sha256:15b7c67fabc7fc240d87fb9aabf999cf82311a6d6fb2c70d00d3d0604878c811"},
-    {file = "pyasn1_modules-0.2.8-py3.5.egg", hash = "sha256:426edb7a5e8879f1ec54a1864f16b882c2837bfd06eee62f2c982315ee2473ed"},
-    {file = "pyasn1_modules-0.2.8-py3.6.egg", hash = "sha256:cbac4bc38d117f2a49aeedec4407d23e8866ea4ac27ff2cf7fb3e5b570df19e0"},
-    {file = "pyasn1_modules-0.2.8-py3.7.egg", hash = "sha256:c29a5e5cc7a3f05926aff34e097e84f8589cd790ce0ed41b67aed6857b26aafd"},
 ]
 pycodestyle = [
     {file = "pycodestyle-2.7.0-py2.py3-none-any.whl", hash = "sha256:514f76d918fcc0b55c6680472f0a37970994e07bbb80725808c17089be302068"},
     {file = "pycodestyle-2.7.0.tar.gz", hash = "sha256:c389c1d06bf7904078ca03399a4816f974a1d590090fecea0c63ec26ebaf1cef"},
 ]
+pycparser = [
+    {file = "pycparser-2.20-py2.py3-none-any.whl", hash = "sha256:7582ad22678f0fcd81102833f60ef8d0e57288b6b5fb00323d101be910e35705"},
+    {file = "pycparser-2.20.tar.gz", hash = "sha256:2d475327684562c3a96cc71adf7dc8c4f0565175cf86b6d7a404ff4c771f15f0"},
+]
 pycryptodome = [
-    {file = "pycryptodome-3.10.1-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:1c5e1ca507de2ad93474be5cfe2bfa76b7cf039a1a32fc196f40935944871a06"},
-    {file = "pycryptodome-3.10.1-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:6260e24d41149268122dd39d4ebd5941e9d107f49463f7e071fd397e29923b0c"},
-    {file = "pycryptodome-3.10.1-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:3f840c49d38986f6e17dbc0673d37947c88bc9d2d9dba1c01b979b36f8447db1"},
-    {file = "pycryptodome-3.10.1-cp27-cp27m-manylinux2010_i686.whl", hash = "sha256:2dea65df54349cdfa43d6b2e8edb83f5f8d6861e5cf7b1fbc3e34c5694c85e27"},
-    {file = "pycryptodome-3.10.1-cp27-cp27m-manylinux2010_x86_64.whl", hash = "sha256:e61e363d9a5d7916f3a4ce984a929514c0df3daf3b1b2eb5e6edbb131ee771cf"},
-    {file = "pycryptodome-3.10.1-cp27-cp27m-manylinux2014_aarch64.whl", hash = "sha256:2603c98ae04aac675fefcf71a6c87dc4bb74a75e9071ae3923bbc91a59f08d35"},
-    {file = "pycryptodome-3.10.1-cp27-cp27m-win32.whl", hash = "sha256:38661348ecb71476037f1e1f553159b80d256c00f6c0b00502acac891f7116d9"},
-    {file = "pycryptodome-3.10.1-cp27-cp27m-win_amd64.whl", hash = "sha256:1723ebee5561628ce96748501cdaa7afaa67329d753933296321f0be55358dce"},
-    {file = "pycryptodome-3.10.1-cp27-cp27mu-manylinux1_i686.whl", hash = "sha256:77997519d8eb8a4adcd9a47b9cec18f9b323e296986528186c0e9a7a15d6a07e"},
-    {file = "pycryptodome-3.10.1-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:99b2f3fc51d308286071d0953f92055504a6ffe829a832a9fc7a04318a7683dd"},
-    {file = "pycryptodome-3.10.1-cp27-cp27mu-manylinux2010_i686.whl", hash = "sha256:e0a4d5933a88a2c98bbe19c0c722f5483dc628d7a38338ac2cb64a7dbd34064b"},
-    {file = "pycryptodome-3.10.1-cp27-cp27mu-manylinux2010_x86_64.whl", hash = "sha256:d3d6958d53ad307df5e8469cc44474a75393a434addf20ecd451f38a72fe29b8"},
-    {file = "pycryptodome-3.10.1-cp27-cp27mu-manylinux2014_aarch64.whl", hash = "sha256:a8eb8b6ea09ec1c2535bf39914377bc8abcab2c7d30fa9225eb4fe412024e427"},
-    {file = "pycryptodome-3.10.1-cp35-abi3-macosx_10_9_x86_64.whl", hash = "sha256:31c1df17b3dc5f39600a4057d7db53ac372f492c955b9b75dd439f5d8b460129"},
-    {file = "pycryptodome-3.10.1-cp35-abi3-manylinux1_i686.whl", hash = "sha256:a3105a0eb63eacf98c2ecb0eb4aa03f77f40fbac2bdde22020bb8a536b226bb8"},
-    {file = "pycryptodome-3.10.1-cp35-abi3-manylinux1_x86_64.whl", hash = "sha256:a92d5c414e8ee1249e850789052608f582416e82422502dc0ac8c577808a9067"},
-    {file = "pycryptodome-3.10.1-cp35-abi3-manylinux2010_i686.whl", hash = "sha256:60386d1d4cfaad299803b45a5bc2089696eaf6cdd56f9fc17479a6f89595cfc8"},
-    {file = "pycryptodome-3.10.1-cp35-abi3-manylinux2010_x86_64.whl", hash = "sha256:501ab36aae360e31d0ec370cf5ce8ace6cb4112060d099b993bc02b36ac83fb6"},
-    {file = "pycryptodome-3.10.1-cp35-abi3-manylinux2014_aarch64.whl", hash = "sha256:fc7489a50323a0df02378bc2fff86eb69d94cc5639914346c736be981c6a02e7"},
-    {file = "pycryptodome-3.10.1-cp35-abi3-win32.whl", hash = "sha256:9b6f711b25e01931f1c61ce0115245a23cdc8b80bf8539ac0363bdcf27d649b6"},
-    {file = "pycryptodome-3.10.1-cp35-abi3-win_amd64.whl", hash = "sha256:7fd519b89585abf57bf47d90166903ec7b43af4fe23c92273ea09e6336af5c07"},
-    {file = "pycryptodome-3.10.1-pp27-pypy_73-macosx_10_9_x86_64.whl", hash = "sha256:09c1555a3fa450e7eaca41ea11cd00afe7c91fef52353488e65663777d8524e0"},
-    {file = "pycryptodome-3.10.1-pp27-pypy_73-manylinux1_x86_64.whl", hash = "sha256:758949ca62690b1540dfb24ad773c6da9cd0e425189e83e39c038bbd52b8e438"},
-    {file = "pycryptodome-3.10.1-pp27-pypy_73-manylinux2010_x86_64.whl", hash = "sha256:e3bf558c6aeb49afa9f0c06cee7fb5947ee5a1ff3bd794b653d39926b49077fa"},
-    {file = "pycryptodome-3.10.1-pp27-pypy_73-win32.whl", hash = "sha256:f977cdf725b20f6b8229b0c87acb98c7717e742ef9f46b113985303ae12a99da"},
-    {file = "pycryptodome-3.10.1-pp36-pypy36_pp73-macosx_10_9_x86_64.whl", hash = "sha256:6d2df5223b12437e644ce0a3be7809471ffa71de44ccd28b02180401982594a6"},
-    {file = "pycryptodome-3.10.1-pp36-pypy36_pp73-manylinux1_x86_64.whl", hash = "sha256:98213ac2b18dc1969a47bc65a79a8fca02a414249d0c8635abb081c7f38c91b6"},
-    {file = "pycryptodome-3.10.1-pp36-pypy36_pp73-manylinux2010_x86_64.whl", hash = "sha256:12222a5edc9ca4a29de15fbd5339099c4c26c56e13c2ceddf0b920794f26165d"},
-    {file = "pycryptodome-3.10.1-pp36-pypy36_pp73-win32.whl", hash = "sha256:6bbf7fee7b7948b29d7e71fcacf48bac0c57fb41332007061a933f2d996f9713"},
-    {file = "pycryptodome-3.10.1.tar.gz", hash = "sha256:3e2e3a06580c5f190df843cdb90ea28d61099cf4924334d5297a995de68e4673"},
+    {file = "pycryptodome-3.11.0-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:ffd0cac13ff41f2d15ed39dc6ba1d2ad88dd2905d656c33d8235852f5d6151fd"},
+    {file = "pycryptodome-3.11.0-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:ead516e03dfe062aefeafe4a29445a6449b0fc43bc8cb30194b2754917a63798"},
+    {file = "pycryptodome-3.11.0-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:4ce6b09547bf2c7cede3a017f79502eaed3e819c13cdb3cb357aea1b004e4cc6"},
+    {file = "pycryptodome-3.11.0-cp27-cp27m-manylinux2010_i686.whl", hash = "sha256:014c758af7fa38cab85b357a496b76f4fc9dda1f731eb28358d66fef7ad4a3e1"},
+    {file = "pycryptodome-3.11.0-cp27-cp27m-manylinux2010_x86_64.whl", hash = "sha256:a843350d08c3d22f6c09c2f17f020d8dcfa59496165d7425a3fba0045543dda7"},
+    {file = "pycryptodome-3.11.0-cp27-cp27m-manylinux2014_aarch64.whl", hash = "sha256:53989477044be41fa4a63da09d5038c2a34b2f4554cfea2e3933b17186ee9e19"},
+    {file = "pycryptodome-3.11.0-cp27-cp27m-win32.whl", hash = "sha256:f9bad2220b80b4ed74f089db012ab5ab5419143a33fad6c8aedcc2a9341eac70"},
+    {file = "pycryptodome-3.11.0-cp27-cp27m-win_amd64.whl", hash = "sha256:3c7ed5b07274535979c730daf5817db5e983ea80b04c22579eee8da4ca3ae4f8"},
+    {file = "pycryptodome-3.11.0-cp27-cp27mu-manylinux1_i686.whl", hash = "sha256:8f3a60926be78422e662b0d0b18351b426ce27657101c8a50bad80300de6a701"},
+    {file = "pycryptodome-3.11.0-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:fce7e22d96030b35345637c563246c24d4513bd3b413e1c40293114837ab8912"},
+    {file = "pycryptodome-3.11.0-cp27-cp27mu-manylinux2010_i686.whl", hash = "sha256:bc3c61ff92efdcc14af4a7b81da71d849c9acee51d8fd8ac9841a7620140d6c6"},
+    {file = "pycryptodome-3.11.0-cp27-cp27mu-manylinux2010_x86_64.whl", hash = "sha256:b33c9b3d1327d821e28e9cc3a6512c14f8b17570ddb4cfb9a52247ed0fcc5d8b"},
+    {file = "pycryptodome-3.11.0-cp27-cp27mu-manylinux2014_aarch64.whl", hash = "sha256:75e78360d1dd6d02eb288fd8275bb4d147d6e3f5337935c096d11dba1fa84748"},
+    {file = "pycryptodome-3.11.0-cp35-abi3-macosx_10_9_x86_64.whl", hash = "sha256:621a90147a5e255fdc2a0fec2d56626b76b5d72ea9e60164c9a5a8976d45b0c9"},
+    {file = "pycryptodome-3.11.0-cp35-abi3-manylinux1_i686.whl", hash = "sha256:0ca7a6b4fc1f9fafe990b95c8cda89099797e2cfbf40e55607f2f2f5a3355dcb"},
+    {file = "pycryptodome-3.11.0-cp35-abi3-manylinux1_x86_64.whl", hash = "sha256:b59bf823cfafde8ef1105d8984f26d1694dff165adb7198b12e3e068d7999b15"},
+    {file = "pycryptodome-3.11.0-cp35-abi3-manylinux2010_i686.whl", hash = "sha256:ce81b9c6aaa0f920e2ab05eb2b9f4ccd102e3016b2f37125593b16a83a4b0cc2"},
+    {file = "pycryptodome-3.11.0-cp35-abi3-manylinux2010_x86_64.whl", hash = "sha256:ae29fcd56152f417bfba50a36a56a7a5f9fb74ff80bab98704cac704de6568ab"},
+    {file = "pycryptodome-3.11.0-cp35-abi3-manylinux2014_aarch64.whl", hash = "sha256:ae31cb874f6f0cedbed457c6374e7e54d7ed45c1a4e11a65a9c80968da90a650"},
+    {file = "pycryptodome-3.11.0-cp35-abi3-win32.whl", hash = "sha256:6db1f9fa1f52226621905f004278ce7bd90c8f5363ffd5d7ab3755363d98549a"},
+    {file = "pycryptodome-3.11.0-cp35-abi3-win_amd64.whl", hash = "sha256:d7e5f6f692421e5219aa3b545eb0cffd832cd589a4b9dcd4a5eb4260e2c0d68a"},
+    {file = "pycryptodome-3.11.0-pp27-pypy_73-macosx_10_9_x86_64.whl", hash = "sha256:da796e9221dda61a0019d01742337eb8a322de8598b678a4344ca0a436380315"},
+    {file = "pycryptodome-3.11.0-pp27-pypy_73-manylinux1_x86_64.whl", hash = "sha256:ed45ef92d21db33685b789de2c015e9d9a18a74760a8df1fc152faee88cdf741"},
+    {file = "pycryptodome-3.11.0-pp27-pypy_73-manylinux2010_x86_64.whl", hash = "sha256:4169ed515742425ff21e4bd3fabbb6994ffb64434472fb72230019bdfa36b939"},
+    {file = "pycryptodome-3.11.0-pp27-pypy_73-win32.whl", hash = "sha256:f19edd42368e9057c39492947bb99570dc927123e210008f2af7cf9b505c6892"},
+    {file = "pycryptodome-3.11.0-pp36-pypy36_pp73-macosx_10_9_x86_64.whl", hash = "sha256:06162fcfed2f9deee8383fd59eaeabc7b7ffc3af50d3fad4000032deb8f700b0"},
+    {file = "pycryptodome-3.11.0-pp36-pypy36_pp73-manylinux1_x86_64.whl", hash = "sha256:6eda8a3157c91ba60b26a07bedd6c44ab8bda6cd79b6b5ea9744ba62c39b7b1e"},
+    {file = "pycryptodome-3.11.0-pp36-pypy36_pp73-manylinux2010_x86_64.whl", hash = "sha256:7ff701fc283412e651eaab4319b3cd4eaa0827e94569cd37ee9075d5c05fe655"},
+    {file = "pycryptodome-3.11.0-pp36-pypy36_pp73-win32.whl", hash = "sha256:2a4bcc8a9977fee0979079cd33a9e9f0d3ddba5660d35ffe874cf84f1dd399d2"},
+    {file = "pycryptodome-3.11.0.tar.gz", hash = "sha256:428096bbf7a77e207f418dfd4d7c284df8ade81d2dc80f010e92753a3e406ad0"},
 ]
 pydocstyle = [
-    {file = "pydocstyle-6.0.0-py3-none-any.whl", hash = "sha256:d4449cf16d7e6709f63192146706933c7a334af7c0f083904799ccb851c50f6d"},
-    {file = "pydocstyle-6.0.0.tar.gz", hash = "sha256:164befb520d851dbcf0e029681b91f4f599c62c5cd8933fd54b1bfbd50e89e1f"},
+    {file = "pydocstyle-6.1.1-py3-none-any.whl", hash = "sha256:6987826d6775056839940041beef5c08cc7e3d71d63149b48e36727f70144dc4"},
+    {file = "pydocstyle-6.1.1.tar.gz", hash = "sha256:1d41b7c459ba0ee6c345f2eb9ae827cab14a7533a88c5c6f7e94923f72df92dc"},
 ]
 pyflakes = [
     {file = "pyflakes-2.3.1-py2.py3-none-any.whl", hash = "sha256:7893783d01b8a89811dd72d7dfd4d84ff098e5eed95cfa8905b22bbffe52efc3"},
     {file = "pyflakes-2.3.1.tar.gz", hash = "sha256:f5bc8ecabc05bb9d291eb5203d6810b49040f6ff446a756326104746cc00c1db"},
 ]
 pygments = [
-    {file = "Pygments-2.8.1-py3-none-any.whl", hash = "sha256:534ef71d539ae97d4c3a4cf7d6f110f214b0e687e92f9cb9d2a3b0d3101289c8"},
-    {file = "Pygments-2.8.1.tar.gz", hash = "sha256:2656e1a6edcdabf4275f9a3640db59fd5de107d88e8663c5d4e9a0fa62f77f94"},
+    {file = "Pygments-2.10.0-py3-none-any.whl", hash = "sha256:b8e67fe6af78f492b3c4b3e2970c0624cbf08beb1e493b2c99b9fa1b67a20380"},
+    {file = "Pygments-2.10.0.tar.gz", hash = "sha256:f398865f7eb6874156579fdf36bc840a03cab64d1cde9e93d68f46a425ec52c6"},
 ]
 pyjwt = [
-    {file = "PyJWT-1.7.1-py2.py3-none-any.whl", hash = "sha256:5c6eca3c2940464d106b99ba83b00c6add741c9becaec087fb7ccdefea71350e"},
-    {file = "PyJWT-1.7.1.tar.gz", hash = "sha256:8d59a976fb773f3e6a39c85636357c4f0e242707394cadadd9814f5cbaa20e96"},
+    {file = "PyJWT-2.3.0-py3-none-any.whl", hash = "sha256:e0c4bb8d9f0af0c7f5b1ec4c5036309617d03d56932877f2f7a0beeb5318322f"},
+    {file = "PyJWT-2.3.0.tar.gz", hash = "sha256:b888b4d56f06f6dcd777210c334e69c737be74755d3e5e9ee3fe67dc18a0ee41"},
 ]
 pyparsing = [
     {file = "pyparsing-2.4.7-py2.py3-none-any.whl", hash = "sha256:ef9d7589ef3c200abe66653d3f1ab1033c3c419ae9b9bdb1240a85b024efc88b"},
     {file = "pyparsing-2.4.7.tar.gz", hash = "sha256:c203ec8783bf771a155b207279b9bccb8dea02d8f0c9e5f8ead507bc3246ecc1"},
 ]
 pytest = [
-    {file = "pytest-6.2.3-py3-none-any.whl", hash = "sha256:6ad9c7bdf517a808242b998ac20063c41532a570d088d77eec1ee12b0b5574bc"},
-    {file = "pytest-6.2.3.tar.gz", hash = "sha256:671238a46e4df0f3498d1c3270e5deb9b32d25134c99b7d75370a68cfbe9b634"},
+    {file = "pytest-6.2.5-py3-none-any.whl", hash = "sha256:7310f8d27bc79ced999e760ca304d69f6ba6c6649c0b60fb0e04a4a77cacc134"},
+    {file = "pytest-6.2.5.tar.gz", hash = "sha256:131b36680866a76e6781d13f101efb86cf674ebb9762eb70d3082b6f29889e89"},
 ]
 pytest-cov = [
-    {file = "pytest-cov-2.11.1.tar.gz", hash = "sha256:359952d9d39b9f822d9d29324483e7ba04a3a17dd7d05aa6beb7ea01e359e5f7"},
-    {file = "pytest_cov-2.11.1-py2.py3-none-any.whl", hash = "sha256:bdb9fdb0b85a7cc825269a4c56b48ccaa5c7e365054b6038772c32ddcdc969da"},
+    {file = "pytest-cov-2.12.1.tar.gz", hash = "sha256:261ceeb8c227b726249b376b8526b600f38667ee314f910353fa318caa01f4d7"},
+    {file = "pytest_cov-2.12.1-py2.py3-none-any.whl", hash = "sha256:261bb9e47e65bd099c89c3edf92972865210c36813f80ede5277dceb77a4a62a"},
 ]
 pytest-django = [
-    {file = "pytest-django-4.2.0.tar.gz", hash = "sha256:80f8875226ec4dc0b205f0578072034563879d98d9b1bec143a80b9045716cb0"},
-    {file = "pytest_django-4.2.0-py3-none-any.whl", hash = "sha256:a51150d8962200250e850c6adcab670779b9c2aa07271471059d1fb92a843fa9"},
+    {file = "pytest-django-4.4.0.tar.gz", hash = "sha256:b5171e3798bf7e3fc5ea7072fe87324db67a4dd9f1192b037fed4cc3c1b7f455"},
+    {file = "pytest_django-4.4.0-py3-none-any.whl", hash = "sha256:65783e78382456528bd9d79a35843adde9e6a47347b20464eb2c885cb0f1f606"},
 ]
 pytest-django-testing-postgresql = [
     {file = "pytest-django-testing-postgresql-0.1.post0.tar.gz", hash = "sha256:78b0c58930084cb4393407b2e5a2a3b8734c627b841ecef7d62d39bbfb8e8a45"},
@@ -3408,49 +4741,61 @@ pytest-sugar = [
     {file = "pytest-sugar-0.9.4.tar.gz", hash = "sha256:b1b2186b0a72aada6859bea2a5764145e3aaa2c1cfbb23c3a19b5f7b697563d3"},
 ]
 python-crontab = [
-    {file = "python-crontab-2.5.1.tar.gz", hash = "sha256:4bbe7e720753a132ca4ca9d4094915f40e9d9dc8a807a4564007651018ce8c31"},
+    {file = "python-crontab-2.6.0.tar.gz", hash = "sha256:1e35ed7a3cdc3100545b43e196d34754e6551e7f95e4caebbe0e1c0ca41c2f1b"},
 ]
 python-dateutil = [
-    {file = "python-dateutil-2.8.1.tar.gz", hash = "sha256:73ebfe9dbf22e832286dafa60473e4cd239f8592f699aa5adaf10050e6e1823c"},
-    {file = "python_dateutil-2.8.1-py2.py3-none-any.whl", hash = "sha256:75bb3f31ea686f1197762692a9ee6a7550b59fc6ca3a1f4b5d7e32fb98e2da2a"},
+    {file = "python-dateutil-2.8.2.tar.gz", hash = "sha256:0123cacc1627ae19ddf3c27a5de5bd67ee4586fbdd6440d9748f8abb483d3e86"},
+    {file = "python_dateutil-2.8.2-py2.py3-none-any.whl", hash = "sha256:961d03dc3453ebbc59dbdea9e4e11c5651520a876d0f4db161e8674aae935da9"},
+]
+python-gnupg = [
+    {file = "python-gnupg-0.4.7.tar.gz", hash = "sha256:2061f56b1942c29b92727bf9aecbd3cea3893acc9cccbdc7eb4604285efe4ac7"},
+    {file = "python_gnupg-0.4.7-py2.py3-none-any.whl", hash = "sha256:3ff5b1bf5e397de6e1fe41a7c0f403dad4e242ac92b345f440eaecfb72a7ebae"},
 ]
 python-ldap = [
     {file = "python-ldap-3.3.1.tar.gz", hash = "sha256:4711cacf013e298754abd70058ccc995758177fb425f1c2d30e71adfc1d00aa5"},
 ]
+python3-openid = [
+    {file = "python3-openid-3.2.0.tar.gz", hash = "sha256:33fbf6928f401e0b790151ed2b5290b02545e8775f982485205a066f874aaeaf"},
+    {file = "python3_openid-3.2.0-py3-none-any.whl", hash = "sha256:6626f771e0417486701e0b4daff762e7212e820ca5b29fcc0d05f6f8736dfa6b"},
+]
 pytz = [
-    {file = "pytz-2021.1-py2.py3-none-any.whl", hash = "sha256:eb10ce3e7736052ed3623d49975ce333bcd712c7bb19a58b9e2089d4057d0798"},
-    {file = "pytz-2021.1.tar.gz", hash = "sha256:83a4a90894bf38e243cf052c8b58f381bfe9a7a483f6a9cab140bc7f702ac4da"},
+    {file = "pytz-2021.3-py2.py3-none-any.whl", hash = "sha256:3672058bc3453457b622aab7a1c3bfd5ab0bdae451512f6cf25f64ed37f5b87c"},
+    {file = "pytz-2021.3.tar.gz", hash = "sha256:acad2d8b20a1af07d4e4c9d2e9285c5ed9104354062f275f3fcd88dcef4f1326"},
 ]
 pyyaml = [
-    {file = "PyYAML-5.4.1-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:3b2b1824fe7112845700f815ff6a489360226a5609b96ec2190a45e62a9fc922"},
-    {file = "PyYAML-5.4.1-cp27-cp27m-win32.whl", hash = "sha256:129def1b7c1bf22faffd67b8f3724645203b79d8f4cc81f674654d9902cb4393"},
-    {file = "PyYAML-5.4.1-cp27-cp27m-win_amd64.whl", hash = "sha256:4465124ef1b18d9ace298060f4eccc64b0850899ac4ac53294547536533800c8"},
-    {file = "PyYAML-5.4.1-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:bb4191dfc9306777bc594117aee052446b3fa88737cd13b7188d0e7aa8162185"},
-    {file = "PyYAML-5.4.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:6c78645d400265a062508ae399b60b8c167bf003db364ecb26dcab2bda048253"},
-    {file = "PyYAML-5.4.1-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:4e0583d24c881e14342eaf4ec5fbc97f934b999a6828693a99157fde912540cc"},
-    {file = "PyYAML-5.4.1-cp36-cp36m-manylinux2014_aarch64.whl", hash = "sha256:72a01f726a9c7851ca9bfad6fd09ca4e090a023c00945ea05ba1638c09dc3347"},
-    {file = "PyYAML-5.4.1-cp36-cp36m-manylinux2014_s390x.whl", hash = "sha256:895f61ef02e8fed38159bb70f7e100e00f471eae2bc838cd0f4ebb21e28f8541"},
-    {file = "PyYAML-5.4.1-cp36-cp36m-win32.whl", hash = "sha256:3bd0e463264cf257d1ffd2e40223b197271046d09dadf73a0fe82b9c1fc385a5"},
-    {file = "PyYAML-5.4.1-cp36-cp36m-win_amd64.whl", hash = "sha256:e4fac90784481d221a8e4b1162afa7c47ed953be40d31ab4629ae917510051df"},
-    {file = "PyYAML-5.4.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:5accb17103e43963b80e6f837831f38d314a0495500067cb25afab2e8d7a4018"},
-    {file = "PyYAML-5.4.1-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:e1d4970ea66be07ae37a3c2e48b5ec63f7ba6804bdddfdbd3cfd954d25a82e63"},
-    {file = "PyYAML-5.4.1-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:cb333c16912324fd5f769fff6bc5de372e9e7a202247b48870bc251ed40239aa"},
-    {file = "PyYAML-5.4.1-cp37-cp37m-manylinux2014_s390x.whl", hash = "sha256:fe69978f3f768926cfa37b867e3843918e012cf83f680806599ddce33c2c68b0"},
-    {file = "PyYAML-5.4.1-cp37-cp37m-win32.whl", hash = "sha256:dd5de0646207f053eb0d6c74ae45ba98c3395a571a2891858e87df7c9b9bd51b"},
-    {file = "PyYAML-5.4.1-cp37-cp37m-win_amd64.whl", hash = "sha256:08682f6b72c722394747bddaf0aa62277e02557c0fd1c42cb853016a38f8dedf"},
-    {file = "PyYAML-5.4.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:d2d9808ea7b4af864f35ea216be506ecec180628aced0704e34aca0b040ffe46"},
-    {file = "PyYAML-5.4.1-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:8c1be557ee92a20f184922c7b6424e8ab6691788e6d86137c5d93c1a6ec1b8fb"},
-    {file = "PyYAML-5.4.1-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:fd7f6999a8070df521b6384004ef42833b9bd62cfee11a09bda1079b4b704247"},
-    {file = "PyYAML-5.4.1-cp38-cp38-manylinux2014_s390x.whl", hash = "sha256:bfb51918d4ff3d77c1c856a9699f8492c612cde32fd3bcd344af9be34999bfdc"},
-    {file = "PyYAML-5.4.1-cp38-cp38-win32.whl", hash = "sha256:fa5ae20527d8e831e8230cbffd9f8fe952815b2b7dae6ffec25318803a7528fc"},
-    {file = "PyYAML-5.4.1-cp38-cp38-win_amd64.whl", hash = "sha256:0f5f5786c0e09baddcd8b4b45f20a7b5d61a7e7e99846e3c799b05c7c53fa696"},
-    {file = "PyYAML-5.4.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:294db365efa064d00b8d1ef65d8ea2c3426ac366c0c4368d930bf1c5fb497f77"},
-    {file = "PyYAML-5.4.1-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:74c1485f7707cf707a7aef42ef6322b8f97921bd89be2ab6317fd782c2d53183"},
-    {file = "PyYAML-5.4.1-cp39-cp39-manylinux2014_aarch64.whl", hash = "sha256:d483ad4e639292c90170eb6f7783ad19490e7a8defb3e46f97dfe4bacae89122"},
-    {file = "PyYAML-5.4.1-cp39-cp39-manylinux2014_s390x.whl", hash = "sha256:fdc842473cd33f45ff6bce46aea678a54e3d21f1b61a7750ce3c498eedfe25d6"},
-    {file = "PyYAML-5.4.1-cp39-cp39-win32.whl", hash = "sha256:49d4cdd9065b9b6e206d0595fee27a96b5dd22618e7520c33204a4a3239d5b10"},
-    {file = "PyYAML-5.4.1-cp39-cp39-win_amd64.whl", hash = "sha256:c20cfa2d49991c8b4147af39859b167664f2ad4561704ee74c1de03318e898db"},
-    {file = "PyYAML-5.4.1.tar.gz", hash = "sha256:607774cbba28732bfa802b54baa7484215f530991055bb562efbed5b2f20a45e"},
+    {file = "PyYAML-6.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:d4db7c7aef085872ef65a8fd7d6d09a14ae91f691dec3e87ee5ee0539d516f53"},
+    {file = "PyYAML-6.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:9df7ed3b3d2e0ecfe09e14741b857df43adb5a3ddadc919a2d94fbdf78fea53c"},
+    {file = "PyYAML-6.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:77f396e6ef4c73fdc33a9157446466f1cff553d979bd00ecb64385760c6babdc"},
+    {file = "PyYAML-6.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:a80a78046a72361de73f8f395f1f1e49f956c6be882eed58505a15f3e430962b"},
+    {file = "PyYAML-6.0-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:f84fbc98b019fef2ee9a1cb3ce93e3187a6df0b2538a651bfb890254ba9f90b5"},
+    {file = "PyYAML-6.0-cp310-cp310-win32.whl", hash = "sha256:2cd5df3de48857ed0544b34e2d40e9fac445930039f3cfe4bcc592a1f836d513"},
+    {file = "PyYAML-6.0-cp310-cp310-win_amd64.whl", hash = "sha256:daf496c58a8c52083df09b80c860005194014c3698698d1a57cbcfa182142a3a"},
+    {file = "PyYAML-6.0-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:897b80890765f037df3403d22bab41627ca8811ae55e9a722fd0392850ec4d86"},
+    {file = "PyYAML-6.0-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:50602afada6d6cbfad699b0c7bb50d5ccffa7e46a3d738092afddc1f9758427f"},
+    {file = "PyYAML-6.0-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:48c346915c114f5fdb3ead70312bd042a953a8ce5c7106d5bfb1a5254e47da92"},
+    {file = "PyYAML-6.0-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:98c4d36e99714e55cfbaaee6dd5badbc9a1ec339ebfc3b1f52e293aee6bb71a4"},
+    {file = "PyYAML-6.0-cp36-cp36m-win32.whl", hash = "sha256:0283c35a6a9fbf047493e3a0ce8d79ef5030852c51e9d911a27badfde0605293"},
+    {file = "PyYAML-6.0-cp36-cp36m-win_amd64.whl", hash = "sha256:07751360502caac1c067a8132d150cf3d61339af5691fe9e87803040dbc5db57"},
+    {file = "PyYAML-6.0-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:819b3830a1543db06c4d4b865e70ded25be52a2e0631ccd2f6a47a2822f2fd7c"},
+    {file = "PyYAML-6.0-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:473f9edb243cb1935ab5a084eb238d842fb8f404ed2193a915d1784b5a6b5fc0"},
+    {file = "PyYAML-6.0-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:0ce82d761c532fe4ec3f87fc45688bdd3a4c1dc5e0b4a19814b9009a29baefd4"},
+    {file = "PyYAML-6.0-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:231710d57adfd809ef5d34183b8ed1eeae3f76459c18fb4a0b373ad56bedcdd9"},
+    {file = "PyYAML-6.0-cp37-cp37m-win32.whl", hash = "sha256:c5687b8d43cf58545ade1fe3e055f70eac7a5a1a0bf42824308d868289a95737"},
+    {file = "PyYAML-6.0-cp37-cp37m-win_amd64.whl", hash = "sha256:d15a181d1ecd0d4270dc32edb46f7cb7733c7c508857278d3d378d14d606db2d"},
+    {file = "PyYAML-6.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:0b4624f379dab24d3725ffde76559cff63d9ec94e1736b556dacdfebe5ab6d4b"},
+    {file = "PyYAML-6.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:213c60cd50106436cc818accf5baa1aba61c0189ff610f64f4a3e8c6726218ba"},
+    {file = "PyYAML-6.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:9fa600030013c4de8165339db93d182b9431076eb98eb40ee068700c9c813e34"},
+    {file = "PyYAML-6.0-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:277a0ef2981ca40581a47093e9e2d13b3f1fbbeffae064c1d21bfceba2030287"},
+    {file = "PyYAML-6.0-cp38-cp38-win32.whl", hash = "sha256:d4eccecf9adf6fbcc6861a38015c2a64f38b9d94838ac1810a9023a0609e1b78"},
+    {file = "PyYAML-6.0-cp38-cp38-win_amd64.whl", hash = "sha256:1e4747bc279b4f613a09eb64bba2ba602d8a6664c6ce6396a4d0cd413a50ce07"},
+    {file = "PyYAML-6.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:055d937d65826939cb044fc8c9b08889e8c743fdc6a32b33e2390f66013e449b"},
+    {file = "PyYAML-6.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:e61ceaab6f49fb8bdfaa0f92c4b57bcfbea54c09277b1b4f7ac376bfb7a7c174"},
+    {file = "PyYAML-6.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d67d839ede4ed1b28a4e8909735fc992a923cdb84e618544973d7dfc71540803"},
+    {file = "PyYAML-6.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:cba8c411ef271aa037d7357a2bc8f9ee8b58b9965831d9e51baf703280dc73d3"},
+    {file = "PyYAML-6.0-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:40527857252b61eacd1d9af500c3337ba8deb8fc298940291486c465c8b46ec0"},
+    {file = "PyYAML-6.0-cp39-cp39-win32.whl", hash = "sha256:b5b9eccad747aabaaffbc6064800670f0c297e52c12754eb1d976c57e4f74dcb"},
+    {file = "PyYAML-6.0-cp39-cp39-win_amd64.whl", hash = "sha256:b3d267842bf12586ba6c734f89d1f5b871df0273157918b0ccefa29deb05c21c"},
+    {file = "PyYAML-6.0.tar.gz", hash = "sha256:68fb519c14306fec9720a2a5b45bc9f0c8d1b9c72adf45c37baedfcd949c35a2"},
 ]
 qrcode = [
     {file = "qrcode-6.1-py2.py3-none-any.whl", hash = "sha256:3996ee560fc39532910603704c82980ff6d4d5d629f9c3f25f34174ce8606cf5"},
@@ -3461,126 +4806,132 @@ redis = [
     {file = "redis-3.5.3.tar.gz", hash = "sha256:0e7e0cfca8660dea8b7d5cd8c4f6c5e29e11f31158c0b0ae91a397f00e5a05a2"},
 ]
 regex = [
-    {file = "regex-2021.4.4-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:619d71c59a78b84d7f18891fe914446d07edd48dc8328c8e149cbe0929b4e000"},
-    {file = "regex-2021.4.4-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:47bf5bf60cf04d72bf6055ae5927a0bd9016096bf3d742fa50d9bf9f45aa0711"},
-    {file = "regex-2021.4.4-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:281d2fd05555079448537fe108d79eb031b403dac622621c78944c235f3fcf11"},
-    {file = "regex-2021.4.4-cp36-cp36m-manylinux2010_i686.whl", hash = "sha256:bd28bc2e3a772acbb07787c6308e00d9626ff89e3bfcdebe87fa5afbfdedf968"},
-    {file = "regex-2021.4.4-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:7c2a1af393fcc09e898beba5dd59196edaa3116191cc7257f9224beaed3e1aa0"},
-    {file = "regex-2021.4.4-cp36-cp36m-manylinux2014_aarch64.whl", hash = "sha256:c38c71df845e2aabb7fb0b920d11a1b5ac8526005e533a8920aea97efb8ec6a4"},
-    {file = "regex-2021.4.4-cp36-cp36m-manylinux2014_i686.whl", hash = "sha256:96fcd1888ab4d03adfc9303a7b3c0bd78c5412b2bfbe76db5b56d9eae004907a"},
-    {file = "regex-2021.4.4-cp36-cp36m-manylinux2014_x86_64.whl", hash = "sha256:ade17eb5d643b7fead300a1641e9f45401c98eee23763e9ed66a43f92f20b4a7"},
-    {file = "regex-2021.4.4-cp36-cp36m-win32.whl", hash = "sha256:e8e5b509d5c2ff12f8418006d5a90e9436766133b564db0abaec92fd27fcee29"},
-    {file = "regex-2021.4.4-cp36-cp36m-win_amd64.whl", hash = "sha256:11d773d75fa650cd36f68d7ca936e3c7afaae41b863b8c387a22aaa78d3c5c79"},
-    {file = "regex-2021.4.4-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:d3029c340cfbb3ac0a71798100ccc13b97dddf373a4ae56b6a72cf70dfd53bc8"},
-    {file = "regex-2021.4.4-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:18c071c3eb09c30a264879f0d310d37fe5d3a3111662438889ae2eb6fc570c31"},
-    {file = "regex-2021.4.4-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:4c557a7b470908b1712fe27fb1ef20772b78079808c87d20a90d051660b1d69a"},
-    {file = "regex-2021.4.4-cp37-cp37m-manylinux2010_i686.whl", hash = "sha256:01afaf2ec48e196ba91b37451aa353cb7eda77efe518e481707e0515025f0cd5"},
-    {file = "regex-2021.4.4-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:3a9cd17e6e5c7eb328517969e0cb0c3d31fd329298dd0c04af99ebf42e904f82"},
-    {file = "regex-2021.4.4-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:90f11ff637fe8798933fb29f5ae1148c978cccb0452005bf4c69e13db951e765"},
-    {file = "regex-2021.4.4-cp37-cp37m-manylinux2014_i686.whl", hash = "sha256:919859aa909429fb5aa9cf8807f6045592c85ef56fdd30a9a3747e513db2536e"},
-    {file = "regex-2021.4.4-cp37-cp37m-manylinux2014_x86_64.whl", hash = "sha256:339456e7d8c06dd36a22e451d58ef72cef293112b559010db3d054d5560ef439"},
-    {file = "regex-2021.4.4-cp37-cp37m-win32.whl", hash = "sha256:67bdb9702427ceddc6ef3dc382455e90f785af4c13d495f9626861763ee13f9d"},
-    {file = "regex-2021.4.4-cp37-cp37m-win_amd64.whl", hash = "sha256:32e65442138b7b76dd8173ffa2cf67356b7bc1768851dded39a7a13bf9223da3"},
-    {file = "regex-2021.4.4-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:1e1c20e29358165242928c2de1482fb2cf4ea54a6a6dea2bd7a0e0d8ee321500"},
-    {file = "regex-2021.4.4-cp38-cp38-manylinux1_i686.whl", hash = "sha256:314d66636c494ed9c148a42731b3834496cc9a2c4251b1661e40936814542b14"},
-    {file = "regex-2021.4.4-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:6d1b01031dedf2503631d0903cb563743f397ccaf6607a5e3b19a3d76fc10480"},
-    {file = "regex-2021.4.4-cp38-cp38-manylinux2010_i686.whl", hash = "sha256:741a9647fcf2e45f3a1cf0e24f5e17febf3efe8d4ba1281dcc3aa0459ef424dc"},
-    {file = "regex-2021.4.4-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:4c46e22a0933dd783467cf32b3516299fb98cfebd895817d685130cc50cd1093"},
-    {file = "regex-2021.4.4-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:e512d8ef5ad7b898cdb2d8ee1cb09a8339e4f8be706d27eaa180c2f177248a10"},
-    {file = "regex-2021.4.4-cp38-cp38-manylinux2014_i686.whl", hash = "sha256:980d7be47c84979d9136328d882f67ec5e50008681d94ecc8afa8a65ed1f4a6f"},
-    {file = "regex-2021.4.4-cp38-cp38-manylinux2014_x86_64.whl", hash = "sha256:ce15b6d103daff8e9fee13cf7f0add05245a05d866e73926c358e871221eae87"},
-    {file = "regex-2021.4.4-cp38-cp38-win32.whl", hash = "sha256:a91aa8619b23b79bcbeb37abe286f2f408d2f2d6f29a17237afda55bb54e7aac"},
-    {file = "regex-2021.4.4-cp38-cp38-win_amd64.whl", hash = "sha256:c0502c0fadef0d23b128605d69b58edb2c681c25d44574fc673b0e52dce71ee2"},
-    {file = "regex-2021.4.4-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:598585c9f0af8374c28edd609eb291b5726d7cbce16be6a8b95aa074d252ee17"},
-    {file = "regex-2021.4.4-cp39-cp39-manylinux1_i686.whl", hash = "sha256:ee54ff27bf0afaf4c3b3a62bcd016c12c3fdb4ec4f413391a90bd38bc3624605"},
-    {file = "regex-2021.4.4-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:7d9884d86dd4dd489e981d94a65cd30d6f07203d90e98f6f657f05170f6324c9"},
-    {file = "regex-2021.4.4-cp39-cp39-manylinux2010_i686.whl", hash = "sha256:bf5824bfac591ddb2c1f0a5f4ab72da28994548c708d2191e3b87dd207eb3ad7"},
-    {file = "regex-2021.4.4-cp39-cp39-manylinux2010_x86_64.whl", hash = "sha256:563085e55b0d4fb8f746f6a335893bda5c2cef43b2f0258fe1020ab1dd874df8"},
-    {file = "regex-2021.4.4-cp39-cp39-manylinux2014_aarch64.whl", hash = "sha256:b9c3db21af35e3b3c05764461b262d6f05bbca08a71a7849fd79d47ba7bc33ed"},
-    {file = "regex-2021.4.4-cp39-cp39-manylinux2014_i686.whl", hash = "sha256:3916d08be28a1149fb97f7728fca1f7c15d309a9f9682d89d79db75d5e52091c"},
-    {file = "regex-2021.4.4-cp39-cp39-manylinux2014_x86_64.whl", hash = "sha256:fd45ff9293d9274c5008a2054ecef86a9bfe819a67c7be1afb65e69b405b3042"},
-    {file = "regex-2021.4.4-cp39-cp39-win32.whl", hash = "sha256:fa4537fb4a98fe8fde99626e4681cc644bdcf2a795038533f9f711513a862ae6"},
-    {file = "regex-2021.4.4-cp39-cp39-win_amd64.whl", hash = "sha256:97f29f57d5b84e73fbaf99ab3e26134e6687348e95ef6b48cfd2c06807005a07"},
-    {file = "regex-2021.4.4.tar.gz", hash = "sha256:52ba3d3f9b942c49d7e4bc105bb28551c44065f139a65062ab7912bef10c9afb"},
+    {file = "regex-2021.11.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:897c539f0f3b2c3a715be651322bef2167de1cdc276b3f370ae81a3bda62df71"},
+    {file = "regex-2021.11.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:886f459db10c0f9d17c87d6594e77be915f18d343ee138e68d259eb385f044a8"},
+    {file = "regex-2021.11.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:075b0fdbaea81afcac5a39a0d1bb91de887dd0d93bf692a5dd69c430e7fc58cb"},
+    {file = "regex-2021.11.2-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:c6238d30dcff141de076344cf7f52468de61729c2f70d776fce12f55fe8df790"},
+    {file = "regex-2021.11.2-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:7fab29411d75c2eb48070020a40f80255936d7c31357b086e5931c107d48306e"},
+    {file = "regex-2021.11.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f0148988af0182a0a4e5020e7c168014f2c55a16d11179610f7883dd48ac0ebe"},
+    {file = "regex-2021.11.2-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:be30cd315db0168063a1755fa20a31119da91afa51da2907553493516e165640"},
+    {file = "regex-2021.11.2-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:e9cec3a62d146e8e122d159ab93ac32c988e2ec0dcb1e18e9e53ff2da4fbd30c"},
+    {file = "regex-2021.11.2-cp310-cp310-win32.whl", hash = "sha256:41c66bd6750237a8ed23028a6c9173dc0c92dc24c473e771d3bfb9ee817700c3"},
+    {file = "regex-2021.11.2-cp310-cp310-win_amd64.whl", hash = "sha256:0075fe4e2c2720a685fef0f863edd67740ff78c342cf20b2a79bc19388edf5db"},
+    {file = "regex-2021.11.2-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:0ed3465acf8c7c10aa2e0f3d9671da410ead63b38a77283ef464cbb64275df58"},
+    {file = "regex-2021.11.2-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ab1fea8832976ad0bebb11f652b692c328043057d35e9ebc78ab0a7a30cf9a70"},
+    {file = "regex-2021.11.2-cp36-cp36m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:cb1e44d860345ab5d4f533b6c37565a22f403277f44c4d2d5e06c325da959883"},
+    {file = "regex-2021.11.2-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:9486ebda015913909bc28763c6b92fcc3b5e5a67dee4674bceed112109f5dfb8"},
+    {file = "regex-2021.11.2-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:20605bfad484e1341b2cbfea0708e4b211d233716604846baa54b94821f487cb"},
+    {file = "regex-2021.11.2-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f20f9f430c33597887ba9bd76635476928e76cad2981643ca8be277b8e97aa96"},
+    {file = "regex-2021.11.2-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:1d85ca137756d62c8138c971453cafe64741adad1f6a7e63a22a5a8abdbd19fa"},
+    {file = "regex-2021.11.2-cp36-cp36m-win32.whl", hash = "sha256:af23b9ca9a874ef0ec20e44467b8edd556c37b0f46f93abfa93752ea7c0e8d1e"},
+    {file = "regex-2021.11.2-cp36-cp36m-win_amd64.whl", hash = "sha256:070336382ca92c16c45b4066c4ba9fa83fb0bd13d5553a82e07d344df8d58a84"},
+    {file = "regex-2021.11.2-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:ef4e53e2fdc997d91f5b682f81f7dc9661db9a437acce28745d765d251902d85"},
+    {file = "regex-2021.11.2-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:35ed5714467fc606551db26f80ee5d6aa1f01185586a7bccd96f179c4b974a11"},
+    {file = "regex-2021.11.2-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:7ee36d5113b6506b97f45f2e8447cb9af146e60e3f527d93013d19f6d0405f3b"},
+    {file = "regex-2021.11.2-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:4fba661a4966adbd2c3c08d3caad6822ecb6878f5456588e2475ae23a6e47929"},
+    {file = "regex-2021.11.2-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:77f9d16f7970791f17ecce7e7f101548314ed1ee2583d4268601f30af3170856"},
+    {file = "regex-2021.11.2-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f6a28e87ba69f3a4f30d775b179aac55be1ce59f55799328a0d9b6df8f16b39d"},
+    {file = "regex-2021.11.2-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:9267e4fba27e6dd1008c4f2983cc548c98b4be4444e3e342db11296c0f45512f"},
+    {file = "regex-2021.11.2-cp37-cp37m-win32.whl", hash = "sha256:d4bfe3bc3976ccaeb4ae32f51e631964e2f0e85b2b752721b7a02de5ce3b7f27"},
+    {file = "regex-2021.11.2-cp37-cp37m-win_amd64.whl", hash = "sha256:2bb7cae741de1aa03e3dd3a7d98c304871eb155921ca1f0d7cc11f5aade913fd"},
+    {file = "regex-2021.11.2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:23f93e74409c210de4de270d4bf88fb8ab736a7400f74210df63a93728cf70d6"},
+    {file = "regex-2021.11.2-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:d8ee91e1c295beb5c132ebd78616814de26fedba6aa8687ea460c7f5eb289b72"},
+    {file = "regex-2021.11.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2e3ff69ab203b54ce5c480c3ccbe959394ea5beef6bd5ad1785457df7acea92e"},
+    {file = "regex-2021.11.2-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:e3c00cb5c71da655e1e5161481455479b613d500dd1bd252aa01df4f037c641f"},
+    {file = "regex-2021.11.2-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:4abf35e16f4b639daaf05a2602c1b1d47370e01babf9821306aa138924e3fe92"},
+    {file = "regex-2021.11.2-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bb11c982a849dc22782210b01d0c1b98eb3696ce655d58a54180774e4880ac66"},
+    {file = "regex-2021.11.2-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:07e3755e0f070bc31567dfe447a02011bfa8444239b3e9e5cca6773a22133839"},
+    {file = "regex-2021.11.2-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:0621c90f28d17260b41838b22c81a79ff436141b322960eb49c7b3f91d1cbab6"},
+    {file = "regex-2021.11.2-cp38-cp38-win32.whl", hash = "sha256:8fbe1768feafd3d0156556677b8ff234c7bf94a8110e906b2d73506f577a3269"},
+    {file = "regex-2021.11.2-cp38-cp38-win_amd64.whl", hash = "sha256:f9ee98d658a146cb6507be720a0ce1b44f2abef8fb43c2859791d91aace17cd5"},
+    {file = "regex-2021.11.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:b3794cea825f101fe0df9af8a00f9fad8e119c91e39a28636b95ee2b45b6c2e5"},
+    {file = "regex-2021.11.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:3576e173e7b4f88f683b4de7db0c2af1b209bb48b2bf1c827a6f3564fad59a97"},
+    {file = "regex-2021.11.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:48b4f4810117a9072a5aa70f7fea5f86fa9efbe9a798312e0a05044bd707cc33"},
+    {file = "regex-2021.11.2-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:f5930d334c2f607711d54761956aedf8137f83f1b764b9640be21d25a976f3a4"},
+    {file = "regex-2021.11.2-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:956187ff49db7014ceb31e88fcacf4cf63371e6e44d209cf8816cd4a2d61e11a"},
+    {file = "regex-2021.11.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:17e095f7f96a4b9f24b93c2c915f31a5201a6316618d919b0593afb070a5270e"},
+    {file = "regex-2021.11.2-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a56735c35a3704603d9d7b243ee06139f0837bcac2171d9ba1d638ce1df0742a"},
+    {file = "regex-2021.11.2-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:adf35d88d9cffc202e6046e4c32e1e11a1d0238b2fcf095c94f109e510ececea"},
+    {file = "regex-2021.11.2-cp39-cp39-win32.whl", hash = "sha256:30fe317332de0e50195665bc61a27d46e903d682f94042c36b3f88cb84bd7958"},
+    {file = "regex-2021.11.2-cp39-cp39-win_amd64.whl", hash = "sha256:85289c25f658e3260b00178757c87f033f3d4b3e40aa4abdd4dc875ff11a94fb"},
+    {file = "regex-2021.11.2.tar.gz", hash = "sha256:5e85dcfc5d0f374955015ae12c08365b565c6f1eaf36dd182476a4d8e5a1cdb7"},
 ]
 requests = [
-    {file = "requests-2.25.1-py2.py3-none-any.whl", hash = "sha256:c210084e36a42ae6b9219e00e48287def368a26d03a048ddad7bfee44f75871e"},
-    {file = "requests-2.25.1.tar.gz", hash = "sha256:27973dd4a904a4f13b263a19c866c13b92a39ed1c964655f025f3f8d3d75b804"},
+    {file = "requests-2.26.0-py2.py3-none-any.whl", hash = "sha256:6c1246513ecd5ecd4528a0906f910e8f0f9c6b8ec72030dc9fd154dc1a6efd24"},
+    {file = "requests-2.26.0.tar.gz", hash = "sha256:b8aa58f8cf793ffd8782d3d8cb19e66ef36f7aba4353eec859e74678b01b07a7"},
+]
+requests-oauthlib = [
+    {file = "requests-oauthlib-1.3.0.tar.gz", hash = "sha256:b4261601a71fd721a8bd6d7aa1cc1d6a8a93b4a9f5e96626f8e4d91e8beeaa6a"},
+    {file = "requests_oauthlib-1.3.0-py2.py3-none-any.whl", hash = "sha256:7f71572defaecd16372f9006f33c2ec8c077c3cfa6f5911a9a90202beb513f3d"},
 ]
 restructuredtext-lint = [
     {file = "restructuredtext_lint-1.3.2.tar.gz", hash = "sha256:d3b10a1fe2ecac537e51ae6d151b223b78de9fafdd50e5eb6b08c243df173c80"},
 ]
 "ruamel.yaml" = [
-    {file = "ruamel.yaml-0.17.4-py3-none-any.whl", hash = "sha256:ac79fb25f5476e8e9ed1c53b8a2286d2c3f5dde49eb37dbcee5c7eb6a8415a22"},
-    {file = "ruamel.yaml-0.17.4.tar.gz", hash = "sha256:44bc6b54fddd45e4bc0619059196679f9e8b79c027f4131bb072e6a22f4d5e28"},
+    {file = "ruamel.yaml-0.17.17-py3-none-any.whl", hash = "sha256:9af3ec5d7f8065582f3aa841305465025d0afd26c5fb54e15b964e11838fc74f"},
+    {file = "ruamel.yaml-0.17.17.tar.gz", hash = "sha256:9751de4cbb57d4bfbf8fc394e125ed4a2f170fbff3dc3d78abf50be85924f8be"},
 ]
 "ruamel.yaml.clib" = [
-    {file = "ruamel.yaml.clib-0.2.2-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:28116f204103cb3a108dfd37668f20abe6e3cafd0d3fd40dba126c732457b3cc"},
-    {file = "ruamel.yaml.clib-0.2.2-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:daf21aa33ee9b351f66deed30a3d450ab55c14242cfdfcd377798e2c0d25c9f1"},
-    {file = "ruamel.yaml.clib-0.2.2-cp27-cp27m-win32.whl", hash = "sha256:30dca9bbcbb1cc858717438218d11eafb78666759e5094dd767468c0d577a7e7"},
-    {file = "ruamel.yaml.clib-0.2.2-cp27-cp27m-win_amd64.whl", hash = "sha256:f6061a31880c1ed6b6ce341215336e2f3d0c1deccd84957b6fa8ca474b41e89f"},
-    {file = "ruamel.yaml.clib-0.2.2-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:73b3d43e04cc4b228fa6fa5d796409ece6fcb53a6c270eb2048109cbcbc3b9c2"},
-    {file = "ruamel.yaml.clib-0.2.2-cp35-cp35m-macosx_10_6_intel.whl", hash = "sha256:53b9dd1abd70e257a6e32f934ebc482dac5edb8c93e23deb663eac724c30b026"},
-    {file = "ruamel.yaml.clib-0.2.2-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:839dd72545ef7ba78fd2aa1a5dd07b33696adf3e68fae7f31327161c1093001b"},
-    {file = "ruamel.yaml.clib-0.2.2-cp35-cp35m-manylinux2014_aarch64.whl", hash = "sha256:1236df55e0f73cd138c0eca074ee086136c3f16a97c2ac719032c050f7e0622f"},
-    {file = "ruamel.yaml.clib-0.2.2-cp35-cp35m-win32.whl", hash = "sha256:b1e981fe1aff1fd11627f531524826a4dcc1f26c726235a52fcb62ded27d150f"},
-    {file = "ruamel.yaml.clib-0.2.2-cp35-cp35m-win_amd64.whl", hash = "sha256:4e52c96ca66de04be42ea2278012a2342d89f5e82b4512fb6fb7134e377e2e62"},
-    {file = "ruamel.yaml.clib-0.2.2-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:a873e4d4954f865dcb60bdc4914af7eaae48fb56b60ed6daa1d6251c72f5337c"},
-    {file = "ruamel.yaml.clib-0.2.2-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:ab845f1f51f7eb750a78937be9f79baea4a42c7960f5a94dde34e69f3cce1988"},
-    {file = "ruamel.yaml.clib-0.2.2-cp36-cp36m-manylinux2014_aarch64.whl", hash = "sha256:2fd336a5c6415c82e2deb40d08c222087febe0aebe520f4d21910629018ab0f3"},
-    {file = "ruamel.yaml.clib-0.2.2-cp36-cp36m-win32.whl", hash = "sha256:e9f7d1d8c26a6a12c23421061f9022bb62704e38211fe375c645485f38df34a2"},
-    {file = "ruamel.yaml.clib-0.2.2-cp36-cp36m-win_amd64.whl", hash = "sha256:2602e91bd5c1b874d6f93d3086f9830f3e907c543c7672cf293a97c3fabdcd91"},
-    {file = "ruamel.yaml.clib-0.2.2-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:44c7b0498c39f27795224438f1a6be6c5352f82cb887bc33d962c3a3acc00df6"},
-    {file = "ruamel.yaml.clib-0.2.2-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:8e8fd0a22c9d92af3a34f91e8a2594eeb35cba90ab643c5e0e643567dc8be43e"},
-    {file = "ruamel.yaml.clib-0.2.2-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:75f0ee6839532e52a3a53f80ce64925ed4aed697dd3fa890c4c918f3304bd4f4"},
-    {file = "ruamel.yaml.clib-0.2.2-cp37-cp37m-win32.whl", hash = "sha256:464e66a04e740d754170be5e740657a3b3b6d2bcc567f0c3437879a6e6087ff6"},
-    {file = "ruamel.yaml.clib-0.2.2-cp37-cp37m-win_amd64.whl", hash = "sha256:52ae5739e4b5d6317b52f5b040b1b6639e8af68a5b8fd606a8b08658fbd0cab5"},
-    {file = "ruamel.yaml.clib-0.2.2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:4df5019e7783d14b79217ad9c56edf1ba7485d614ad5a385d1b3c768635c81c0"},
-    {file = "ruamel.yaml.clib-0.2.2-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:5254af7d8bdf4d5484c089f929cb7f5bafa59b4f01d4f48adda4be41e6d29f99"},
-    {file = "ruamel.yaml.clib-0.2.2-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:8be05be57dc5c7b4a0b24edcaa2f7275866d9c907725226cdde46da09367d923"},
-    {file = "ruamel.yaml.clib-0.2.2-cp38-cp38-win32.whl", hash = "sha256:74161d827407f4db9072011adcfb825b5258a5ccb3d2cd518dd6c9edea9e30f1"},
-    {file = "ruamel.yaml.clib-0.2.2-cp38-cp38-win_amd64.whl", hash = "sha256:058a1cc3df2a8aecc12f983a48bda99315cebf55a3b3a5463e37bb599b05727b"},
-    {file = "ruamel.yaml.clib-0.2.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:c6ac7e45367b1317e56f1461719c853fd6825226f45b835df7436bb04031fd8a"},
-    {file = "ruamel.yaml.clib-0.2.2-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:b4b0d31f2052b3f9f9b5327024dc629a253a83d8649d4734ca7f35b60ec3e9e5"},
-    {file = "ruamel.yaml.clib-0.2.2-cp39-cp39-manylinux2014_aarch64.whl", hash = "sha256:1f8c0a4577c0e6c99d208de5c4d3fd8aceed9574bb154d7a2b21c16bb924154c"},
-    {file = "ruamel.yaml.clib-0.2.2-cp39-cp39-win32.whl", hash = "sha256:46d6d20815064e8bb023ea8628cfb7402c0f0e83de2c2227a88097e239a7dffd"},
-    {file = "ruamel.yaml.clib-0.2.2-cp39-cp39-win_amd64.whl", hash = "sha256:6c0a5dc52fc74eb87c67374a4e554d4761fd42a4d01390b7e868b30d21f4b8bb"},
-    {file = "ruamel.yaml.clib-0.2.2.tar.gz", hash = "sha256:2d24bd98af676f4990c4d715bcdc2a60b19c56a3fb3a763164d2d8ca0e806ba7"},
+    {file = "ruamel.yaml.clib-0.2.6-cp35-cp35m-macosx_10_6_intel.whl", hash = "sha256:cfdb9389d888c5b74af297e51ce357b800dd844898af9d4a547ffc143fa56751"},
+    {file = "ruamel.yaml.clib-0.2.6-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:7b2927e92feb51d830f531de4ccb11b320255ee95e791022555971c466af4527"},
+    {file = "ruamel.yaml.clib-0.2.6-cp35-cp35m-win32.whl", hash = "sha256:ada3f400d9923a190ea8b59c8f60680c4ef8a4b0dfae134d2f2ff68429adfab5"},
+    {file = "ruamel.yaml.clib-0.2.6-cp35-cp35m-win_amd64.whl", hash = "sha256:de9c6b8a1ba52919ae919f3ae96abb72b994dd0350226e28f3686cb4f142165c"},
+    {file = "ruamel.yaml.clib-0.2.6-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:d67f273097c368265a7b81e152e07fb90ed395df6e552b9fa858c6d2c9f42502"},
+    {file = "ruamel.yaml.clib-0.2.6-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:72a2b8b2ff0a627496aad76f37a652bcef400fd861721744201ef1b45199ab78"},
+    {file = "ruamel.yaml.clib-0.2.6-cp36-cp36m-win32.whl", hash = "sha256:9efef4aab5353387b07f6b22ace0867032b900d8e91674b5d8ea9150db5cae94"},
+    {file = "ruamel.yaml.clib-0.2.6-cp36-cp36m-win_amd64.whl", hash = "sha256:846fc8336443106fe23f9b6d6b8c14a53d38cef9a375149d61f99d78782ea468"},
+    {file = "ruamel.yaml.clib-0.2.6-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:0847201b767447fc33b9c235780d3aa90357d20dd6108b92be544427bea197dd"},
+    {file = "ruamel.yaml.clib-0.2.6-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:78988ed190206672da0f5d50c61afef8f67daa718d614377dcd5e3ed85ab4a99"},
+    {file = "ruamel.yaml.clib-0.2.6-cp37-cp37m-win32.whl", hash = "sha256:a49e0161897901d1ac9c4a79984b8410f450565bbad64dbfcbf76152743a0cdb"},
+    {file = "ruamel.yaml.clib-0.2.6-cp37-cp37m-win_amd64.whl", hash = "sha256:bf75d28fa071645c529b5474a550a44686821decebdd00e21127ef1fd566eabe"},
+    {file = "ruamel.yaml.clib-0.2.6-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:a32f8d81ea0c6173ab1b3da956869114cae53ba1e9f72374032e33ba3118c233"},
+    {file = "ruamel.yaml.clib-0.2.6-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:7f7ecb53ae6848f959db6ae93bdff1740e651809780822270eab111500842a84"},
+    {file = "ruamel.yaml.clib-0.2.6-cp38-cp38-win32.whl", hash = "sha256:89221ec6d6026f8ae859c09b9718799fea22c0e8da8b766b0b2c9a9ba2db326b"},
+    {file = "ruamel.yaml.clib-0.2.6-cp38-cp38-win_amd64.whl", hash = "sha256:31ea73e564a7b5fbbe8188ab8b334393e06d997914a4e184975348f204790277"},
+    {file = "ruamel.yaml.clib-0.2.6-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:dc6a613d6c74eef5a14a214d433d06291526145431c3b964f5e16529b1842bed"},
+    {file = "ruamel.yaml.clib-0.2.6-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:1866cf2c284a03b9524a5cc00daca56d80057c5ce3cdc86a52020f4c720856f0"},
+    {file = "ruamel.yaml.clib-0.2.6-cp39-cp39-win32.whl", hash = "sha256:3fb9575a5acd13031c57a62cc7823e5d2ff8bc3835ba4d94b921b4e6ee664104"},
+    {file = "ruamel.yaml.clib-0.2.6-cp39-cp39-win_amd64.whl", hash = "sha256:825d5fccef6da42f3c8eccd4281af399f21c02b32d98e113dbc631ea6a6ecbc7"},
+    {file = "ruamel.yaml.clib-0.2.6.tar.gz", hash = "sha256:4ff604ce439abb20794f05613c374759ce10e3595d1867764dd1ae675b85acbd"},
 ]
 rules = [
     {file = "rules-2.2.tar.gz", hash = "sha256:9bae429f9d4f91a375402990da1541f9e093b0ac077221d57124d06eeeca4405"},
 ]
 s3transfer = [
-    {file = "s3transfer-0.3.7-py2.py3-none-any.whl", hash = "sha256:efa5bd92a897b6a8d5c1383828dca3d52d0790e0756d49740563a3fb6ed03246"},
-    {file = "s3transfer-0.3.7.tar.gz", hash = "sha256:35627b86af8ff97e7ac27975fe0a98a312814b46c6333d8a6b889627bcd80994"},
+    {file = "s3transfer-0.5.0-py3-none-any.whl", hash = "sha256:9c1dc369814391a6bda20ebbf4b70a0f34630592c9aa520856bf384916af2803"},
+    {file = "s3transfer-0.5.0.tar.gz", hash = "sha256:50ed823e1dc5868ad40c8dc92072f757aa0e653a192845c94a3b676f4a62da4c"},
 ]
 safety = [
     {file = "safety-1.10.3-py2.py3-none-any.whl", hash = "sha256:5f802ad5df5614f9622d8d71fedec2757099705c2356f862847c58c6dfe13e84"},
     {file = "safety-1.10.3.tar.gz", hash = "sha256:30e394d02a20ac49b7f65292d19d38fa927a8f9582cdfd3ad1adbbc66c641ad5"},
 ]
 scramp = [
-    {file = "scramp-1.4.0-py3-none-any.whl", hash = "sha256:27349d6839038fe3b56c641ea2a8703df065c1d605fdee67275857c0a82122b4"},
-    {file = "scramp-1.4.0.tar.gz", hash = "sha256:d27d768408c6fc025a0e567eed84325b0aaf24364c81ea5974e8334ae3c4fda3"},
+    {file = "scramp-1.4.1-py3-none-any.whl", hash = "sha256:93c9cc2ffe54a451e02981c07a5a23cbd830701102789939cfb4ff91efd6ca8c"},
+    {file = "scramp-1.4.1.tar.gz", hash = "sha256:f964801077be9be2a1416ffe255d2d78834b3d9d5c8ce5d28f76a856f209f70e"},
 ]
 selenium = [
     {file = "selenium-3.141.0-py2.py3-none-any.whl", hash = "sha256:2d7131d7bc5a5b99a2d9b04aaf2612c411b03b8ca1b1ee8d3de5845a9be2cb3c"},
     {file = "selenium-3.141.0.tar.gz", hash = "sha256:deaf32b60ad91a4611b98d8002757f29e6f2c2d5fcaf202e1c9ad06d6772300d"},
 ]
+sentry-sdk = [
+    {file = "sentry-sdk-1.4.3.tar.gz", hash = "sha256:b9844751e40710e84a457c5bc29b21c383ccb2b63d76eeaad72f7f1c808c8828"},
+    {file = "sentry_sdk-1.4.3-py2.py3-none-any.whl", hash = "sha256:c091cc7115ff25fe3a0e410dbecd7a996f81a3f6137d2272daef32d6c3cfa6dc"},
+]
 six = [
-    {file = "six-1.15.0-py2.py3-none-any.whl", hash = "sha256:8b74bedcbbbaca38ff6d7491d76f2b06b3592611af620f8426e82dddb04a5ced"},
-    {file = "six-1.15.0.tar.gz", hash = "sha256:30639c035cdb23534cd4aa2dd52c3bf48f06e5f4a941509c8bafd8ce11080259"},
+    {file = "six-1.16.0-py2.py3-none-any.whl", hash = "sha256:8abb2f1d86890a2dfb989f9a77cfcfd3e47c2a354b01111771326f8aa26e0254"},
+    {file = "six-1.16.0.tar.gz", hash = "sha256:1e61c37477a1626458e36f7b1d82aa5c9b094fa4802892072e49de9c60c4c926"},
 ]
 smmap = [
-    {file = "smmap-4.0.0-py2.py3-none-any.whl", hash = "sha256:a9a7479e4c572e2e775c404dcd3080c8dc49f39918c2cf74913d30c4c478e3c2"},
-    {file = "smmap-4.0.0.tar.gz", hash = "sha256:7e65386bd122d45405ddf795637b7f7d2b532e7e401d46bbe3fb49b9986d5182"},
+    {file = "smmap-5.0.0-py3-none-any.whl", hash = "sha256:2aba19d6a040e78d8b09de5c57e96207b09ed71d8e55ce0959eeee6c8e190d94"},
+    {file = "smmap-5.0.0.tar.gz", hash = "sha256:c840e62059cd3be204b0c9c9f74be2c09d5648eddd4580d9314c3ecde0b30936"},
 ]
 snowballstemmer = [
     {file = "snowballstemmer-2.1.0-py2.py3-none-any.whl", hash = "sha256:b51b447bea85f9968c13b650126a888aabd4cb4463fca868ec596826325dedc2"},
     {file = "snowballstemmer-2.1.0.tar.gz", hash = "sha256:e997baa4f2e9139951b6f4c631bad912dfd3c792467e2f03d7239464af90e914"},
 ]
 soupsieve = [
-    {file = "soupsieve-2.2.1-py3-none-any.whl", hash = "sha256:c2c1c2d44f158cdbddab7824a9af8c4f83c76b1e23e049479aa432feb6c4c23b"},
-    {file = "soupsieve-2.2.1.tar.gz", hash = "sha256:052774848f448cf19c7e959adf5566904d525f33a3f8b6ba6f6f8f26ec7de0cc"},
+    {file = "soupsieve-2.3-py3-none-any.whl", hash = "sha256:617ffc4d0dfd39c66f4d1413a6e165663a34eca86be9b54f97b91756300ff6df"},
+    {file = "soupsieve-2.3.tar.gz", hash = "sha256:e4860f889dfa88774c07da0b276b70c073b6470fa1a4a8350800bb7bce3dcc76"},
 ]
 spdx-license-list = [
     {file = "spdx_license_list-0.5.2-py3-none-any.whl", hash = "sha256:1b338470c7b403dbecceca563a316382c7977516128ca6c1e8f7078e3ed6e7b0"},
@@ -3594,6 +4945,9 @@ sphinx-autodoc-typehints = [
     {file = "sphinx-autodoc-typehints-1.12.0.tar.gz", hash = "sha256:193617d9dbe0847281b1399d369e74e34cd959c82e02c7efde077fca908a9f52"},
     {file = "sphinx_autodoc_typehints-1.12.0-py3-none-any.whl", hash = "sha256:5e81776ec422dd168d688ab60f034fccfafbcd94329e9537712c93003bddc04a"},
 ]
+sphinx-materialdesign-theme = [
+    {file = "sphinx_materialdesign_theme-0.1.11.tar.gz", hash = "sha256:6e9dae4c6e5ba23c0657a94c1cf65f64be9c8bc1594a6fb41815f7daa3326aa9"},
+]
 sphinxcontrib-applehelp = [
     {file = "sphinxcontrib-applehelp-1.0.2.tar.gz", hash = "sha256:a072735ec80e7675e3f432fcae8610ecf509c5f1869d17e2eecff44389cdbc58"},
     {file = "sphinxcontrib_applehelp-1.0.2-py2.py3-none-any.whl", hash = "sha256:806111e5e962be97c29ec4c1e7fe277bfd19e9652fb1a4392105b43e01af885a"},
@@ -3607,8 +4961,8 @@ sphinxcontrib-django = [
     {file = "sphinxcontrib_django-0.5.1-py2.py3-none-any.whl", hash = "sha256:73ef7fdbf2ed6d4f35b7ae709032bd5ac493d93cedd0624ea7b51bf5fce41267"},
 ]
 sphinxcontrib-htmlhelp = [
-    {file = "sphinxcontrib-htmlhelp-1.0.3.tar.gz", hash = "sha256:e8f5bb7e31b2dbb25b9cc435c8ab7a79787ebf7f906155729338f3156d93659b"},
-    {file = "sphinxcontrib_htmlhelp-1.0.3-py2.py3-none-any.whl", hash = "sha256:3c0bc24a2c41e340ac37c85ced6dafc879ab485c095b1d65d2461ac2f7cca86f"},
+    {file = "sphinxcontrib-htmlhelp-2.0.0.tar.gz", hash = "sha256:f5f8bb2d0d629f398bf47d0d69c07bc13b65f75a81ad9e2f71a63d4b7a2f6db2"},
+    {file = "sphinxcontrib_htmlhelp-2.0.0-py2.py3-none-any.whl", hash = "sha256:d412243dfb797ae3ec2b59eca0e52dac12e75a241bf0e4eb861e450d06c6ed07"},
 ]
 sphinxcontrib-jsmath = [
     {file = "sphinxcontrib-jsmath-1.0.1.tar.gz", hash = "sha256:a9925e4a4587247ed2191a22df5f6970656cb8ca2bd6284309578f2153e0c4b8"},
@@ -3619,23 +4973,23 @@ sphinxcontrib-qthelp = [
     {file = "sphinxcontrib_qthelp-1.0.3-py2.py3-none-any.whl", hash = "sha256:bd9fc24bcb748a8d51fd4ecaade681350aa63009a347a8c14e637895444dfab6"},
 ]
 sphinxcontrib-serializinghtml = [
-    {file = "sphinxcontrib-serializinghtml-1.1.4.tar.gz", hash = "sha256:eaa0eccc86e982a9b939b2b82d12cc5d013385ba5eadcc7e4fed23f4405f77bc"},
-    {file = "sphinxcontrib_serializinghtml-1.1.4-py2.py3-none-any.whl", hash = "sha256:f242a81d423f59617a8e5cf16f5d4d74e28ee9a66f9e5b637a18082991db5a9a"},
+    {file = "sphinxcontrib-serializinghtml-1.1.5.tar.gz", hash = "sha256:aa5f6de5dfdf809ef505c4895e51ef5c9eac17d0f287933eb49ec495280b6952"},
+    {file = "sphinxcontrib_serializinghtml-1.1.5-py2.py3-none-any.whl", hash = "sha256:352a9a00ae864471d3a7ead8d7d79f5fc0b57e8b3f95e9867eb9eb28999b92fd"},
 ]
 sqlparse = [
-    {file = "sqlparse-0.4.1-py3-none-any.whl", hash = "sha256:017cde379adbd6a1f15a61873f43e8274179378e95ef3fede90b5aa64d304ed0"},
-    {file = "sqlparse-0.4.1.tar.gz", hash = "sha256:0f91fd2e829c44362cbcfab3e9ae12e22badaa8a29ad5ff599f9ec109f0454e8"},
+    {file = "sqlparse-0.4.2-py3-none-any.whl", hash = "sha256:48719e356bb8b42991bdbb1e8b83223757b93789c00910a616a071910ca4a64d"},
+    {file = "sqlparse-0.4.2.tar.gz", hash = "sha256:0c00730c74263a94e5a9919ade150dfc3b19c574389985446148402998287dae"},
 ]
 stevedore = [
-    {file = "stevedore-3.3.0-py3-none-any.whl", hash = "sha256:50d7b78fbaf0d04cd62411188fa7eedcb03eb7f4c4b37005615ceebe582aa82a"},
-    {file = "stevedore-3.3.0.tar.gz", hash = "sha256:3a5bbd0652bf552748871eaa73a4a8dc2899786bc497a2aa1fcb4dcdb0debeee"},
+    {file = "stevedore-3.5.0-py3-none-any.whl", hash = "sha256:a547de73308fd7e90075bb4d301405bebf705292fa90a90fc3bcf9133f58616c"},
+    {file = "stevedore-3.5.0.tar.gz", hash = "sha256:f40253887d8712eaa2bb0ea3830374416736dc8ec0e22f5a65092c1174c44335"},
 ]
 termcolor = [
     {file = "termcolor-1.1.0.tar.gz", hash = "sha256:1d6d69ce66211143803fbc56652b41d73b4a400a2891d7bf7a1cdf4c02de613b"},
 ]
 testfixtures = [
-    {file = "testfixtures-6.17.1-py2.py3-none-any.whl", hash = "sha256:9ed31e83f59619e2fa17df053b241e16e0608f4580f7b5a9333a0c9bdcc99137"},
-    {file = "testfixtures-6.17.1.tar.gz", hash = "sha256:5ec3a0dd6f71cc4c304fbc024a10cc293d3e0b852c868014b9f233203e149bda"},
+    {file = "testfixtures-6.18.3-py2.py3-none-any.whl", hash = "sha256:6ddb7f56a123e1a9339f130a200359092bd0a6455e31838d6c477e8729bb7763"},
+    {file = "testfixtures-6.18.3.tar.gz", hash = "sha256:2600100ae96ffd082334b378e355550fef8b4a529a6fa4c34f47130905c7426d"},
 ]
 "testing.common.database" = [
     {file = "testing.common.database-2.0.3-py2.py3-none-any.whl", hash = "sha256:e3ed492bf480a87f271f74c53b262caf5d85c8bc09989a8f534fa2283ec52492"},
@@ -3645,24 +4999,17 @@ testfixtures = [
     {file = "testing.postgresql-1.3.0-py2.py3-none-any.whl", hash = "sha256:1b41daeb98dfc8cd4a584bb91e8f5f4ab182993870f95257afe5f1ba6151a598"},
     {file = "testing.postgresql-1.3.0.tar.gz", hash = "sha256:8e1a69760369a7a8ffe63a66b6d95a5cd82db2fb976e4a8f85ffd24fbfc447d8"},
 ]
-text-unidecode = [
-    {file = "text-unidecode-1.3.tar.gz", hash = "sha256:bad6603bb14d279193107714b288be206cac565dfa49aa5b105294dd5c4aab93"},
-    {file = "text_unidecode-1.3-py2.py3-none-any.whl", hash = "sha256:1311f10e8b895935241623731c2ba64f4c455287888b18189350b67134a822e8"},
-]
 toml = [
     {file = "toml-0.10.2-py2.py3-none-any.whl", hash = "sha256:806143ae5bfb6a3c6e736a764057db0e6a0e05e338b5630894a5f779cabb4f9b"},
     {file = "toml-0.10.2.tar.gz", hash = "sha256:b3bda1d108d5dd99f4a20d24d9c348e91c4db7ab1b749200bded2f839ccbe68f"},
 ]
-tqdm = [
-    {file = "tqdm-4.60.0-py2.py3-none-any.whl", hash = "sha256:daec693491c52e9498632dfbe9ccfc4882a557f5fa08982db1b4d3adbe0887c3"},
-    {file = "tqdm-4.60.0.tar.gz", hash = "sha256:ebdebdb95e3477ceea267decfc0784859aa3df3e27e22d23b83e9b272bf157ae"},
-]
 traitlets = [
-    {file = "traitlets-5.0.5-py3-none-any.whl", hash = "sha256:69ff3f9d5351f31a7ad80443c2674b7099df13cc41fc5fa6e2f6d3b0330b0426"},
-    {file = "traitlets-5.0.5.tar.gz", hash = "sha256:178f4ce988f69189f7e523337a3e11d91c786ded9360174a3d9ca83e79bc5396"},
+    {file = "traitlets-5.1.1-py3-none-any.whl", hash = "sha256:2d313cc50a42cd6c277e7d7dc8d4d7fedd06a2c215f78766ae7b1a66277e0033"},
+    {file = "traitlets-5.1.1.tar.gz", hash = "sha256:059f456c5a7c1c82b98c2e8c799f39c9b8128f6d0d46941ee118daace9eb70c7"},
 ]
 twilio = [
-    {file = "twilio-6.56.0.tar.gz", hash = "sha256:f3d65dcea106383e005460d1cd71703d4f106bcef1e8d289720009ef0f219f28"},
+    {file = "twilio-7.3.0-py2.py3-none-any.whl", hash = "sha256:61fea551066fe4f55b9cbaf0568b54f2d8706a0a63ee77e87f78c1f3a1739935"},
+    {file = "twilio-7.3.0.tar.gz", hash = "sha256:2fe11888675165ab0301d00b9af0c206fd1dfa8d6e5dee6e6b64e42d6e690b83"},
 ]
 typed-ast = [
     {file = "typed_ast-1.4.3-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:2068531575a125b87a41802130fa7e29f26c09a2833fea68d9a40cf33902eba6"},
@@ -3696,17 +5043,25 @@ typed-ast = [
     {file = "typed_ast-1.4.3-cp39-cp39-win_amd64.whl", hash = "sha256:9c6d1a54552b5330bc657b7ef0eae25d00ba7ffe85d9ea8ae6540d2197a3788c"},
     {file = "typed_ast-1.4.3.tar.gz", hash = "sha256:fb1bbeac803adea29cedd70781399c99138358c26d05fcbd23c13016b7f5ec65"},
 ]
+types-pytz = [
+    {file = "types-pytz-2021.3.0.tar.gz", hash = "sha256:86a61967834dceeaaf98b6902ed8357efdd262bb8afcaf4bc8ccecf748592778"},
+    {file = "types_pytz-2021.3.0-py3-none-any.whl", hash = "sha256:b5027e5de50a4c978cd60ca16849d934d44c44ebd7d29cf13ada009efaa9feef"},
+]
+types-pyyaml = [
+    {file = "types-PyYAML-6.0.0.tar.gz", hash = "sha256:3d3591ddfc488fc30be3c506a0c0fe54da968fe98d8b76ab12e59d455330ffca"},
+    {file = "types_PyYAML-6.0.0-py3-none-any.whl", hash = "sha256:746f23d351245d176d7bc89eef79e2ee94b4e7306f7d23bfefb3dc946c0fb58d"},
+]
 typing-extensions = [
-    {file = "typing_extensions-3.7.4.3-py2-none-any.whl", hash = "sha256:dafc7639cde7f1b6e1acc0f457842a83e722ccca8eef5270af2d74792619a89f"},
-    {file = "typing_extensions-3.7.4.3-py3-none-any.whl", hash = "sha256:7cb407020f00f7bfc3cb3e7881628838e69d8f3fcab2f64742a5e76b2f841918"},
-    {file = "typing_extensions-3.7.4.3.tar.gz", hash = "sha256:99d4073b617d30288f569d3f13d2bd7548c3a7e4c8de87db09a9d29bb3a4a60c"},
+    {file = "typing_extensions-3.10.0.2-py2-none-any.whl", hash = "sha256:d8226d10bc02a29bcc81df19a26e56a9647f8b0a6d4a83924139f4a8b01f17b7"},
+    {file = "typing_extensions-3.10.0.2-py3-none-any.whl", hash = "sha256:f1d25edafde516b146ecd0613dabcc61409817af4766fbbcfb8d1ad4ec441a34"},
+    {file = "typing_extensions-3.10.0.2.tar.gz", hash = "sha256:49f75d16ff11f1cd258e1b988ccff82a3ca5570217d7ad8c5f48205dd99a677e"},
 ]
 urllib3 = [
-    {file = "urllib3-1.26.4-py2.py3-none-any.whl", hash = "sha256:2f4da4594db7e1e110a944bb1b551fdf4e6c136ad42e4234131391e21eb5b0df"},
-    {file = "urllib3-1.26.4.tar.gz", hash = "sha256:e7b021f7241115872f92f43c6508082facffbd1c048e3c6e2bb9c2a157e28937"},
+    {file = "urllib3-1.26.7-py2.py3-none-any.whl", hash = "sha256:c4fdf4019605b6e5423637e01bc9fe4daef873709a7973e195ceba0a62bbc844"},
+    {file = "urllib3-1.26.7.tar.gz", hash = "sha256:4987c65554f7a2dbf30c18fd48778ef124af6fab771a377103da0585e2336ece"},
 ]
 uwsgi = [
-    {file = "uWSGI-2.0.19.1.tar.gz", hash = "sha256:faa85e053c0b1be4d5585b0858d3a511d2cd10201802e8676060fd0a109e5869"},
+    {file = "uwsgi-2.0.20.tar.gz", hash = "sha256:88ab9867d8973d8ae84719cf233b7dafc54326fcaec89683c3f9f77c002cdff9"},
 ]
 vine = [
     {file = "vine-5.0.0-py2.py3-none-any.whl", hash = "sha256:4c9dceab6f76ed92105027c49c823800dd33cacce13bdedc5b914e3514b7fb30"},
@@ -3720,11 +5075,65 @@ webencodings = [
     {file = "webencodings-0.5.1-py2.py3-none-any.whl", hash = "sha256:a0af1213f3c2226497a97e2b3aa01a7e4bee4f403f95be16fc9acd2947514a78"},
     {file = "webencodings-0.5.1.tar.gz", hash = "sha256:b36a1c245f2d304965eb4e0a82848379241dc04b865afcc4aab16748587e1923"},
 ]
+whoosh = [
+    {file = "Whoosh-2.7.4-py2.py3-none-any.whl", hash = "sha256:aa39c3c3426e3fd107dcb4bde64ca1e276a65a889d9085a6e4b54ba82420a852"},
+    {file = "Whoosh-2.7.4.tar.gz", hash = "sha256:7ca5633dbfa9e0e0fa400d3151a8a0c4bec53bd2ecedc0a67705b17565c31a83"},
+    {file = "Whoosh-2.7.4.zip", hash = "sha256:e0857375f63e9041e03fedd5b7541f97cf78917ac1b6b06c1fcc9b45375dda69"},
+]
+wrapt = [
+    {file = "wrapt-1.13.3-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:e05e60ff3b2b0342153be4d1b597bbcfd8330890056b9619f4ad6b8d5c96a81a"},
+    {file = "wrapt-1.13.3-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:85148f4225287b6a0665eef08a178c15097366d46b210574a658c1ff5b377489"},
+    {file = "wrapt-1.13.3-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:2dded5496e8f1592ec27079b28b6ad2a1ef0b9296d270f77b8e4a3a796cf6909"},
+    {file = "wrapt-1.13.3-cp27-cp27m-manylinux2010_i686.whl", hash = "sha256:e94b7d9deaa4cc7bac9198a58a7240aaf87fe56c6277ee25fa5b3aa1edebd229"},
+    {file = "wrapt-1.13.3-cp27-cp27m-manylinux2010_x86_64.whl", hash = "sha256:498e6217523111d07cd67e87a791f5e9ee769f9241fcf8a379696e25806965af"},
+    {file = "wrapt-1.13.3-cp27-cp27mu-manylinux1_i686.whl", hash = "sha256:ec7e20258ecc5174029a0f391e1b948bf2906cd64c198a9b8b281b811cbc04de"},
+    {file = "wrapt-1.13.3-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:87883690cae293541e08ba2da22cacaae0a092e0ed56bbba8d018cc486fbafbb"},
+    {file = "wrapt-1.13.3-cp27-cp27mu-manylinux2010_i686.whl", hash = "sha256:f99c0489258086308aad4ae57da9e8ecf9e1f3f30fa35d5e170b4d4896554d80"},
+    {file = "wrapt-1.13.3-cp27-cp27mu-manylinux2010_x86_64.whl", hash = "sha256:6a03d9917aee887690aa3f1747ce634e610f6db6f6b332b35c2dd89412912bca"},
+    {file = "wrapt-1.13.3-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:936503cb0a6ed28dbfa87e8fcd0a56458822144e9d11a49ccee6d9a8adb2ac44"},
+    {file = "wrapt-1.13.3-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:f9c51d9af9abb899bd34ace878fbec8bf357b3194a10c4e8e0a25512826ef056"},
+    {file = "wrapt-1.13.3-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:220a869982ea9023e163ba915077816ca439489de6d2c09089b219f4e11b6785"},
+    {file = "wrapt-1.13.3-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:0877fe981fd76b183711d767500e6b3111378ed2043c145e21816ee589d91096"},
+    {file = "wrapt-1.13.3-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:43e69ffe47e3609a6aec0fe723001c60c65305784d964f5007d5b4fb1bc6bf33"},
+    {file = "wrapt-1.13.3-cp310-cp310-win32.whl", hash = "sha256:78dea98c81915bbf510eb6a3c9c24915e4660302937b9ae05a0947164248020f"},
+    {file = "wrapt-1.13.3-cp310-cp310-win_amd64.whl", hash = "sha256:ea3e746e29d4000cd98d572f3ee2a6050a4f784bb536f4ac1f035987fc1ed83e"},
+    {file = "wrapt-1.13.3-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:8c73c1a2ec7c98d7eaded149f6d225a692caa1bd7b2401a14125446e9e90410d"},
+    {file = "wrapt-1.13.3-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:086218a72ec7d986a3eddb7707c8c4526d677c7b35e355875a0fe2918b059179"},
+    {file = "wrapt-1.13.3-cp35-cp35m-manylinux2010_i686.whl", hash = "sha256:e92d0d4fa68ea0c02d39f1e2f9cb5bc4b4a71e8c442207433d8db47ee79d7aa3"},
+    {file = "wrapt-1.13.3-cp35-cp35m-manylinux2010_x86_64.whl", hash = "sha256:d4a5f6146cfa5c7ba0134249665acd322a70d1ea61732723c7d3e8cc0fa80755"},
+    {file = "wrapt-1.13.3-cp35-cp35m-win32.whl", hash = "sha256:8aab36778fa9bba1a8f06a4919556f9f8c7b33102bd71b3ab307bb3fecb21851"},
+    {file = "wrapt-1.13.3-cp35-cp35m-win_amd64.whl", hash = "sha256:944b180f61f5e36c0634d3202ba8509b986b5fbaf57db3e94df11abee244ba13"},
+    {file = "wrapt-1.13.3-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:2ebdde19cd3c8cdf8df3fc165bc7827334bc4e353465048b36f7deeae8ee0918"},
+    {file = "wrapt-1.13.3-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:610f5f83dd1e0ad40254c306f4764fcdc846641f120c3cf424ff57a19d5f7ade"},
+    {file = "wrapt-1.13.3-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:5601f44a0f38fed36cc07db004f0eedeaadbdcec90e4e90509480e7e6060a5bc"},
+    {file = "wrapt-1.13.3-cp36-cp36m-musllinux_1_1_i686.whl", hash = "sha256:e6906d6f48437dfd80464f7d7af1740eadc572b9f7a4301e7dd3d65db285cacf"},
+    {file = "wrapt-1.13.3-cp36-cp36m-musllinux_1_1_x86_64.whl", hash = "sha256:766b32c762e07e26f50d8a3468e3b4228b3736c805018e4b0ec8cc01ecd88125"},
+    {file = "wrapt-1.13.3-cp36-cp36m-win32.whl", hash = "sha256:5f223101f21cfd41deec8ce3889dc59f88a59b409db028c469c9b20cfeefbe36"},
+    {file = "wrapt-1.13.3-cp36-cp36m-win_amd64.whl", hash = "sha256:f122ccd12fdc69628786d0c947bdd9cb2733be8f800d88b5a37c57f1f1d73c10"},
+    {file = "wrapt-1.13.3-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:46f7f3af321a573fc0c3586612db4decb7eb37172af1bc6173d81f5b66c2e068"},
+    {file = "wrapt-1.13.3-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:778fd096ee96890c10ce96187c76b3e99b2da44e08c9e24d5652f356873f6709"},
+    {file = "wrapt-1.13.3-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:0cb23d36ed03bf46b894cfec777eec754146d68429c30431c99ef28482b5c1df"},
+    {file = "wrapt-1.13.3-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:96b81ae75591a795d8c90edc0bfaab44d3d41ffc1aae4d994c5aa21d9b8e19a2"},
+    {file = "wrapt-1.13.3-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:7dd215e4e8514004c8d810a73e342c536547038fb130205ec4bba9f5de35d45b"},
+    {file = "wrapt-1.13.3-cp37-cp37m-win32.whl", hash = "sha256:47f0a183743e7f71f29e4e21574ad3fa95676136f45b91afcf83f6a050914829"},
+    {file = "wrapt-1.13.3-cp37-cp37m-win_amd64.whl", hash = "sha256:fd76c47f20984b43d93de9a82011bb6e5f8325df6c9ed4d8310029a55fa361ea"},
+    {file = "wrapt-1.13.3-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:b73d4b78807bd299b38e4598b8e7bd34ed55d480160d2e7fdaabd9931afa65f9"},
+    {file = "wrapt-1.13.3-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:ec9465dd69d5657b5d2fa6133b3e1e989ae27d29471a672416fd729b429eb554"},
+    {file = "wrapt-1.13.3-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:dd91006848eb55af2159375134d724032a2d1d13bcc6f81cd8d3ed9f2b8e846c"},
+    {file = "wrapt-1.13.3-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:ae9de71eb60940e58207f8e71fe113c639da42adb02fb2bcbcaccc1ccecd092b"},
+    {file = "wrapt-1.13.3-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:51799ca950cfee9396a87f4a1240622ac38973b6df5ef7a41e7f0b98797099ce"},
+    {file = "wrapt-1.13.3-cp38-cp38-win32.whl", hash = "sha256:4b9c458732450ec42578b5642ac53e312092acf8c0bfce140ada5ca1ac556f79"},
+    {file = "wrapt-1.13.3-cp38-cp38-win_amd64.whl", hash = "sha256:7dde79d007cd6dfa65afe404766057c2409316135cb892be4b1c768e3f3a11cb"},
+    {file = "wrapt-1.13.3-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:981da26722bebb9247a0601e2922cedf8bb7a600e89c852d063313102de6f2cb"},
+    {file = "wrapt-1.13.3-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:705e2af1f7be4707e49ced9153f8d72131090e52be9278b5dbb1498c749a1e32"},
+    {file = "wrapt-1.13.3-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:25b1b1d5df495d82be1c9d2fad408f7ce5ca8a38085e2da41bb63c914baadff7"},
+    {file = "wrapt-1.13.3-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:77416e6b17926d953b5c666a3cb718d5945df63ecf922af0ee576206d7033b5e"},
+    {file = "wrapt-1.13.3-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:865c0b50003616f05858b22174c40ffc27a38e67359fa1495605f96125f76640"},
+    {file = "wrapt-1.13.3-cp39-cp39-win32.whl", hash = "sha256:0a017a667d1f7411816e4bf214646d0ad5b1da2c1ea13dec6c162736ff25a374"},
+    {file = "wrapt-1.13.3-cp39-cp39-win_amd64.whl", hash = "sha256:81bd7c90d28a4b2e1df135bfbd7c23aee3050078ca6441bead44c42483f9ebfb"},
+    {file = "wrapt-1.13.3.tar.gz", hash = "sha256:1fea9cd438686e6682271d36f3481a9f3636195578bab9ca3382e2f5f01fc185"},
+]
 yubiotp = [
     {file = "YubiOTP-1.0.0.post1-py2.py3-none-any.whl", hash = "sha256:7ad57011866e0bc6c6d179ffbc3926fcc0e82d410178a6d01ba4da0f88332878"},
     {file = "YubiOTP-1.0.0.post1.tar.gz", hash = "sha256:c13825f7b76a69afb92f19521f4dea9f5031d70f45123b505dc2e0ac03132065"},
 ]
-zipp = [
-    {file = "zipp-3.4.1-py3-none-any.whl", hash = "sha256:51cb66cc54621609dd593d1787f286ee42a5c0adbb4b29abea5a63edc3e03098"},
-    {file = "zipp-3.4.1.tar.gz", hash = "sha256:3607921face881ba3e026887d8150cca609d517579abe052ac81fc5aeffdbd76"},
-]
diff --git a/pyproject.toml b/pyproject.toml
index 204a9a147eefbff4354efebb204b212cee1d50d4..5077986b42c5b530ecd42c2a26851027873e7519 100644
--- a/pyproject.toml
+++ b/pyproject.toml
@@ -1,22 +1,35 @@
 [tool.poetry]
 name = "AlekSIS-Core"
-version = "2.0a5.dev0"
+version = "2.2.dev0"
 packages = [
     { include = "aleksis" }
 ]
 readme = "README.rst"
-include = ["CHANGELOG.rst", "LICENCE.rst", "docs/*", "docs/**/*", "aleksis/**/*.mo", "tox.ini"]
+include = ["CHANGELOG.rst", "LICENCE.rst", "docs/*", "docs/**/*", "aleksis/**/*.mo", "conftest.py", "tox.ini"]
 
 description = "AlekSIS (School Information System) — Core"
-authors = ["Dominik George <dominik.george@teckids.org>", "Julian Leucker <leuckeju@katharineum.de>", "mirabilos <thorsten.glaser@teckids.org>", "Frank Poetzsch-Heffter <p-h@katharineum.de>", "Tom Teichler <tom.teichler@teckids.org>", "Jonathan Weth <wethjo@katharineum.de>", "Hangzhi Yu <yuha@katharineum.de>"]
-maintainers = ["Jonathan Weth <wethjo@katharineum.de>", "Dominik George <dominik.george@teckids.org>"]
+authors = [
+    "Dominik George <dominik.george@teckids.org>",
+    "Julian Leucker <leuckeju@katharineum.de>",
+    "mirabilos <thorsten.glaser@teckids.org>",
+    "Frank Poetzsch-Heffter <p-h@katharineum.de>",
+    "Tom Teichler <tom.teichler@teckids.org>",
+    "Jonathan Weth <dev@jonathanweth.de>",
+    "Hangzhi Yu <yuha@katharineum.de>",
+    "Lloyd Meins <meinsll@katharineum.de>",
+    "magicfelix <felix@felix-zauberer.de>"
+]
+maintainers = [
+    "Jonathan Weth <dev@jonathanweth.de>",
+    "Dominik George <dominik.george@teckids.org>"
+]
 license = "EUPL-1.2-or-later"
 homepage = "https://aleksis.org/"
 repository = "https://edugit.org/AlekSIS/official/AlekSIS-Core"
 documentation = "https://aleksis.org/AlekSIS-Core/docs/html/"
 keywords = ["SIS", "education", "school", "digitisation", "school apps"]
 classifiers = [
-    "Development Status :: 3 - Alpha",
+    "Development Status :: 4 - Beta",
     "Environment :: Web Environment",
     "Framework :: Django :: 3.0",
     "Intended Audience :: Developers",
@@ -33,36 +46,30 @@ url = "https://edugit.org/api/v4/projects/461/packages/pypi/simple"
 secondary = true
 
 [tool.poetry.dependencies]
-python = "^3.7"
-Django = "^3.2"
+python = "^3.9"
+Django = "^3.2.5"
 django-any-js = "^1.1"
 django-debug-toolbar = "^3.2"
-django-middleware-global-request = "^0.1.2"
 django-menu-generator-ng = "^1.2.3"
 django-tables2 = "^2.1"
-Pillow = "^8.0"
-django-phonenumber-field = {version = "<5.2", extras = ["phonenumbers"]}
-django-sass-processor = "^1.0"
-libsass = "^0.20.0"
+django-phonenumber-field = {version = "^5.2", extras = ["phonenumbers"]}
+django-sass-processor = "1.0"
+libsass = "^0.21.0"
 colour = "^0.1.5"
 dynaconf = {version = "^3.1", extras = ["yaml", "toml", "ini"]}
-django-settings-context-processor = "^0.2"
-django-auth-ldap = { version = "^2.2", optional = true }
+django-auth-ldap = { version = "^3.0", optional = true }
 django-maintenance-mode = "^0.16.0"
-django-ipware = "^3.0"
+django-ipware = "^4.0"
 django-impersonate = "^1.4"
-django-hattori = "^0.2"
 psycopg2 = "^2.8"
 django_select2 = "^7.1"
-requests = "^2.22"
 django-two-factor-auth = { version = "^1.12.1", extras = [ "yubikey", "phonenumbers", "call", "sms" ] }
 django-yarnpkg = "^6.0"
 django-material = "^1.6.0"
-django-pwa = "^1.0.8"
-django-dynamic-preferences = "^1.9"
+django-dynamic-preferences = "^1.11"
 django_widget_tweaks = "^1.4.5"
 django-filter = "^2.2.0"
-django-templated-email = "^2.3.0"
+django-templated-email = "^3.0.0"
 html2text = "^2020.0.0"
 django-ckeditor = "^6.0.0"
 django-js-reverse = "^0.9.1"
@@ -74,36 +81,44 @@ django-celery-email = "^3.0.0"
 django-jsonstore = "^0.5.0"
 django-polymorphic = "^3.0.0"
 django-colorfield = "^0.4.0"
-django-bleach = "^0.6.1"
+django-bleach = "^0.9.0"
 django-guardian = "^2.2.0"
 rules = "^2.2"
 django-cache-memoize = "^0.1.6"
-django-haystack = {version="3.0", allow-prereleases = true}
+django-haystack = "^3.1"
 celery-haystack-ng = "^0.20"
 django-dbbackup = "^3.3.0"
 spdx-license-list = "^0.5.0"
 license-expression = "^1.2"
-django-reversion = "^3.0.7"
-django-favicon-plus-reloaded = "^1.0.4"
+django-reversion = "^4.0.0"
+django-favicon-plus-reloaded = "^1.1.2"
 django-health-check = "^3.12.1"
 psutil = "^5.7.0"
 celery-progress = "^0.1.0"
 django-cachalot = "^2.3.2"
 django-prometheus = "^2.1.0"
-importlib-metadata = {version = "^3.0.0", python = "<3.9"}
 django-model-utils = "^4.0.0"
 bs4 = "^0.0.1"
+django-allauth = "^0.45.0"
 django-uwsgi-ng = "^1.1.0"
 django-extensions = "^3.1.1"
 ipython = "^7.20.0"
-django-redis = "^4.12.1"
+django-oauth-toolkit = "^1.5.0"
+django-redis = "^5.0.0"
 django-storages = {version = "^1.11.1", optional = true}
 boto3 = {version = "^1.17.33", optional = true}
 django-cleanup = "^5.1.0"
+djangorestframework = "^3.12.4"
+Whoosh = "^2.7.4"
+django-titofisto = "^0.1.0"
+haystack-redis = "^0.0.1"
+python-gnupg = "^0.4.7"
+sentry-sdk = {version = "^1.4.3", optional = true}
 
 [tool.poetry.extras]
 ldap = ["django-auth-ldap"]
 s3 = ["boto3", "django-storages"]
+sentry = ["sentry"]
 
 [tool.poetry.dev-dependencies]
 aleksis-builddeps = "*"
diff --git a/tox.ini b/tox.ini
index 2aed3362cf7e62c15839c37ec5fb191dcd1a2f4d..6ba5d926ea520a574125f4a9717b93ba0020a2c3 100644
--- a/tox.ini
+++ b/tox.ini
@@ -47,6 +47,11 @@ commands =
     poetry run isort aleksis/
     poetry run black aleksis/
 
+[testenv:makemessages]
+commands =
+    poetry run aleksis-admin makemessages --no-wrap -e html,txt,py,email -i static -l ar -l de_DE -l fr -l nb_NO -l tr_TR -l la
+    poetry run aleksis-admin makemessages --no-wrap -d djangojs -i **/node_modules -l ar -l de_DE -l fr -l nb_NO -l tr_TR -l la
+
 [flake8]
 max_line_length = 100
 exclude = migrations,tests