Prevent database error when calling permission_order.register on app ready

Fixes #12742

Previously, `permission_order.register` performed a database lookup for the content type. This is invalid if called from an app `ready` method as the documentation suggests, because this may run before the database has been initialised. Instead, `register` now queues up the arguments it receives, and the content type lookup is constructed lazily on first call to `get_content_type_order_lookup` (which happens when the group edit view is requested).
pull/12821/head
Matt Westcott 2025-01-24 01:49:26 +00:00
rodzic 77a55e3f60
commit 6489eae6cf
3 zmienionych plików z 29 dodań i 4 usunięć

Wyświetl plik

@ -7,3 +7,13 @@ class WagtailSnippetsTestsAppConfig(AppConfig):
name = "wagtail.test.snippets"
label = "snippetstests"
verbose_name = _("Wagtail snippets tests")
def ready(self):
# Test registration of permission order within the group permissions view,
# as per https://docs.wagtail.org/en/stable/extending/customizing_group_views.html#customizing-the-group-editor-permissions-ordering
# Invoking `register` from `ready` confirms that it does not perform any database queries -
# if it did, it would fail (on a standard test run without --keepdb at least) because the
# test database hasn't been migrated yet.
from wagtail.users.permission_order import register
register("snippetstests.fancysnippet", order=999)

Wyświetl plik

@ -2,6 +2,7 @@ from django.contrib.contenttypes.models import ContentType
from wagtail.coreutils import resolve_model_string
content_types_to_register = []
CONTENT_TYPE_ORDER = {}
@ -13,5 +14,18 @@ def register(model, **kwargs):
"""
order = kwargs.pop("order", None)
if order is not None:
content_type = ContentType.objects.get_for_model(resolve_model_string(model))
CONTENT_TYPE_ORDER[content_type.id] = order
# We typically call this at application startup, when the database may not be ready,
# and so we can't look up the content type yet. Instead we will queue up the
# (model, order) pair to be processed when the lookup is requested.
content_types_to_register.append((model, order))
def get_content_type_order_lookup():
if content_types_to_register:
for model, order in content_types_to_register:
content_type = ContentType.objects.get_for_model(
resolve_model_string(model)
)
CONTENT_TYPE_ORDER[content_type.id] = order
content_types_to_register.clear()
return CONTENT_TYPE_ORDER

Wyświetl plik

@ -11,7 +11,7 @@ from django.utils.translation import gettext_noop
from wagtail import hooks
from wagtail.admin.models import Admin
from wagtail.coreutils import accepts_kwarg
from wagtail.users.permission_order import CONTENT_TYPE_ORDER
from wagtail.users.permission_order import get_content_type_order_lookup
from wagtail.utils.deprecation import RemovedInWagtail70Warning
register = template.Library()
@ -96,9 +96,10 @@ def format_permissions(permission_bound_field):
# get a distinct and ordered list of the content types that these permissions relate to.
# relies on Permission model default ordering, dict.fromkeys() retaining that order
# from the queryset, and the stability of sorted().
content_type_order = get_content_type_order_lookup()
content_type_ids = sorted(
dict.fromkeys(permissions.values_list("content_type_id", flat=True)),
key=lambda ct: CONTENT_TYPE_ORDER.get(ct, float("inf")),
key=lambda ct: content_type_order.get(ct, float("inf")),
)
# iterate over permission_bound_field to build a lookup of individual renderable