manga
This commit is contained in:
parent
4c3f4c6640
commit
b4ab7055af
@ -142,7 +142,7 @@ class Media(SoftDeleteModel):
|
|||||||
name = models.CharField(max_length=255, null=False)
|
name = models.CharField(max_length=255, null=False)
|
||||||
media = models.FileField(
|
media = models.FileField(
|
||||||
max_length=255,
|
max_length=255,
|
||||||
upload_to="uploads/media/",
|
upload_to="uploads/media/contents/",
|
||||||
validators=[validate_file_size],
|
validators=[validate_file_size],
|
||||||
null=False, blank=False)
|
null=False, blank=False)
|
||||||
|
|
||||||
|
|||||||
@ -2,7 +2,6 @@ from django.forms import ImageField, FileField
|
|||||||
from rest_framework import serializers
|
from rest_framework import serializers
|
||||||
from content import models
|
from content import models
|
||||||
|
|
||||||
|
|
||||||
class ContentSerializer(serializers.ModelSerializer):
|
class ContentSerializer(serializers.ModelSerializer):
|
||||||
featured_image = ImageField(max_length=255, allow_empty_file=False)
|
featured_image = ImageField(max_length=255, allow_empty_file=False)
|
||||||
|
|
||||||
|
|||||||
0
freekake_api/entertainment/__init__.py
Normal file
0
freekake_api/entertainment/__init__.py
Normal file
3
freekake_api/entertainment/admin.py
Normal file
3
freekake_api/entertainment/admin.py
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
from django.contrib import admin
|
||||||
|
|
||||||
|
# Register your models here.
|
||||||
6
freekake_api/entertainment/apps.py
Normal file
6
freekake_api/entertainment/apps.py
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
from django.apps import AppConfig
|
||||||
|
|
||||||
|
|
||||||
|
class EntertainmentsConfig(AppConfig):
|
||||||
|
default_auto_field = 'django.db.models.BigAutoField'
|
||||||
|
name = 'entertainment'
|
||||||
101
freekake_api/entertainment/models.py
Normal file
101
freekake_api/entertainment/models.py
Normal file
@ -0,0 +1,101 @@
|
|||||||
|
from django.db import models
|
||||||
|
from django.core.exceptions import ValidationError
|
||||||
|
from django_softdelete.models import SoftDeleteModel
|
||||||
|
from PIL import Image
|
||||||
|
|
||||||
|
def validate_image_size(image):
|
||||||
|
if image.size > 1 * 1024 * 1024:
|
||||||
|
raise ValidationError("File size exceeds 1MB.")
|
||||||
|
|
||||||
|
def validate_image_ext(image):
|
||||||
|
allowed_extensions = ['png', 'jpg', 'jpeg']
|
||||||
|
ext = image.name.split('.')[-1].lower()
|
||||||
|
if ext not in allowed_extensions:
|
||||||
|
raise ValidationError("Invalid file type. Only PNG or JPEG allowed.")
|
||||||
|
|
||||||
|
def validate_image(image):
|
||||||
|
try:
|
||||||
|
img = Image.open(image)
|
||||||
|
img.verify()
|
||||||
|
except Exception:
|
||||||
|
raise ValidationError("Invalid image file.")
|
||||||
|
|
||||||
|
def validate_file_size(image):
|
||||||
|
if image.size > 1 * 1024 * 1024:
|
||||||
|
raise ValidationError("File size exceeds 1MB.")
|
||||||
|
|
||||||
|
class Manga(SoftDeleteModel):
|
||||||
|
|
||||||
|
MANGA_GENRE_CHOICES = [
|
||||||
|
('komedi','Komedi'),
|
||||||
|
('petualangan', 'Petualangan'),
|
||||||
|
('fiksi-ilmiah', 'Fiksi Ilmiah'),
|
||||||
|
('fantasi', 'Fantasi'),
|
||||||
|
('slice-of-life', 'Kehidupan Sehari-hari'),
|
||||||
|
]
|
||||||
|
|
||||||
|
MANGA_STATUS_CHOICES = [
|
||||||
|
('draft', 'Draf'),
|
||||||
|
('published', 'Dipublikasikan'),
|
||||||
|
('archived', 'Diarsipkan'),
|
||||||
|
]
|
||||||
|
|
||||||
|
title = models.CharField(max_length=255, null=False)
|
||||||
|
slug = models.SlugField(max_length=255, unique=True)
|
||||||
|
|
||||||
|
featured_image = models.ImageField(
|
||||||
|
max_length=255,
|
||||||
|
upload_to="uploads/entertainments/",
|
||||||
|
validators=[validate_image_size, validate_image_ext, validate_image],
|
||||||
|
null=False, blank=False)
|
||||||
|
|
||||||
|
genre = models.CharField(max_length=50, choices=MANGA_GENRE_CHOICES)
|
||||||
|
|
||||||
|
synopsis = models.CharField(max_length=1000, null=True, blank=True)
|
||||||
|
|
||||||
|
coin = models.IntegerField()
|
||||||
|
|
||||||
|
color = models.CharField(max_length=7, null=True, blank=True)
|
||||||
|
|
||||||
|
status = models.CharField(max_length=50, choices=MANGA_STATUS_CHOICES, default='draft')
|
||||||
|
|
||||||
|
created_at = models.DateTimeField(auto_now_add=True)
|
||||||
|
updated_at = models.DateTimeField(auto_now=True)
|
||||||
|
|
||||||
|
posted_at = models.DateTimeField(null=True, blank=True)
|
||||||
|
archived_at = models.DateTimeField(null=True, blank=True)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return self.title
|
||||||
|
|
||||||
|
class MangaChapter(SoftDeleteModel):
|
||||||
|
|
||||||
|
manga = models.ForeignKey(Manga, related_name='chapters', on_delete=models.CASCADE)
|
||||||
|
|
||||||
|
chapter = models.PositiveSmallIntegerField(null=False)
|
||||||
|
|
||||||
|
status = models.CharField(max_length=50, choices=Manga.MANGA_STATUS_CHOICES, default='draft')
|
||||||
|
|
||||||
|
created_at = models.DateTimeField(auto_now_add=True)
|
||||||
|
updated_at = models.DateTimeField(auto_now=True)
|
||||||
|
|
||||||
|
posted_at = models.DateTimeField(null=True, blank=True)
|
||||||
|
archived_at = models.DateTimeField(null=True, blank=True)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return self.manga.title + " - Chapter: " + str(self.chapter)
|
||||||
|
|
||||||
|
class MangaChapterPage(SoftDeleteModel):
|
||||||
|
|
||||||
|
chapter = models.ForeignKey(MangaChapter, related_name='media', on_delete=models.CASCADE)
|
||||||
|
|
||||||
|
order = models.PositiveSmallIntegerField(null=False)
|
||||||
|
|
||||||
|
media = models.FileField(
|
||||||
|
max_length=255,
|
||||||
|
upload_to="uploads/entertainments/manga/chapters/media/",
|
||||||
|
validators=[validate_file_size],
|
||||||
|
null=False, blank=False)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return self.chapter.manga.title + " - Chapter: " + self.chapter.chapter + " - Page: " + str(self.order)
|
||||||
103
freekake_api/entertainment/serializers.py
Normal file
103
freekake_api/entertainment/serializers.py
Normal file
@ -0,0 +1,103 @@
|
|||||||
|
from django.forms import ImageField, FileField
|
||||||
|
from rest_framework import serializers
|
||||||
|
from entertainment import models
|
||||||
|
|
||||||
|
class MangaSerializer(serializers.ModelSerializer):
|
||||||
|
featured_image = ImageField(max_length=255, allow_empty_file=False)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = models.Manga
|
||||||
|
fields = [
|
||||||
|
'id',
|
||||||
|
'title',
|
||||||
|
'slug',
|
||||||
|
'featured_image',
|
||||||
|
'genre',
|
||||||
|
'synopsis',
|
||||||
|
'coin',
|
||||||
|
'color',
|
||||||
|
'status',
|
||||||
|
'posted_at',
|
||||||
|
'archived_at',
|
||||||
|
]
|
||||||
|
|
||||||
|
def __init__(self, *args, **kwargs):
|
||||||
|
super().__init__(*args, **kwargs)
|
||||||
|
request_method = self.context.get('request').method if self.context.get('request') else None
|
||||||
|
|
||||||
|
if request_method in ['PUT', 'PATCH']:
|
||||||
|
self.fields['featured_image'].required = False
|
||||||
|
|
||||||
|
class MangaDetailSerializer(serializers.ModelSerializer):
|
||||||
|
featured_image = ImageField(max_length=255, allow_empty_file=False)
|
||||||
|
genre_display = serializers.SerializerMethodField()
|
||||||
|
status_display = serializers.SerializerMethodField()
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = models.Manga
|
||||||
|
fields = [
|
||||||
|
'id',
|
||||||
|
'title',
|
||||||
|
'slug',
|
||||||
|
'featured_image',
|
||||||
|
'genre',
|
||||||
|
'synopsis',
|
||||||
|
'coin',
|
||||||
|
'color',
|
||||||
|
'status',
|
||||||
|
'posted_at',
|
||||||
|
'archived_at',
|
||||||
|
'genre_display',
|
||||||
|
'status_display',
|
||||||
|
]
|
||||||
|
|
||||||
|
def validate(self, data):
|
||||||
|
genre = data.get('genre')
|
||||||
|
|
||||||
|
allowed_genres = dict(models.Manga.MANGA_GENRE_CHOICES).get(genre, [])
|
||||||
|
valid_genre_keys = [key for key, _ in allowed_genres]
|
||||||
|
|
||||||
|
if genre not in valid_genre_keys:
|
||||||
|
raise serializers.ValidationError({'topic': f'Genre "{genre}" is not valid for genre "{genre}"'})
|
||||||
|
|
||||||
|
return data
|
||||||
|
|
||||||
|
def get_genre_display(self, obj):
|
||||||
|
return dict(models.Manga.MANGA_GENRE_CHOICES).get(obj.genre)
|
||||||
|
|
||||||
|
def get_status_display(self, obj):
|
||||||
|
return dict(models.Manga.CONTENT_STATUS_CHOICES).get(obj.status, obj.status)
|
||||||
|
|
||||||
|
class MangaChapterSerializer(serializers.ModelSerializer):
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = models.MangaChapter
|
||||||
|
fields = ['id', 'manga', 'chapter', 'status', 'posted_at', 'archived_at']
|
||||||
|
|
||||||
|
class MangaChapterDetailSerializer(serializers.ModelSerializer):
|
||||||
|
manga = MangaSerializer(read_only=True)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = models.MangaChapter
|
||||||
|
fields = ['id', 'manga', 'chapter', 'status', 'posted_at', 'archived_at']
|
||||||
|
|
||||||
|
class MangaChapterPageSerializer(serializers.ModelSerializer):
|
||||||
|
media = FileField(max_length=255, allow_empty_file=False)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = models.MangaChapterPage
|
||||||
|
fields = ['id', 'chapter', 'order', 'media']
|
||||||
|
|
||||||
|
def __init__(self, *args, **kwargs):
|
||||||
|
super().__init__(*args, **kwargs)
|
||||||
|
request_method = self.context.get('request').method if self.context.get('request') else None
|
||||||
|
|
||||||
|
if request_method in ['PUT', 'PATCH']:
|
||||||
|
self.fields['media'].required = False
|
||||||
|
|
||||||
|
class MangaChapterPageDetailSerializer(serializers.ModelSerializer):
|
||||||
|
chapter = MangaChapterDetailSerializer(read_only=True)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = models.MangaChapterPage
|
||||||
|
fields = ['id', 'chapter', 'order', 'media']
|
||||||
3
freekake_api/entertainment/tests.py
Normal file
3
freekake_api/entertainment/tests.py
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
from django.test import TestCase
|
||||||
|
|
||||||
|
# Create your tests here.
|
||||||
16
freekake_api/entertainment/urls.py
Normal file
16
freekake_api/entertainment/urls.py
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
from django.urls import path
|
||||||
|
from entertainment import views
|
||||||
|
|
||||||
|
urlpatterns = [
|
||||||
|
path('manga/', views.MangaList.as_view()),
|
||||||
|
path('manga/<int:pk>/', views.MangaDetail.as_view()),
|
||||||
|
|
||||||
|
path('genres/', views.MangaGenreChoices.as_view()),
|
||||||
|
path('statuses/', views.MangaStatusChoices.as_view()),
|
||||||
|
|
||||||
|
path('manga/<int:manga_id>/chapters/', views.MangaChapterList.as_view()),
|
||||||
|
path('manga/<int:manga_id>/chapters/<int:pk>/', views.MangaChapterDetail.as_view()),
|
||||||
|
|
||||||
|
path('manga/<int:manga_id>/chapters/<int:chapter_id>/pages/', views.MangaChapterPageList.as_view()),
|
||||||
|
path('manga/<int:manga_id>/chapters/<int:chapter_id>/pages/<int:pk>/', views.MangaChapterPageDetail.as_view()),
|
||||||
|
]
|
||||||
91
freekake_api/entertainment/views.py
Normal file
91
freekake_api/entertainment/views.py
Normal file
@ -0,0 +1,91 @@
|
|||||||
|
from rest_framework.response import Response
|
||||||
|
from rest_framework import generics, filters, views
|
||||||
|
from django_filters.rest_framework import DjangoFilterBackend, FilterSet, CharFilter, ChoiceFilter
|
||||||
|
|
||||||
|
from entertainment import models, serializers
|
||||||
|
|
||||||
|
class MangaList(generics.ListCreateAPIView):
|
||||||
|
queryset = models.Manga.objects.all()
|
||||||
|
serializer_class = serializers.MangaSerializer
|
||||||
|
filter_backends = [filters.SearchFilter, filters.OrderingFilter, DjangoFilterBackend]
|
||||||
|
search_fields = ['title']
|
||||||
|
filterset_fields = ['genre', 'status']
|
||||||
|
ordering_fields = '__all__'
|
||||||
|
|
||||||
|
def get_serializer_class(self):
|
||||||
|
if self.request.method in ['GET']:
|
||||||
|
return serializers.MangaDetailSerializer
|
||||||
|
return serializers.MangaSerializer
|
||||||
|
|
||||||
|
class MangaDetail(generics.RetrieveUpdateDestroyAPIView):
|
||||||
|
queryset = models.Manga.objects.all()
|
||||||
|
serializer_class = serializers.MangaDetailSerializer
|
||||||
|
|
||||||
|
def get_serializer_class(self):
|
||||||
|
if self.request.method in ['GET']:
|
||||||
|
return serializers.MangaDetailSerializer
|
||||||
|
return serializers.MangaSerializer
|
||||||
|
|
||||||
|
class MangaGenreChoices(views.APIView):
|
||||||
|
def get(self, request, *args, **kwargs):
|
||||||
|
return Response({
|
||||||
|
"count": len(models.Manga.MANGA_GENRE_CHOICES),
|
||||||
|
"results": [
|
||||||
|
{"value": choice[0], "label": choice[1]}
|
||||||
|
for choice in models.Manga.MANGA_GENRE_CHOICES
|
||||||
|
]
|
||||||
|
})
|
||||||
|
|
||||||
|
class MangaStatusChoices(views.APIView):
|
||||||
|
def get(self, request, *args, **kwargs):
|
||||||
|
return Response({
|
||||||
|
"count": len(models.Manga.MANGA_STATUS_CHOICES),
|
||||||
|
"results": [
|
||||||
|
{"value": choice[0], "label": choice[1]}
|
||||||
|
for choice in models.Manga.MANGA_STATUS_CHOICES
|
||||||
|
]
|
||||||
|
})
|
||||||
|
|
||||||
|
class MangaChapterList(generics.ListCreateAPIView):
|
||||||
|
serializer_class = serializers.MangaChapterSerializer
|
||||||
|
ordering_fields = ['chapter']
|
||||||
|
|
||||||
|
def get_queryset(self):
|
||||||
|
manga_id = self.kwargs['manga_id']
|
||||||
|
return models.MangaChapter.objects.filter(manga_id=manga_id)
|
||||||
|
|
||||||
|
def get_serializer_class(self):
|
||||||
|
if self.request.method in ['GET']:
|
||||||
|
return serializers.MangaChapterDetailSerializer
|
||||||
|
return serializers.MangaChapterSerializer
|
||||||
|
|
||||||
|
class MangaChapterDetail(generics.RetrieveUpdateDestroyAPIView):
|
||||||
|
queryset = models.MangaChapter.objects.all()
|
||||||
|
serializer_class = serializers.MangaChapterDetailSerializer
|
||||||
|
|
||||||
|
def get_serializer_class(self):
|
||||||
|
if self.request.method in ['GET']:
|
||||||
|
return serializers.MangaChapterDetailSerializer
|
||||||
|
return serializers.MangaChapterSerializer
|
||||||
|
|
||||||
|
class MangaChapterPageList(generics.ListCreateAPIView):
|
||||||
|
serializer_class = serializers.MangaChapterPageSerializer
|
||||||
|
ordering_fields = ['order']
|
||||||
|
|
||||||
|
def get_queryset(self):
|
||||||
|
chapter_id = self.kwargs['chapter_id']
|
||||||
|
return models.MangaChapterPage.objects.filter(chapter_id=chapter_id)
|
||||||
|
|
||||||
|
def get_serializer_class(self):
|
||||||
|
if self.request.method in ['GET']:
|
||||||
|
return serializers.MangaChapterPageDetailSerializer
|
||||||
|
return serializers.MangaChapterPageSerializer
|
||||||
|
|
||||||
|
class MangaChapterPageDetail(generics.RetrieveUpdateDestroyAPIView):
|
||||||
|
queryset = models.MangaChapterPage.objects.all()
|
||||||
|
serializer_class = serializers.MangaChapterPageDetailSerializer
|
||||||
|
|
||||||
|
def get_serializer_class(self):
|
||||||
|
if self.request.method in ['GET']:
|
||||||
|
return serializers.MangaChapterPageDetailSerializer
|
||||||
|
return serializers.MangaChapterPageSerializer
|
||||||
@ -51,6 +51,7 @@ INSTALLED_APPS = [
|
|||||||
'core',
|
'core',
|
||||||
'content',
|
'content',
|
||||||
'character',
|
'character',
|
||||||
|
'entertainment',
|
||||||
]
|
]
|
||||||
|
|
||||||
MIDDLEWARE = [
|
MIDDLEWARE = [
|
||||||
@ -200,6 +201,9 @@ SIMPLE_JWT = {
|
|||||||
|
|
||||||
DJOSER = {
|
DJOSER = {
|
||||||
'TOKEN_MODEL': None,
|
'TOKEN_MODEL': None,
|
||||||
|
'SEND_ACTIVATION_EMAIL': True,
|
||||||
|
'ACTIVATION_URL': 'accounts/activation/{uid}/{token}/',
|
||||||
|
"SEND_CONFIRMATION_EMAIL": True,
|
||||||
}
|
}
|
||||||
|
|
||||||
SIMPLE_JWT = {
|
SIMPLE_JWT = {
|
||||||
@ -210,3 +214,8 @@ SIMPLE_JWT = {
|
|||||||
"ISSUER": 'freekake_api',
|
"ISSUER": 'freekake_api',
|
||||||
}
|
}
|
||||||
|
|
||||||
|
EMAIL_BACKEND = 'django.core.mail.backends.smtp.EmailBackend'
|
||||||
|
EMAIL_HOST = 'sandbox.smtp.mailtrap.io'
|
||||||
|
EMAIL_HOST_USER = 'a14db434093c11'
|
||||||
|
EMAIL_HOST_PASSWORD = '3524293f1ca830'
|
||||||
|
EMAIL_PORT = '2525'
|
||||||
|
|||||||
@ -30,4 +30,6 @@ urlpatterns = [
|
|||||||
path('core/', include('core.urls')),
|
path('core/', include('core.urls')),
|
||||||
path('content/', include('content.urls')),
|
path('content/', include('content.urls')),
|
||||||
path('character/', include('character.urls')),
|
path('character/', include('character.urls')),
|
||||||
|
path('entertainment/', include('entertainment.urls')),
|
||||||
|
|
||||||
]+ debug_toolbar_urls() + static(settings.MEDIA_URL, document_root=settings.MEDIA_ROOT)
|
]+ debug_toolbar_urls() + static(settings.MEDIA_URL, document_root=settings.MEDIA_ROOT)
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user