Skip to content

Commit

Permalink
feat: Implement ObjectTag retrieve REST API
Browse files Browse the repository at this point in the history
Retrieve ObjectTags for given Object IDs, and optionally filter by
taxonomy.
  • Loading branch information
yusuf-musleh committed Aug 14, 2023
1 parent 9f7d5c1 commit 41e7034
Show file tree
Hide file tree
Showing 7 changed files with 373 additions and 11 deletions.
3 changes: 2 additions & 1 deletion openedx_tagging/core/tagging/api.py
Original file line number Diff line number Diff line change
Expand Up @@ -96,13 +96,14 @@ def resync_object_tags(object_tags: QuerySet = None) -> int:


def get_object_tags(
object_id: str, taxonomy: Taxonomy = None
object_id: str, taxonomy_id: str = None
) -> QuerySet:
"""
Returns a Queryset of object tags for a given object.
Pass taxonomy to limit the returned object_tags to a specific taxonomy.
"""
taxonomy = get_taxonomy(taxonomy_id)
ObjectTagClass = taxonomy.object_tag_class if taxonomy else ObjectTag
tags = (
ObjectTagClass.objects.filter(
Expand Down
14 changes: 13 additions & 1 deletion openedx_tagging/core/tagging/rest_api/v1/permissions.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
"""
Taxonomy permissions
Tagging permissions
"""

from rest_framework.permissions import DjangoObjectPermissions
Expand All @@ -15,3 +15,15 @@ class TaxonomyObjectPermissions(DjangoObjectPermissions):
"PATCH": ["%(app_label)s.change_%(model_name)s"],
"DELETE": ["%(app_label)s.delete_%(model_name)s"],
}


class ObjectTagObjectPermissions(DjangoObjectPermissions):
perms_map = {
"GET": ["%(app_label)s.view_%(model_name)s"],
"OPTIONS": [],
"HEAD": ["%(app_label)s.view_%(model_name)s"],
"POST": ["%(app_label)s.add_%(model_name)s"],
"PUT": ["%(app_label)s.change_%(model_name)s"],
"PATCH": ["%(app_label)s.change_%(model_name)s"],
"DELETE": ["%(app_label)s.delete_%(model_name)s"],
}
28 changes: 27 additions & 1 deletion openedx_tagging/core/tagging/rest_api/v1/serializers.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@

from rest_framework import serializers

from openedx_tagging.core.tagging.models import Taxonomy
from openedx_tagging.core.tagging.models import Taxonomy, ObjectTag


class TaxonomyListQueryParamsSerializer(serializers.Serializer):
Expand All @@ -29,3 +29,29 @@ class Meta:
"system_defined",
"visible_to_authors",
]


class ObjectTagListQueryParamsSerializer(serializers.Serializer):
"""
Serializer for the query params for the ObjectTag GET view
"""

taxonomy = serializers.PrimaryKeyRelatedField(
queryset=Taxonomy.objects.all(), required=False
)


class ObjectTagSerializer(serializers.ModelSerializer):
"""
Serializer for the ObjectTag model.
"""

class Meta:
model = ObjectTag
fields = [
"name",
"value",
"taxonomy_id",
"tag_ref",
"is_valid",
]
1 change: 1 addition & 0 deletions openedx_tagging/core/tagging/rest_api/v1/urls.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,5 +10,6 @@

router = DefaultRouter()
router.register("taxonomies", views.TaxonomyView, basename="taxonomy")
router.register("object_tags", views.ObjectTagView, basename="object_tag")

urlpatterns = [path("", include(router.urls))]
86 changes: 83 additions & 3 deletions openedx_tagging/core/tagging/rest_api/v1/views.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,15 +2,23 @@
Tagging API Views
"""
from django.http import Http404
from rest_framework.viewsets import ModelViewSet
from rest_framework import status
from rest_framework.viewsets import ModelViewSet, ReadOnlyModelViewSet
from rest_framework.response import Response

from ...api import (
create_taxonomy,
get_taxonomy,
get_taxonomies,
get_object_tags,
)
from .permissions import TaxonomyObjectPermissions, ObjectTagObjectPermissions
from .serializers import (
TaxonomyListQueryParamsSerializer,
TaxonomySerializer,
ObjectTagListQueryParamsSerializer,
ObjectTagSerializer,
)
from .permissions import TaxonomyObjectPermissions
from .serializers import TaxonomyListQueryParamsSerializer, TaxonomySerializer


class TaxonomyView(ModelViewSet):
Expand Down Expand Up @@ -145,3 +153,75 @@ def perform_create(self, serializer):
Create a new taxonomy.
"""
serializer.instance = create_taxonomy(**serializer.validated_data)


