Merge pull request 'add-schedule-support' (#1) from add-schedule-support into main
Reviewed-on: #1
This commit is contained in:
commit
7e38ee8533
9 changed files with 216 additions and 14 deletions
4
.idea/misc.xml
generated
4
.idea/misc.xml
generated
|
@ -3,7 +3,7 @@
|
||||||
<component name="Black">
|
<component name="Black">
|
||||||
<option name="enabledOnReformat" value="true" />
|
<option name="enabledOnReformat" value="true" />
|
||||||
<option name="enabledOnSave" value="true" />
|
<option name="enabledOnSave" value="true" />
|
||||||
<option name="sdkName" value="Python 3.13 (movie-night-py)" />
|
<option name="sdkName" value="Python 3.13 (api)" />
|
||||||
</component>
|
</component>
|
||||||
<component name="ProjectRootManager" version="2" project-jdk-name="Remote Python 3.12.7 Docker (<none>:<none>)" project-jdk-type="Python SDK" />
|
<component name="ProjectRootManager" version="2" project-jdk-name="Python 3.13 (api)" project-jdk-type="Python SDK" />
|
||||||
</project>
|
</project>
|
3
.idea/movie-night-py.iml
generated
3
.idea/movie-night-py.iml
generated
|
@ -15,8 +15,9 @@
|
||||||
<component name="NewModuleRootManager">
|
<component name="NewModuleRootManager">
|
||||||
<content url="file://$MODULE_DIR$">
|
<content url="file://$MODULE_DIR$">
|
||||||
<excludeFolder url="file://$MODULE_DIR$/.venv" />
|
<excludeFolder url="file://$MODULE_DIR$/.venv" />
|
||||||
|
<excludeFolder url="file://$MODULE_DIR$/.venv1" />
|
||||||
</content>
|
</content>
|
||||||
<orderEntry type="jdk" jdkName="Remote Python 3.12.7 Docker (<none>:<none>)" jdkType="Python SDK" />
|
<orderEntry type="jdk" jdkName="Python 3.13 (api)" jdkType="Python SDK" />
|
||||||
<orderEntry type="sourceFolder" forTests="false" />
|
<orderEntry type="sourceFolder" forTests="false" />
|
||||||
</component>
|
</component>
|
||||||
<component name="PyDocumentationSettings">
|
<component name="PyDocumentationSettings">
|
||||||
|
|
|
@ -0,0 +1,50 @@
|
||||||
|
# Generated by Django 5.1.4 on 2025-04-08 03:39
|
||||||
|
|
||||||
|
import django.db.models.deletion
|
||||||
|
from django.conf import settings
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('movie_manager', '0003_movie_added_by'),
|
||||||
|
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='movie',
|
||||||
|
name='last_watched',
|
||||||
|
field=models.DateTimeField(blank=True, null=True),
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='Showing',
|
||||||
|
fields=[
|
||||||
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('public', models.BooleanField(default=False)),
|
||||||
|
('showtime', models.DateTimeField()),
|
||||||
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||||
|
('updated_at', models.DateTimeField(auto_now=True)),
|
||||||
|
('deleted_at', models.DateTimeField(blank=True, null=True)),
|
||||||
|
('movie', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='movie_manager.movie')),
|
||||||
|
('owner', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL)),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'ordering': ['showtime'],
|
||||||
|
},
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='Schedule',
|
||||||
|
fields=[
|
||||||
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('name', models.CharField(max_length=100)),
|
||||||
|
('public', models.BooleanField(default=False)),
|
||||||
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||||
|
('updated_at', models.DateTimeField(auto_now=True)),
|
||||||
|
('deleted_at', models.DateTimeField(blank=True, null=True)),
|
||||||
|
('owner', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL)),
|
||||||
|
('showings', models.ManyToManyField(to='movie_manager.showing')),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
]
|
18
movie_manager/migrations/0005_showing_slug.py
Normal file
18
movie_manager/migrations/0005_showing_slug.py
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
# Generated by Django 5.1.4 on 2025-04-08 03:45
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('movie_manager', '0004_alter_movie_last_watched_showing_schedule'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='showing',
|
||||||
|
name='slug',
|
||||||
|
field=models.SlugField(default='', max_length=100),
|
||||||
|
),
|
||||||
|
]
|
|
@ -0,0 +1,22 @@
|
||||||
|
# Generated by Django 5.1.4 on 2025-04-08 04:03
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('movie_manager', '0005_showing_slug'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name='showing',
|
||||||
|
name='slug',
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='schedule',
|
||||||
|
name='slug',
|
||||||
|
field=models.SlugField(default='', max_length=100),
|
||||||
|
),
|
||||||
|
]
|
|
@ -1,7 +1,6 @@
|
||||||
from django.db import models
|
from django.db import models
|
||||||
from django.contrib.auth.models import User
|
from django.contrib.auth.models import User
|
||||||
|
|
||||||
# Create your models here.
|
|
||||||
class Movie(models.Model):
|
class Movie(models.Model):
|
||||||
title = models.CharField(max_length=100)
|
title = models.CharField(max_length=100)
|
||||||
imdb_id = models.CharField(max_length=100)
|
imdb_id = models.CharField(max_length=100)
|
||||||
|
@ -12,7 +11,7 @@ class Movie(models.Model):
|
||||||
actors = models.CharField(max_length=500)
|
actors = models.CharField(max_length=500)
|
||||||
plot = models.CharField(max_length=500)
|
plot = models.CharField(max_length=500)
|
||||||
poster = models.CharField(max_length=500)
|
poster = models.CharField(max_length=500)
|
||||||
last_watched = models.DateTimeField()
|
last_watched = models.DateTimeField(null=True, blank=True)
|
||||||
added_by = models.ForeignKey(User, on_delete=models.CASCADE)
|
added_by = models.ForeignKey(User, on_delete=models.CASCADE)
|
||||||
created_at = models.DateTimeField(auto_now_add=True)
|
created_at = models.DateTimeField(auto_now_add=True)
|
||||||
updated_at = models.DateTimeField(auto_now=True)
|
updated_at = models.DateTimeField(auto_now=True)
|
||||||
|
@ -39,3 +38,25 @@ class MovieList(models.Model):
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.name
|
return self.name
|
||||||
|
|
||||||
|
class Schedule(models.Model):
|
||||||
|
name = models.CharField(max_length=100)
|
||||||
|
owner = models.ForeignKey(User, on_delete=models.CASCADE)
|
||||||
|
public = models.BooleanField(default=False)
|
||||||
|
showings = models.ManyToManyField("Showing")
|
||||||
|
slug = models.SlugField(max_length=100, default="")
|
||||||
|
created_at = models.DateTimeField(auto_now_add=True)
|
||||||
|
updated_at = models.DateTimeField(auto_now=True)
|
||||||
|
deleted_at = models.DateTimeField(null=True, blank=True)
|
||||||
|
|
||||||
|
class Showing(models.Model):
|
||||||
|
movie = models.ForeignKey(Movie, on_delete=models.CASCADE)
|
||||||
|
owner = models.ForeignKey(User, on_delete=models.CASCADE)
|
||||||
|
public = models.BooleanField(default=False)
|
||||||
|
showtime = models.DateTimeField()
|
||||||
|
created_at = models.DateTimeField(auto_now_add=True)
|
||||||
|
updated_at = models.DateTimeField(auto_now=True)
|
||||||
|
deleted_at = models.DateTimeField(null=True, blank=True)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
ordering = ["showtime"]
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
from itertools import count
|
from gunicorn.config import User
|
||||||
|
|
||||||
from rest_framework import serializers
|
from rest_framework import serializers
|
||||||
from movie_manager.models import Movie, MovieList
|
from movie_manager.models import Movie, MovieList, Schedule, Showing
|
||||||
|
|
||||||
|
|
||||||
class MovieSerializer(serializers.ModelSerializer):
|
class MovieSerializer(serializers.ModelSerializer):
|
||||||
class Meta:
|
class Meta:
|
||||||
|
@ -20,3 +20,26 @@ class MovieListSerializer(serializers.ModelSerializer):
|
||||||
|
|
||||||
def get_movie_count(self, obj):
|
def get_movie_count(self, obj):
|
||||||
return len(obj.movies.all())
|
return len(obj.movies.all())
|
||||||
|
|
||||||
|
class UserSerializer(serializers.Serializer):
|
||||||
|
class Meta:
|
||||||
|
model = User
|
||||||
|
fields = ["id", "username"]
|
||||||
|
|
||||||
|
|
||||||
|
class ShowingSerializer(serializers.ModelSerializer):
|
||||||
|
movie = MovieSerializer(read_only=True)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = Showing
|
||||||
|
fields = ["public", "showtime", "movie", "owner"]
|
||||||
|
|
||||||
|
|
||||||
|
class ScheduleSerializer(serializers.ModelSerializer):
|
||||||
|
name = serializers.CharField(read_only=True)
|
||||||
|
showings = ShowingSerializer(read_only=True, many=True)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = Schedule
|
||||||
|
fields = ["name", "owner","public","slug", "showings"]
|
||||||
|
|
||||||
|
|
|
@ -1,17 +1,18 @@
|
||||||
from django.http import HttpResponse, JsonResponse
|
import datetime
|
||||||
|
|
||||||
|
from django.http import JsonResponse
|
||||||
from django.contrib.auth.models import User
|
from django.contrib.auth.models import User
|
||||||
from rest_framework import permissions, viewsets
|
from rest_framework import permissions, viewsets
|
||||||
from knox.auth import TokenAuthentication
|
from knox.auth import TokenAuthentication
|
||||||
from rest_framework.decorators import action
|
from rest_framework.decorators import action
|
||||||
from rest_framework.exceptions import NotFound
|
from rest_framework.exceptions import NotFound
|
||||||
|
|
||||||
from movie_manager.models import Movie, MovieList
|
from movie_manager.models import Movie, MovieList, Schedule, Showing
|
||||||
from movie_manager.serializers import MovieListSerializer, MovieSerializer
|
from movie_manager.serializers import MovieListSerializer, MovieSerializer, ScheduleSerializer, ShowingSerializer
|
||||||
|
|
||||||
|
|
||||||
# Create your views here.
|
# Create your views here.
|
||||||
class MovieViewset(viewsets.ModelViewSet):
|
class MovieViewset(viewsets.ModelViewSet):
|
||||||
fields = '__all__'
|
|
||||||
queryset = Movie.objects.all().order_by("title")
|
queryset = Movie.objects.all().order_by("title")
|
||||||
authentication_classes = [TokenAuthentication]
|
authentication_classes = [TokenAuthentication]
|
||||||
permission_classes = [permissions.IsAuthenticated]
|
permission_classes = [permissions.IsAuthenticated]
|
||||||
|
@ -19,7 +20,6 @@ class MovieViewset(viewsets.ModelViewSet):
|
||||||
serializer_class = MovieSerializer
|
serializer_class = MovieSerializer
|
||||||
|
|
||||||
class MovieListViewset(viewsets.ModelViewSet):
|
class MovieListViewset(viewsets.ModelViewSet):
|
||||||
fields = '__all__'
|
|
||||||
queryset = MovieList.objects.all().order_by("name")
|
queryset = MovieList.objects.all().order_by("name")
|
||||||
authentication_classes = [TokenAuthentication]
|
authentication_classes = [TokenAuthentication]
|
||||||
permission_classes = [permissions.IsAuthenticated]
|
permission_classes = [permissions.IsAuthenticated]
|
||||||
|
@ -66,8 +66,73 @@ class MovieListViewset(viewsets.ModelViewSet):
|
||||||
return JsonResponse(MovieListSerializer(movie_list).data)
|
return JsonResponse(MovieListSerializer(movie_list).data)
|
||||||
|
|
||||||
def remove_movie(self, request, pk=None, movie_id=None, *args, **kwargs):
|
def remove_movie(self, request, pk=None, movie_id=None, *args, **kwargs):
|
||||||
movie_list = MovieList.objects.get(pk=pk)
|
|
||||||
movie = Movie.objects.get(pk=movie_id)
|
movie = Movie.objects.get(pk=movie_id)
|
||||||
|
|
||||||
|
movie_list = MovieList.objects.get(pk=pk)
|
||||||
movie_list.movies.remove(movie)
|
movie_list.movies.remove(movie)
|
||||||
|
|
||||||
return JsonResponse(MovieListSerializer(movie_list).data)
|
return JsonResponse(MovieListSerializer(movie_list).data)
|
||||||
|
|
||||||
|
class ScheduleViewset(viewsets.ModelViewSet):
|
||||||
|
queryset = Schedule.objects.all().order_by("name")
|
||||||
|
authentication_classes = [TokenAuthentication]
|
||||||
|
permission_classes = [permissions.IsAuthenticated]
|
||||||
|
|
||||||
|
serializer_class = ScheduleSerializer
|
||||||
|
|
||||||
|
def retrieve(self, request, pk=None, *args, **kwargs):
|
||||||
|
# Get the schedule instance
|
||||||
|
instance = self.get_object()
|
||||||
|
today = datetime.datetime.now()
|
||||||
|
|
||||||
|
upcoming_showings = instance.showings.filter(showtime__gte=today)
|
||||||
|
|
||||||
|
# Create a serialized response
|
||||||
|
serializer = self.get_serializer(instance)
|
||||||
|
data = serializer.data
|
||||||
|
|
||||||
|
# Replace all showings with only future showings
|
||||||
|
data['showings'] = ShowingSerializer(upcoming_showings, many=True).data
|
||||||
|
|
||||||
|
|
||||||
|
if request.GET.get('past_showings') == 'true':
|
||||||
|
past_showings = instance.showings.filter(showtime__lt=today)
|
||||||
|
|
||||||
|
# Add both to the response
|
||||||
|
data['past_showings'] = [
|
||||||
|
{'id': showing.id, 'showtime': showing.showtime.isoformat(), "movie": MovieSerializer(showing.movie).data}
|
||||||
|
for showing in past_showings
|
||||||
|
]
|
||||||
|
else:
|
||||||
|
data['past_showings'] = []
|
||||||
|
|
||||||
|
return JsonResponse(data)
|
||||||
|
|
||||||
|
|
||||||
|
class ShowingViewset(viewsets.ModelViewSet):
|
||||||
|
queryset = Showing.objects.all().order_by("showtime")
|
||||||
|
authentication_classes = [TokenAuthentication]
|
||||||
|
permission_classes = [permissions.IsAuthenticated]
|
||||||
|
|
||||||
|
serializer_class = ShowingSerializer
|
||||||
|
|
||||||
|
def create(self, request, *args, **kwargs):
|
||||||
|
movie_id = request.data.get('movie')
|
||||||
|
movie = Movie.objects.get(pk=movie_id)
|
||||||
|
|
||||||
|
schedule_id = request.data.get('schedule')
|
||||||
|
schedule = Schedule.objects.get(pk=schedule_id)
|
||||||
|
|
||||||
|
showing = Showing.objects.create(
|
||||||
|
movie=movie,
|
||||||
|
schedule=schedule,
|
||||||
|
showtime=request.data.get("showtime"),
|
||||||
|
public=request.data.get("public"),
|
||||||
|
owner=User.objects.get(pk=request.data.get("owner"))
|
||||||
|
)
|
||||||
|
|
||||||
|
schedule.showings.add(showing)
|
||||||
|
|
||||||
|
return JsonResponse(ShowingSerializer(showing).data)
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -19,6 +19,8 @@ router.register(r"api/users", user_views.UserViewSet)
|
||||||
router.register(r"api/groups", user_views.GroupViewSet)
|
router.register(r"api/groups", user_views.GroupViewSet)
|
||||||
router.register(r"api/movies", movie_views.MovieViewset)
|
router.register(r"api/movies", movie_views.MovieViewset)
|
||||||
router.register(r"api/lists", movie_views.MovieListViewset)
|
router.register(r"api/lists", movie_views.MovieListViewset)
|
||||||
|
router.register(r"api/schedules", movie_views.ScheduleViewset)
|
||||||
|
router.register(r"api/showings", movie_views.ShowingViewset)
|
||||||
|
|
||||||
urlpatterns = [
|
urlpatterns = [
|
||||||
path("", include(router.urls)),
|
path("", include(router.urls)),
|
||||||
|
|
Loading…
Add table
Reference in a new issue