Skip to content

Commit de946eb

Browse files
feat: new REST API for units in content libraries
1 parent a85c2b8 commit de946eb

File tree

5 files changed

+65
-2
lines changed

5 files changed

+65
-2
lines changed

cms/envs/common.py

+1
Original file line numberDiff line numberDiff line change
@@ -1876,6 +1876,7 @@
18761876
"openedx_learning.apps.authoring.components",
18771877
"openedx_learning.apps.authoring.contents",
18781878
"openedx_learning.apps.authoring.publishing",
1879+
"openedx_learning.apps.authoring.units",
18791880
]
18801881

18811882

lms/envs/common.py

+1
Original file line numberDiff line numberDiff line change
@@ -3373,6 +3373,7 @@ def _make_locale_paths(settings): # pylint: disable=missing-function-docstring
33733373
"openedx_learning.apps.authoring.components",
33743374
"openedx_learning.apps.authoring.contents",
33753375
"openedx_learning.apps.authoring.publishing",
3376+
"openedx_learning.apps.authoring.units",
33763377
]
33773378

33783379

openedx/core/djangoapps/content_libraries/serializers.py

+22
Original file line numberDiff line numberDiff line change
@@ -230,6 +230,28 @@ class LibraryXBlockStaticFilesSerializer(serializers.Serializer):
230230
files = LibraryXBlockStaticFileSerializer(many=True)
231231

232232

233+
class LibraryContainerMetadataSerializer(serializers.Serializer):
234+
"""
235+
Serializer for LibraryXBlockMetadata
236+
"""
237+
id = serializers.CharField(source="container_key", read_only=True)
238+
container_type = serializers.CharField(source="container_key.container_type")
239+
title = serializers.CharField(read_only=True)
240+
last_published = serializers.DateTimeField(format=DATETIME_FORMAT, read_only=True)
241+
published_by = serializers.CharField(read_only=True)
242+
last_draft_created = serializers.DateTimeField(format=DATETIME_FORMAT, read_only=True)
243+
last_draft_created_by = serializers.CharField(read_only=True)
244+
has_unpublished_changes = serializers.BooleanField(read_only=True)
245+
created = serializers.DateTimeField(format=DATETIME_FORMAT, read_only=True)
246+
modified = serializers.DateTimeField(format=DATETIME_FORMAT, read_only=True)
247+
tags_count = serializers.IntegerField(read_only=True)
248+
collections = CollectionMetadataSerializer(many=True, required=False, read_only=True)
249+
250+
# When creating a new container in a library, the slug becomes the ID part of
251+
# the definition key and usage key:
252+
slug = serializers.CharField(write_only=True)
253+
254+
233255
class ContentLibraryBlockImportTaskSerializer(serializers.ModelSerializer):
234256
"""
235257
Serializer for a Content Library block import task.

openedx/core/djangoapps/content_libraries/urls.py

+12-2
Original file line numberDiff line numberDiff line change
@@ -39,7 +39,9 @@
3939
path('block_types/', views.LibraryBlockTypesView.as_view()),
4040
# Get the list of XBlocks in this library, or add a new one:
4141
path('blocks/', views.LibraryBlocksView.as_view()),
42-
# Commit (POST) or revert (DELETE) all pending changes to this library:
42+
# Get the list of XBlocks in this library, or add a new one:
43+
path('containers/', views.LibraryContainersView.as_view()),
44+
# Publish (POST) or revert (DELETE) all pending changes to this library:
4345
path('commit/', views.LibraryCommitView.as_view()),
4446
# Get the list of users/groups who have permission to view/edit/administer this library:
4547
path('team/', views.LibraryTeamView.as_view()),
@@ -57,6 +59,7 @@
5759
path('blocks/<str:usage_key_str>/', include([
5860
# Get metadata about a specific XBlock in this library, or delete the block:
5961
path('', views.LibraryBlockView.as_view()),
62+
# Restore a soft-deleted block
6063
path('restore/', views.LibraryBlockRestore.as_view()),
6164
# Update collections for a given component
6265
path('collections/', views.LibraryBlockCollectionsView.as_view(), name='update-collections'),
@@ -69,7 +72,14 @@
6972
path('assets/<path:file_path>', views.LibraryBlockAssetView.as_view()),
7073
path('publish/', views.LibraryBlockPublishView.as_view()),
7174
# Future: discard changes for just this one block
72-
# Future: set a block's tags (tags are stored in a Tag bundle and linked in)
75+
])),
76+
# Containers are Sections, Subsections, and Units
77+
path('containers/<usage_v2:container_key>/', include([
78+
# Get metadata about a specific container in this library, or delete the container:
79+
# path('', views.LibraryContainerView.as_view()),
80+
# Update collections for a given container
81+
# path('collections/', views.LibraryContainerCollectionsView.as_view(), name='update-collections-ct'),
82+
# path('publish/', views.LibraryContainerPublishView.as_view()),
7383
])),
7484
re_path(r'^lti/1.3/', include([
7585
path('login/', views.LtiToolLoginView.as_view(), name='lti-login'),

openedx/core/djangoapps/content_libraries/views.py

+29
Original file line numberDiff line numberDiff line change
@@ -114,6 +114,7 @@
114114
ContentLibraryPermissionSerializer,
115115
ContentLibraryUpdateSerializer,
116116
ContentLibraryComponentCollectionsUpdateSerializer,
117+
LibraryContainerMetadataSerializer,
117118
LibraryXBlockCreationSerializer,
118119
LibraryXBlockMetadataSerializer,
119120
LibraryXBlockTypeSerializer,
@@ -877,6 +878,34 @@ def post(self, request, usage_key_str):
877878
return Response({})
878879

879880

881+
@method_decorator(non_atomic_requests, name="dispatch")
882+
@view_auth_classes()
883+
class LibraryContainersView(GenericAPIView):
884+
"""
885+
Views to work with Containers in a specific content library.
886+
"""
887+
serializer_class = LibraryContainerMetadataSerializer
888+
889+
@convert_exceptions
890+
@swagger_auto_schema(
891+
request_body=LibraryContainerMetadataSerializer,
892+
responses={200: LibraryContainerMetadataSerializer}
893+
)
894+
def post(self, request, lib_key_str):
895+
"""
896+
Add a new XBlock to this content library
897+
"""
898+
library_key = LibraryLocatorV2.from_string(lib_key_str)
899+
api.require_permission_for_library_key(library_key, request.user, permissions.CAN_EDIT_THIS_CONTENT_LIBRARY)
900+
serializer = LibraryContainerMetadataSerializer(data=request.data)
901+
serializer.is_valid(raise_exception=True)
902+
903+
# Create a new regular top-level block:
904+
result = api.create_container(...)
905+
906+
return Response(LibraryContainerMetadataSerializer(result).data)
907+
908+
880909
@method_decorator(non_atomic_requests, name="dispatch")
881910
@view_auth_classes()
882911
class LibraryImportTaskViewSet(GenericViewSet):

0 commit comments

Comments
 (0)