Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit 12f33cd

Browse files
committedMar 15, 2025·
🐛(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 computes on all the ancestors but not from the highest readable ancestor for the current user. We have to compute them with this constraints and to avoid to compute it again and again we save the result in a cache.
1 parent 7007d56 commit 12f33cd

File tree

5 files changed

+113
-10
lines changed

5 files changed

+113
-10
lines changed
 

‎CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@ and this project adheres to
2424
- 🔒️(back) restrict access to favorite_list endpoint #690
2525
- 🐛(backend) refactor to fix filtering on children
2626
and descendants views #695
27+
- 🐛(backend) compute ancestor_links in get_abilities if needed #725
2728

2829

2930
## [2.4.0] - 2025-03-06

‎src/backend/core/api/viewsets.py

+14-2
Original file line numberDiff line numberDiff line change
@@ -833,14 +833,15 @@ def tree(self, request, pk, *args, **kwargs):
833833
)
834834

835835
# Get the highest readable ancestor
836-
highest_readable = ancestors.readable_per_se(request.user).only("depth").first()
836+
highest_readable = (
837+
ancestors.readable_per_se(request.user).only("depth", "path").first()
838+
)
837839
if highest_readable is None:
838840
raise (
839841
drf.exceptions.PermissionDenied()
840842
if request.user.is_authenticated
841843
else drf.exceptions.NotAuthenticated()
842844
)
843-
844845
paths_links_mapping = {}
845846
ancestors_links = []
846847
children_clause = db.Q()
@@ -863,6 +864,17 @@ def tree(self, request, pk, *args, **kwargs):
863864

864865
queryset = ancestors.filter(depth__gte=highest_readable.depth) | children
865866
queryset = queryset.order_by("path")
867+
# Annotate if the current document is the highest ancestor for the user
868+
queryset = queryset.annotate(
869+
is_highest_ancestor_for_user=db.Case(
870+
db.When(
871+
path=db.Value(highest_readable.path),
872+
then=db.Value(True),
873+
),
874+
default=db.Value(False),
875+
output_field=db.BooleanField(),
876+
)
877+
)
866878
queryset = self.annotate_user_roles(queryset)
867879
queryset = self.annotate_is_favorite(queryset)
868880

‎src/backend/core/models.py

+37-1
Original file line numberDiff line numberDiff line change
@@ -730,14 +730,50 @@ def get_links_definitions(self, ancestors_links):
730730

731731
return dict(links_definitions) # Convert defaultdict back to a normal dict
732732

733+
def compute_ancestors_links(self, user):
734+
"""
735+
Compute the ancestors links for the current document up to the highest readable ancestor.
736+
"""
737+
cache_key = f"ancestors_links_{self.id!s}_{user.id!s}"
738+
ancestors_links = cache.get(cache_key)
739+
if ancestors_links:
740+
return ancestors_links
741+
742+
ancestors = (
743+
(self.get_ancestors() | self._meta.model.objects.filter(pk=self.pk))
744+
.filter(ancestors_deleted_at__isnull=True)
745+
.order_by("path")
746+
)
747+
highest_readable = ancestors.readable_per_se(user).only("depth", "path").first()
748+
749+
if highest_readable is None:
750+
return []
751+
752+
ancestors_links = []
753+
paths_links_mapping = {}
754+
for ancestor in ancestors:
755+
if ancestor.depth < highest_readable.depth:
756+
continue
757+
758+
ancestors_links.append(
759+
{"link_reach": ancestor.link_reach, "link_role": ancestor.link_role}
760+
)
761+
paths_links_mapping[ancestor.path] = ancestors_links.copy()
762+
763+
ancestors_links = paths_links_mapping.get(self.path[: -self.steplen], [])
764+
765+
cache.set(cache_key, ancestors_links)
766+
767+
return ancestors_links
768+
733769
def get_abilities(self, user, ancestors_links=None):
734770
"""
735771
Compute and return abilities for a given user on the document.
736772
"""
737773
if self.depth <= 1 or getattr(self, "is_highest_ancestor_for_user", False):
738774
ancestors_links = []
739775
elif ancestors_links is None:
740-
ancestors_links = self.get_ancestors().values("link_reach", "link_role")
776+
ancestors_links = self.compute_ancestors_links(user=user)
741777

