Skip to content

Commit ae97d2e

Browse files
committed
🐛(backend) compute ancestor_links in get_abilities if needed
The refactor made in the tree view caching the ancestors_links to not compute them again in the document.get_abilities method lead to a bug. If the get_abilities method is called without ancestors_links, then they are computed on all the ancestors but not from the highest readable ancestor for the current user. We have to compute them with this constraint.
1 parent c882f13 commit ae97d2e

File tree

5 files changed

+90
-4
lines changed

5 files changed

+90
-4
lines changed

CHANGELOG.md

+4
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,10 @@ and this project adheres to
1616

1717
- ♻️(frontend) Integrate UI kit #783
1818

19+
## Fixed
20+
21+
- 🐛(backend) compute ancestor_links in get_abilities if needed #725
22+
1923
## [2.6.0] - 2025-03-21
2024

2125
## Added

src/backend/core/api/viewsets.py

+14-2
Original file line numberDiff line numberDiff line change
@@ -846,14 +846,15 @@ def tree(self, request, pk, *args, **kwargs):
846846
)
847847

848848
# Get the highest readable ancestor
849-
highest_readable = ancestors.readable_per_se(request.user).only("depth").first()
849+
highest_readable = (
850+
ancestors.readable_per_se(request.user).only("depth", "path").first()
851+
)
850852
if highest_readable is None:
851853
raise (
852854
drf.exceptions.PermissionDenied()
853855
if request.user.is_authenticated
854856
else drf.exceptions.NotAuthenticated()
855857
)
856-
857858
paths_links_mapping = {}
858859
ancestors_links = []
859860
children_clause = db.Q()
@@ -876,6 +877,17 @@ def tree(self, request, pk, *args, **kwargs):
876877

877878
queryset = ancestors.filter(depth__gte=highest_readable.depth) | children
878879
queryset = queryset.order_by("path")
880+
# Annotate if the current document is the highest ancestor for the user
881+
queryset = queryset.annotate(
882+
is_highest_ancestor_for_user=db.Case(
883+
db.When(
884+
path=db.Value(highest_readable.path),
885+
then=db.Value(True),
886+
),
887+
default=db.Value(False),
888+
output_field=db.BooleanField(),
889+
)
890+
)
879891
queryset = self.annotate_user_roles(queryset)
880892
queryset = self.annotate_is_favorite(queryset)
881893

src/backend/core/models.py

+27-1
Original file line numberDiff line numberDiff line change
@@ -754,14 +754,40 @@ def get_links_definitions(self, ancestors_links):
754754

755755
return dict(links_definitions) # Convert defaultdict back to a normal dict
756756

757+
def compute_ancestors_links(self, user):
758+
"""
759+
Compute the ancestors links for the current document up to the highest readable ancestor.
760+
"""
761+
ancestors = (
762+
(self.get_ancestors() | self._meta.model.objects.filter(pk=self.pk))
763+
.filter(ancestors_deleted_at__isnull=True)
764+
.order_by("path")
765+
)
766+
highest_readable = ancestors.readable_per_se(user).only("depth").first()
767+
768+
if highest_readable is None:
769+
return []
770+
771+
ancestors_links = []
772+
paths_links_mapping = {}
773+
for ancestor in ancestors.filter(depth__gte=highest_readable.depth):
774+
ancestors_links.append(
775+
{"link_reach": ancestor.link_reach, "link_role": ancestor.link_role}
776+
)
777+
paths_links_mapping[ancestor.path] = ancestors_links.copy()
778+
779+
ancestors_links = paths_links_mapping.get(self.path[: -self.steplen], [])
780+
781+
return ancestors_links
782+
757783
def get_abilities(self, user, ancestors_links=None):
758784
"""
759785
Compute and return abilities for a given user on the document.
760786
"""
761787
if self.depth <= 1 or getattr(self, "is_highest_ancestor_for_user", False):
762788
ancestors_links = []
763789
elif ancestors_links is None:
764-
ancestors_links = self.get_ancestors().values("link_reach", "link_role")
790+
ancestors_links = self.compute_ancestors_links(user=user)
765791

766792
roles = set(
767793
self.get_roles(user)

src/backend/core/tests/documents/test_api_documents_retrieve.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -789,7 +789,7 @@ def test_api_documents_retrieve_user_roles(django_assert_max_num_queries):
789789
)
790790
expected_roles = {access.role for access in accesses}
791791

792-
with django_assert_max_num_queries(12):
792+
with django_assert_max_num_queries(14):
793793
response = client.get(f"/api/v1.0/documents/{document.id!s}/")
794794

795795
assert response.status_code == 200

src/backend/core/tests/test_models_documents.py

+44
Original file line numberDiff line numberDiff line change
@@ -1305,3 +1305,47 @@ def test_models_documents_restore_complex_bis(django_assert_num_queries):
13051305
def test_models_documents_get_select_options(ancestors_links, select_options):
13061306
"""Validate that the "get_select_options" method operates as expected."""
13071307
assert models.LinkReachChoices.get_select_options(ancestors_links) == select_options
1308+
1309+
1310+
def test_models_documents_compute_ancestors_links_no_highest_readable():
1311+
"""Test the compute_ancestors_links method."""
1312+
document = factories.DocumentFactory(link_reach="public")
1313+
assert document.compute_ancestors_links(user=AnonymousUser()) == []
1314+
1315+
1316+
def test_models_documents_compute_ancestors_links_highest_readable(
1317+
django_assert_num_queries,
1318+
):
1319+
"""Test the compute_ancestors_links method."""
1320+
user = factories.UserFactory()
1321+
other_user = factories.UserFactory()
1322+
root = factories.DocumentFactory(
1323+
link_reach="restricted", link_role="reader", users=[user]
1324+
)
1325+
1326+
factories.DocumentFactory(
1327+
parent=root, link_reach="public", link_role="reader", users=[user]
1328+
)
1329+
child2 = factories.DocumentFactory(
1330+
parent=root,
1331+
link_reach="authenticated",
1332+
link_role="editor",
1333+
users=[user, other_user],
1334+
)
1335+
child3 = factories.DocumentFactory(
1336+
parent=child2,
1337+
link_reach="authenticated",
1338+
link_role="reader",
1339+
users=[user, other_user],
1340+
)
1341+
1342+
with django_assert_num_queries(2):
1343+
assert child3.compute_ancestors_links(user=user) == [
1344+
{"link_reach": root.link_reach, "link_role": root.link_role},
1345+
{"link_reach": child2.link_reach, "link_role": child2.link_role},
1346+
]
1347+
1348+
with django_assert_num_queries(2):
1349+
assert child3.compute_ancestors_links(user=other_user) == [
1350+
{"link_reach": child2.link_reach, "link_role": child2.link_role},
1351+
]

0 commit comments

Comments
 (0)