class ObjectTagView(ReadOnlyModelViewSet):
"""
View to retrieve paginated ObjectTags for a provided Object ID (object_id).
**Retrieve Parameters**
* object_id (required): - The Object ID to retrieve ObjectTags for.
**Retrieve Query Parameters**
* taxonomy (optional) - PK of taxonomy to filter ObjectTags for.
* page (optional) - Page number of paginated results.
* page_size (optional) - Number of results included in each page.
**Retrieve Example Requests**
GET api/tagging/v1/object_tags/:object_id
GET api/tagging/v1/object_tags/:object_id?taxonomy=1
GET api/tagging/v1/object_tags/:object_id?taxonomy=1&page=2
GET api/tagging/v1/object_tags/:object_id?taxonomy=1&page=2&page_size=10
**Retrieve Query Returns**
* 200 - Success
* 400 - Invalid query parameter
* 403 - Permission denied
**Create Query Returns**
* 403 - Permission denied
* 405 - Method not allowed
**Update Query Returns**
* 403 - Permission denied
* 405 - Method not allowed
**Delete Query Returns**
* 403 - Permission denied
* 405 - Method not allowed
"""

serializer_class = ObjectTagSerializer
permission_classes = [ObjectTagObjectPermissions]
lookup_field = "object_id"

def get_queryset(self):
"""
Return a queryset of object tags for a given object.
If a taxonomy is passed in, object tags are limited to that taxonomy.
"""
object_id = self.kwargs.get("object_id")
query_params = ObjectTagListQueryParamsSerializer(
data=self.request.query_params.dict()
)
query_params.is_valid(raise_exception=True)
taxonomy_id = query_params.data.get("taxonomy", None)
return get_object_tags(object_id, taxonomy_id)

def retrieve(self, request, object_id=None):
"""
Retrieve ObjectTags that belong to a given object_id and
return paginated results.
Note: We override `retrieve` here instead of `list` because we are
passing in the Object ID (object_id) in the path (as opposed to passing
it in as a query_param) to retrieve the related ObjectTags.
By default retrieve would expect an ObjectTag ID to be passed in the
path and returns a it as a single result however that is not
behavior we want.
"""
object_tags = self.get_queryset()
paginated_object_tags = self.paginate_queryset(object_tags)
serializer = ObjectTagSerializer(paginated_object_tags, many=True)
return self.get_paginated_response(serializer.data)
8 changes: 4 additions & 4 deletions tests/openedx_tagging/core/tagging/test_api.py
Original file line number Diff line number Diff line change
Expand Up @@ -256,7 +256,7 @@ def test_tag_object(self):
assert (
list(
tagging_api.get_object_tags(
taxonomy=self.taxonomy,
taxonomy_id=self.taxonomy.pk,
object_id="biology101",
)
)
Expand Down Expand Up @@ -337,7 +337,7 @@ def test_tag_object_language_taxonomy(self):
assert (
list(
tagging_api.get_object_tags(
taxonomy=self.language_taxonomy,
taxonomy_id=self.language_taxonomy.pk,
object_id="biology101",
)
)
Expand Down Expand Up @@ -383,7 +383,7 @@ def test_tag_object_model_system_taxonomy(self):
assert (
list(
tagging_api.get_object_tags(
taxonomy=self.user_taxonomy,
taxonomy_id=self.user_taxonomy.pk,
object_id="biology101",
)
)
Expand Down Expand Up @@ -437,7 +437,7 @@ def test_get_object_tags(self):
assert list(
tagging_api.get_object_tags(
object_id="abc",
taxonomy=self.taxonomy,
taxonomy_id=self.taxonomy.pk,
)
) == [
beta,
Expand Down
Loading

0 comments on commit 41e7034

Please sign in to comment.