742778
roles = set(
743779
self.get_roles(user)

‎src/backend/core/tests/documents/test_api_documents_tree.py

+24-7
Original file line numberDiff line numberDiff line change
@@ -139,10 +139,13 @@ def test_api_documents_tree_list_anonymous_public_parent():
139139
has a public ancestor but only up to the highest public ancestor.
140140
"""
141141
great_grand_parent = factories.DocumentFactory(
142-
link_reach=random.choice(["authenticated", "restricted"])
142+
link_reach="authenticated",
143+
link_role="editor",
143144
)
144145
grand_parent = factories.DocumentFactory(
145-
link_reach="public", parent=great_grand_parent
146+
link_reach="public",
147+
parent=great_grand_parent,
148+
link_role="reader",
146149
)
147150
factories.DocumentFactory(link_reach="public", parent=great_grand_parent)
148151
factories.DocumentFactory(
@@ -151,14 +154,28 @@ def test_api_documents_tree_list_anonymous_public_parent():
151154
)
152155

153156
parent = factories.DocumentFactory(
154-
parent=grand_parent, link_reach=random.choice(["authenticated", "restricted"])
157+
link_role="reader",
158+
link_reach="authenticated",
159+
parent=grand_parent,
160+
)
161+
parent_sibling = factories.DocumentFactory(
162+
parent=grand_parent,
163+
link_role="reader",
164+
link_reach="restricted",
155165
)
156-
parent_sibling = factories.DocumentFactory(parent=grand_parent)
157166
document = factories.DocumentFactory(
158-
link_reach=random.choice(["authenticated", "restricted"]), parent=parent
167+
parent=parent,
168+
link_role="editor",
169+
link_reach="restricted",
170+
)
171+
document_sibling = factories.DocumentFactory(
172+
parent=parent,
173+
link_role="reader",
174+
link_reach="restricted",
175+
)
176+
child = factories.DocumentFactory(
177+
link_reach="public", link_role="editor", parent=document
159178
)
160-
document_sibling = factories.DocumentFactory(parent=parent)
161-
child = factories.DocumentFactory(link_reach="public", parent=document)
162179

163180
response = APIClient().get(f"/api/v1.0/documents/{document.id!s}/tree/")
164181

‎src/backend/core/tests/test_models_documents.py

+37
Original file line numberDiff line numberDiff line change
@@ -1297,3 +1297,40 @@ def test_models_documents_restore_complex_bis(django_assert_num_queries):
12971297
def test_models_documents_get_select_options(ancestors_links, select_options):
12981298
"""Validate that the "get_select_options" method operates as expected."""
12991299
assert models.LinkReachChoices.get_select_options(ancestors_links) == select_options
1300+
1301+
def test_models_documents_compute_ancestors_links_no_highest_readable():
1302+
"""Test the compute_ancestors_links method."""
1303+
document = factories.DocumentFactory(link_reach="public")
1304+
assert document.compute_ancestors_links(user=AnonymousUser()) == []
1305+
1306+
def test_models_documents_compute_ancestors_links_highest_readable(django_assert_num_queries):
1307+
"""Test the compute_ancestors_links method."""
1308+
user = factories.UserFactory()
1309+
other_user = factories.UserFactory()
1310+
root = factories.DocumentFactory(link_reach="restricted", link_role="reader", users=[user])
1311+
1312+
factories.DocumentFactory(parent=root, link_reach="public", link_role="reader", users=[user])
1313+
child2 = factories.DocumentFactory(parent=root, link_reach="authenticated", link_role="editor", users=[user, other_user])
1314+
child3 = factories.DocumentFactory(parent=child2, link_reach="authenticated", link_role="reader", users=[user, other_user])
1315+
1316+
with django_assert_num_queries(2):
1317+
assert child3.compute_ancestors_links(user=user) == [
1318+
{"link_reach": root.link_reach, "link_role": root.link_role},
1319+
{"link_reach": child2.link_reach, "link_role": child2.link_role},
1320+
]
1321+
1322+
with django_assert_num_queries(0):
1323+
assert child3.compute_ancestors_links(user=user) == [
1324+
{"link_reach": root.link_reach, "link_role": root.link_role},
1325+
{"link_reach": child2.link_reach, "link_role": child2.link_role},
1326+
]
1327+
1328+
with django_assert_num_queries(2):
1329+
assert child3.compute_ancestors_links(user=other_user) == [
1330+
{"link_reach": child2.link_reach, "link_role": child2.link_role},
1331+
]
1332+
1333+
with django_assert_num_queries(0):
1334+
assert child3.compute_ancestors_links(user=other_user) == [
1335+
{"link_reach": child2.link_reach, "link_role": child2.link_role},
1336+
]

0 commit comments

Comments
 (0)
Please sign in to comment.