feat: add Exercise model and CRUD API for tutorial exercises

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-04-23 01:44:14 -06:00
parent c4ddfa6841
commit bd9bd84f2d
7 changed files with 143 additions and 11 deletions

View File

@@ -1,12 +1,15 @@
from account.decorators import super_admin_required
from utils.api import APIView, validate_serializer
from tutorial.models import Tutorial
from tutorial.models import Tutorial, Exercise
from tutorial.serializers import (
TutorialSerializer,
TutorialListSerializer,
CreateTutorialSerializer,
EditTutorialSerializer,
ExerciseSerializer,
CreateExerciseSerializer,
EditExerciseSerializer,
)
@@ -90,3 +93,51 @@ class TutorialVisibilityAPI(APIView):
tutorial.is_public = is_public
tutorial.save()
return self.success(TutorialSerializer(tutorial).data)
class ExerciseAdminAPI(APIView):
@validate_serializer(CreateExerciseSerializer)
@super_admin_required
def post(self, request):
data = request.data
try:
tutorial = Tutorial.objects.get(id=data["tutorial_id"])
except Tutorial.DoesNotExist:
return self.error("Tutorial does not exist")
exercise = Exercise.objects.create(
tutorial=tutorial,
type=data["type"],
data=data["data"],
order=data.get("order", 0),
)
return self.success(ExerciseSerializer(exercise).data)
@validate_serializer(EditExerciseSerializer)
@super_admin_required
def put(self, request):
data = request.data
try:
exercise = Exercise.objects.get(id=data["id"])
except Exercise.DoesNotExist:
return self.error("Exercise does not exist")
exercise.type = data["type"]
exercise.data = data["data"]
exercise.order = data.get("order", exercise.order)
exercise.save()
return self.success(ExerciseSerializer(exercise).data)
@super_admin_required
def get(self, request):
tutorial_id = request.GET.get("tutorial_id")
if not tutorial_id:
return self.error("tutorial_id is required")
exercises = Exercise.objects.filter(tutorial_id=tutorial_id)
return self.success(ExerciseSerializer(exercises, many=True).data)
@super_admin_required
def delete(self, request):
exercise_id = request.GET.get("id")
if not exercise_id:
return self.error("id is required")
Exercise.objects.filter(id=exercise_id).delete()
return self.success()

View File

@@ -1,7 +1,7 @@
from utils.api import APIView
from tutorial.models import Tutorial
from tutorial.serializers import TutorialSerializer
from tutorial.models import Tutorial, Exercise
from tutorial.serializers import TutorialSerializer, ExerciseSerializer
class TutorialAPI(APIView):
@@ -21,3 +21,16 @@ class TutorialTitlesAPI(APIView):
"id", "title"
)
return self.success(list(tutorials))
class ExerciseAPI(APIView):
def get(self, request):
tutorial_id = request.GET.get("tutorial_id")
if not tutorial_id:
return self.error("tutorial_id is required")
try:
tutorial = Tutorial.objects.get(id=tutorial_id, is_public=True)
except Tutorial.DoesNotExist:
return self.error("Tutorial does not exist")
exercises = Exercise.objects.filter(tutorial=tutorial)
return self.success(ExerciseSerializer(exercises, many=True).data)