Skip to content

Commit

Permalink
🛂(backend) use playlist token in markdown apis
Browse files Browse the repository at this point in the history
As we are now using playlist tokens, it has to be allowed on apis.
Also, old routes can't be used anymore, as token resource is now a
playlist.
  • Loading branch information
kernicPanel committed Aug 7, 2023
1 parent 79d5b65 commit 198299f
Show file tree
Hide file tree
Showing 15 changed files with 180 additions and 113 deletions.
26 changes: 21 additions & 5 deletions src/backend/marsha/markdown/api.py
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,22 @@
from .utils.converter import LatexConversionException, render_latex_to_image


class ObjectMarkdownDocumentRelatedMixin:
"""
Get the related markdown document id contained in resource.
It exposes a function used to get the related markdown document.
It is also useful to avoid URL crafting (when the url markdown_document_id doesn't
match token resource markdown document id).
"""

def get_related_markdown_document_id(self):
"""Get the related markdown document ID from the request."""

# The video ID in the URL is mandatory.
return self.kwargs.get("markdown_document_id")


class MarkdownDocumentFilter(django_filters.FilterSet):
"""Filter for file depository."""

Expand Down Expand Up @@ -57,7 +73,7 @@ class MarkdownDocumentViewSet(

permission_classes = [
(
core_permissions.IsTokenResourceRouteObject
core_permissions.IsPlaylistToken
& (core_permissions.IsTokenInstructor | core_permissions.IsTokenAdmin)
)
| markdown_permissions.IsMarkdownDocumentPlaylistOrOrganizationAdmin
Expand Down Expand Up @@ -272,6 +288,7 @@ class MarkdownImageViewSet(
APIViewMixin,
ObjectPkMixin,
ObjectRelatedMixin,
ObjectMarkdownDocumentRelatedMixin,
mixins.CreateModelMixin,
mixins.DestroyModelMixin,
mixins.RetrieveModelMixin,
Expand All @@ -293,9 +310,7 @@ def get_permissions(self):
else:
permission_classes = [
markdown_permissions.IsTokenResourceRouteObjectRelatedMarkdownDocument
& core_permissions.IsTokenInstructor
| markdown_permissions.IsTokenResourceRouteObjectRelatedMarkdownDocument
& core_permissions.IsTokenAdmin
& (core_permissions.IsTokenInstructor | core_permissions.IsTokenAdmin)
| IsRelatedMarkdownDocumentPlaylistOrOrganizationAdmin
]
return [permission() for permission in permission_classes]
Expand All @@ -306,7 +321,8 @@ def get_queryset(self):
"""
if self.request.resource:
return MarkdownImage.objects.filter(
markdown_document__id=self.request.resource.id,
markdown_document__id=self.get_related_markdown_document_id(),
markdown_document__playlist__id=self.request.resource.id,
)
return MarkdownImage.objects.all()

Expand Down
46 changes: 13 additions & 33 deletions src/backend/marsha/markdown/permissions.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
from rest_framework import permissions

from marsha.core import models
from marsha.markdown.models import MarkdownDocument


def _is_organization_admin(user_id, markdown_document_id):
Expand Down Expand Up @@ -32,26 +33,25 @@ def _is_playlist_admin(user_id, markdown_document_id):
).exists()


class IsTokenResourceRouteObjectRelatedResource(permissions.BasePermission):
class IsTokenResourceRouteObjectRelatedMarkdownDocument(permissions.BasePermission):
"""
Base permission class for JWT Tokens related to a resource object linked to a resource.
Base permission class for JWT Tokens related to a resource object linked to a
Markdown document.
These permissions grant access to users authenticated with a resource JWT token built from a
resource.
These permissions grants access to users authenticated with a JWT token built from a
resource ie related to a TokenUser as defined in `rest_framework_simplejwt`.
"""

linked_resource_attribute = ""

def has_permission(self, request, view):
"""
Allow the request if the JWT resource matches the resource
related to the object in the url.
Allow the request if the JWT resource matches the Markdown document related to the object
in the url.
Parameters
----------
request : Type[rest_framework.request.Request]
request : Type[django.http.request.HttpRequest]
The request that holds the authenticated user
view : Type[rest_framework.viewsets or rest_framework.views]
view : Type[restframework.viewsets or restframework.views]
The API view for which permissions are being checked
Returns
Expand All @@ -62,29 +62,9 @@ def has_permission(self, request, view):
if not request.resource:
return False

try:
return (
str(
getattr(
view.get_related_object(),
self.linked_resource_attribute,
).id
)
== request.resource.id
)
except ObjectDoesNotExist:
return False


class IsTokenResourceRouteObjectRelatedMarkdownDocument(
IsTokenResourceRouteObjectRelatedResource
):
"""
Base permission class for JWT Tokens related to a resource object
linked to a Markdown document.
"""

linked_resource_attribute = "markdown_document"
return MarkdownDocument.objects.filter(
pk=view.get_related_markdown_document_id(), playlist_id=request.resource.id
).exists()


class IsMarkdownDocumentPlaylistOrOrganizationAdmin(permissions.BasePermission):
Expand Down
9 changes: 2 additions & 7 deletions src/backend/marsha/markdown/serializers.py
Original file line number Diff line number Diff line change
Expand Up @@ -64,14 +64,9 @@ def create(self, validated_data):
The "validated_data" dictionary is returned after modification.
"""
# resource here is a Markdown document
resource = self.context["request"].resource
markdown_document_id = self.context["request"].data.get("markdown_document")
markdown_document_id = self.context["view"].get_related_markdown_document_id()
if not validated_data.get("markdown_document_id"):
if resource:
validated_data["markdown_document_id"] = resource.id
elif markdown_document_id:
validated_data["markdown_document_id"] = markdown_document_id
validated_data["markdown_document_id"] = markdown_document_id

return super().create(validated_data)

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,7 @@ def test_api_document_create_student(self):
markdown_document = MarkdownDocumentFactory()

jwt_token = StudentLtiTokenFactory(
resource=markdown_document,
resource=markdown_document.playlist,
permissions__can_update=True,
)

Expand All @@ -60,7 +60,9 @@ def test_api_document_create_instructor(self):
"""An instructor should not be able to create a Markdown document."""
markdown_document = MarkdownDocumentFactory()

jwt_token = InstructorOrAdminLtiTokenFactory(resource=markdown_document)
jwt_token = InstructorOrAdminLtiTokenFactory(
resource=markdown_document.playlist
)

response = self.client.post(
"/api/markdown-documents/",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,7 @@ def test_api_document_delete_student(self):
markdown_document = MarkdownDocumentFactory()

jwt_token = StudentLtiTokenFactory(
resource=markdown_document,
resource=markdown_document.playlist,
permissions__can_update=True,
)

Expand All @@ -52,7 +52,9 @@ def test_api_document_delete_instructor(self):
"""An instructor should not be able to create a Markdown document."""
markdown_document = MarkdownDocumentFactory()

jwt_token = InstructorOrAdminLtiTokenFactory(resource=markdown_document)
jwt_token = InstructorOrAdminLtiTokenFactory(
resource=markdown_document.playlist
)

response = self.client.delete(
f"/api/markdown-documents/{markdown_document.pk}/",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,7 @@ def test_api_document_fetch_list_student(self):
markdown_document = MarkdownDocumentFactory()

jwt_token = StudentLtiTokenFactory(
resource=markdown_document,
resource=markdown_document.playlist,
permissions__can_update=True,
)

Expand All @@ -49,7 +49,9 @@ def test_api_document_fetch_list_instructor(self):
"""An instrustor should not be able to fetch a Markdown document list."""
markdown_document = MarkdownDocumentFactory()

jwt_token = InstructorOrAdminLtiTokenFactory(resource=markdown_document)
jwt_token = InstructorOrAdminLtiTokenFactory(
resource=markdown_document.playlist
)

response = self.client.get(
"/api/markdown-documents/", HTTP_AUTHORIZATION=f"Bearer {jwt_token}"
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,7 @@ def test_api_document_render_latex_student(self):
markdown_document = MarkdownDocumentFactory()

jwt_token = StudentLtiTokenFactory(
resource=markdown_document,
resource=markdown_document.playlist,
permissions__can_update=True,
)

Expand All @@ -48,7 +48,9 @@ def test_api_document_render_latex_instructor(self):
"""An instructor should be able to render LaTeX content."""
markdown_document = MarkdownDocumentFactory(is_draft=True)

jwt_token = InstructorOrAdminLtiTokenFactory(resource=markdown_document)
jwt_token = InstructorOrAdminLtiTokenFactory(
resource=markdown_document.playlist
)

response = self.client.post(
f"/api/markdown-documents/{markdown_document.pk}/latex-rendering/",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -43,7 +43,7 @@ def test_api_document_fetch_student(self):
markdown_document = MarkdownDocumentFactory()

jwt_token = StudentLtiTokenFactory(
resource=markdown_document,
resource=markdown_document.playlist,
permissions__can_update=True,
)

Expand All @@ -69,7 +69,9 @@ def test_api_document_fetch_instructor(self):
translations__rendered_content="<h1>Heading1</h1>\n<p>Some content</p>",
)

jwt_token = InstructorOrAdminLtiTokenFactory(resource=markdown_document)
jwt_token = InstructorOrAdminLtiTokenFactory(
resource=markdown_document.playlist
)

response = self.client.get(
f"/api/markdown-documents/{markdown_document.pk}/",
Expand Down Expand Up @@ -130,7 +132,7 @@ def test_api_document_fetch_instructor_read_only(self):
markdown_document = MarkdownDocumentFactory()

jwt_token = InstructorOrAdminLtiTokenFactory(
resource=markdown_document,
resource=markdown_document.playlist,
permissions__can_update=False,
)

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -48,7 +48,7 @@ def test_api_document_update_student(self):
markdown_document = MarkdownDocumentFactory()

jwt_token = StudentLtiTokenFactory(
resource=markdown_document,
resource=markdown_document.playlist,
permissions__can_update=True,
)
data = {"title": "new title"}
Expand All @@ -66,7 +66,7 @@ def test_api_document_update_instructor_read_only(self):
markdown_document = MarkdownDocumentFactory()

jwt_token = InstructorOrAdminLtiTokenFactory(
resource=markdown_document,
resource=markdown_document.playlist,
permissions__can_update=False,
)
data = {"title": "new title"}
Expand Down Expand Up @@ -99,7 +99,9 @@ def test_api_document_update_instructor(self):
"""An instructor should be able to update a Markdown document."""
markdown_document = MarkdownDocumentFactory(is_draft=True)

jwt_token = InstructorOrAdminLtiTokenFactory(resource=markdown_document)
jwt_token = InstructorOrAdminLtiTokenFactory(
resource=markdown_document.playlist
)

data = {"is_draft": False}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,7 @@ def test_api_document_translation_update_student(self):
markdown_document = MarkdownDocumentFactory()

jwt_token = StudentLtiTokenFactory(
resource=markdown_document,
resource=markdown_document.playlist,
permissions__can_update=True,
)

Expand All @@ -53,7 +53,9 @@ def test_api_document_translation_update_instructor(self):
"""An instructor should be able to update a Markdown document translated content."""
markdown_document = MarkdownDocumentFactory(is_draft=True)

jwt_token = InstructorOrAdminLtiTokenFactory(resource=markdown_document)
jwt_token = InstructorOrAdminLtiTokenFactory(
resource=markdown_document.playlist
)

data = {
"language_code": "en",
Expand Down
Loading

0 comments on commit 198299f

Please sign in to comment.