mirror of
https://github.com/pacnpal/thrillwiki_django_no_react.git
synced 2025-12-27 17:27:01 -05:00
feat: Add blog, media, and support apps, implement ride credits and image API, and remove toplist feature.
This commit is contained in:
0
backend/apps/blog/__init__.py
Normal file
0
backend/apps/blog/__init__.py
Normal file
6
backend/apps/blog/apps.py
Normal file
6
backend/apps/blog/apps.py
Normal file
@@ -0,0 +1,6 @@
|
||||
from django.apps import AppConfig
|
||||
|
||||
class BlogConfig(AppConfig):
|
||||
default_auto_field = "django.db.models.BigAutoField"
|
||||
name = "apps.blog"
|
||||
verbose_name = "Blog"
|
||||
69
backend/apps/blog/migrations/0001_initial.py
Normal file
69
backend/apps/blog/migrations/0001_initial.py
Normal file
@@ -0,0 +1,69 @@
|
||||
# Generated by Django 5.1.6 on 2025-12-26 14:32
|
||||
|
||||
import django.db.models.deletion
|
||||
from django.conf import settings
|
||||
from django.db import migrations, models
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
initial = True
|
||||
|
||||
dependencies = [
|
||||
("django_cloudflareimages_toolkit", "0001_initial"),
|
||||
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.CreateModel(
|
||||
name="Tag",
|
||||
fields=[
|
||||
("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")),
|
||||
("created_at", models.DateTimeField(auto_now_add=True)),
|
||||
("updated_at", models.DateTimeField(auto_now=True)),
|
||||
("slug", models.SlugField(help_text="URL-friendly identifier", max_length=200, unique=True)),
|
||||
("name", models.CharField(max_length=50, unique=True)),
|
||||
],
|
||||
options={
|
||||
"ordering": ["name"],
|
||||
},
|
||||
),
|
||||
migrations.CreateModel(
|
||||
name="Post",
|
||||
fields=[
|
||||
("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")),
|
||||
("created_at", models.DateTimeField(auto_now_add=True)),
|
||||
("updated_at", models.DateTimeField(auto_now=True)),
|
||||
("name", models.CharField(help_text="Name of the object", max_length=200)),
|
||||
("slug", models.SlugField(help_text="URL-friendly identifier", max_length=200, unique=True)),
|
||||
("title", models.CharField(max_length=255)),
|
||||
("content", models.TextField(help_text="Markdown content supported")),
|
||||
("excerpt", models.TextField(blank=True, help_text="Short summary for lists")),
|
||||
("published_at", models.DateTimeField(blank=True, db_index=True, null=True)),
|
||||
("is_published", models.BooleanField(db_index=True, default=False)),
|
||||
(
|
||||
"author",
|
||||
models.ForeignKey(
|
||||
on_delete=django.db.models.deletion.CASCADE,
|
||||
related_name="blog_posts",
|
||||
to=settings.AUTH_USER_MODEL,
|
||||
),
|
||||
),
|
||||
(
|
||||
"image",
|
||||
models.ForeignKey(
|
||||
blank=True,
|
||||
help_text="Featured image",
|
||||
null=True,
|
||||
on_delete=django.db.models.deletion.SET_NULL,
|
||||
related_name="blog_posts",
|
||||
to="django_cloudflareimages_toolkit.cloudflareimage",
|
||||
),
|
||||
),
|
||||
("tags", models.ManyToManyField(blank=True, related_name="posts", to="blog.tag")),
|
||||
],
|
||||
options={
|
||||
"ordering": ["-published_at", "-created_at"],
|
||||
},
|
||||
),
|
||||
]
|
||||
0
backend/apps/blog/migrations/__init__.py
Normal file
0
backend/apps/blog/migrations/__init__.py
Normal file
45
backend/apps/blog/models.py
Normal file
45
backend/apps/blog/models.py
Normal file
@@ -0,0 +1,45 @@
|
||||
from django.db import models
|
||||
from django.conf import settings
|
||||
from apps.core.models import SluggedModel
|
||||
# Using string reference for CloudflareImage
|
||||
from django_cloudflareimages_toolkit.models import CloudflareImage
|
||||
|
||||
class Tag(SluggedModel):
|
||||
name = models.CharField(max_length=50, unique=True)
|
||||
|
||||
class Meta:
|
||||
ordering = ["name"]
|
||||
|
||||
def __str__(self):
|
||||
return self.name
|
||||
|
||||
class Post(SluggedModel):
|
||||
title = models.CharField(max_length=255)
|
||||
content = models.TextField(help_text="Markdown content supported")
|
||||
excerpt = models.TextField(blank=True, help_text="Short summary for lists")
|
||||
|
||||
image = models.ForeignKey(
|
||||
CloudflareImage,
|
||||
on_delete=models.SET_NULL,
|
||||
null=True,
|
||||
blank=True,
|
||||
related_name="blog_posts",
|
||||
help_text="Featured image"
|
||||
)
|
||||
|
||||
author = models.ForeignKey(
|
||||
settings.AUTH_USER_MODEL,
|
||||
on_delete=models.CASCADE,
|
||||
related_name="blog_posts"
|
||||
)
|
||||
|
||||
published_at = models.DateTimeField(null=True, blank=True, db_index=True)
|
||||
is_published = models.BooleanField(default=False, db_index=True)
|
||||
|
||||
tags = models.ManyToManyField(Tag, blank=True, related_name="posts")
|
||||
|
||||
class Meta:
|
||||
ordering = ["-published_at", "-created_at"]
|
||||
|
||||
def __str__(self):
|
||||
return self.title
|
||||
60
backend/apps/blog/serializers.py
Normal file
60
backend/apps/blog/serializers.py
Normal file
@@ -0,0 +1,60 @@
|
||||
from rest_framework import serializers
|
||||
from .models import Post, Tag
|
||||
from apps.accounts.serializers import UserSerializer
|
||||
from apps.media.serializers import CloudflareImageSerializer
|
||||
from django_cloudflareimages_toolkit.models import CloudflareImage
|
||||
|
||||
class TagSerializer(serializers.ModelSerializer):
|
||||
class Meta:
|
||||
model = Tag
|
||||
fields = ["id", "name", "slug"]
|
||||
|
||||
class PostListSerializer(serializers.ModelSerializer):
|
||||
"""Lighter serializer for lists"""
|
||||
author = UserSerializer(read_only=True)
|
||||
tags = TagSerializer(many=True, read_only=True)
|
||||
image = CloudflareImageSerializer(read_only=True)
|
||||
|
||||
class Meta:
|
||||
model = Post
|
||||
fields = [
|
||||
"id",
|
||||
"title",
|
||||
"slug",
|
||||
"excerpt",
|
||||
"image",
|
||||
"author",
|
||||
"published_at",
|
||||
"tags",
|
||||
]
|
||||
|
||||
class PostDetailSerializer(serializers.ModelSerializer):
|
||||
author = UserSerializer(read_only=True)
|
||||
tags = TagSerializer(many=True, read_only=True)
|
||||
image = CloudflareImageSerializer(read_only=True)
|
||||
image_id = serializers.PrimaryKeyRelatedField(
|
||||
queryset=CloudflareImage.objects.all(),
|
||||
source='image',
|
||||
write_only=True,
|
||||
required=False,
|
||||
allow_null=True
|
||||
)
|
||||
|
||||
class Meta:
|
||||
model = Post
|
||||
fields = [
|
||||
"id",
|
||||
"title",
|
||||
"slug",
|
||||
"content",
|
||||
"excerpt",
|
||||
"image",
|
||||
"image_id",
|
||||
"author",
|
||||
"published_at",
|
||||
"is_published",
|
||||
"tags",
|
||||
"created_at",
|
||||
"updated_at",
|
||||
]
|
||||
read_only_fields = ["id", "slug", "created_at", "updated_at", "author"]
|
||||
11
backend/apps/blog/urls.py
Normal file
11
backend/apps/blog/urls.py
Normal file
@@ -0,0 +1,11 @@
|
||||
from django.urls import path, include
|
||||
from rest_framework.routers import DefaultRouter
|
||||
from .views import PostViewSet, TagViewSet
|
||||
|
||||
router = DefaultRouter()
|
||||
router.register(r"posts", PostViewSet, basename="post")
|
||||
router.register(r"tags", TagViewSet, basename="tag")
|
||||
|
||||
urlpatterns = [
|
||||
path("", include(router.urls)),
|
||||
]
|
||||
42
backend/apps/blog/views.py
Normal file
42
backend/apps/blog/views.py
Normal file
@@ -0,0 +1,42 @@
|
||||
from rest_framework import viewsets, permissions, filters
|
||||
from django_filters.rest_framework import DjangoFilterBackend
|
||||
from django.utils import timezone
|
||||
from .models import Post, Tag
|
||||
from .serializers import PostListSerializer, PostDetailSerializer, TagSerializer
|
||||
from apps.core.permissions import IsStaffOrReadOnly
|
||||
|
||||
class TagViewSet(viewsets.ReadOnlyModelViewSet):
|
||||
queryset = Tag.objects.all()
|
||||
serializer_class = TagSerializer
|
||||
permission_classes = [permissions.AllowAny]
|
||||
filter_backends = [filters.SearchFilter]
|
||||
search_fields = ["name"]
|
||||
pagination_class = None # Tags are usually few
|
||||
|
||||
class PostViewSet(viewsets.ModelViewSet):
|
||||
"""
|
||||
Public API: Read Only (unless staff).
|
||||
Only published posts unless staff.
|
||||
"""
|
||||
permission_classes = [IsStaffOrReadOnly]
|
||||
filter_backends = [DjangoFilterBackend, filters.SearchFilter, filters.OrderingFilter]
|
||||
search_fields = ["title", "excerpt", "content"]
|
||||
filterset_fields = ["tags__slug", "is_published"]
|
||||
ordering_fields = ["published_at", "created_at"]
|
||||
ordering = ["-published_at"]
|
||||
lookup_field = "slug"
|
||||
|
||||
def get_queryset(self):
|
||||
qs = Post.objects.all()
|
||||
# If not staff, filter only published and past posts
|
||||
if not self.request.user.is_staff:
|
||||
qs = qs.filter(is_published=True, published_at__lte=timezone.now())
|
||||
return qs.select_related("author", "image").prefetch_related("tags")
|
||||
|
||||
def get_serializer_class(self):
|
||||
if self.action == "list":
|
||||
return PostListSerializer
|
||||
return PostDetailSerializer
|
||||
|
||||
def perform_create(self, serializer):
|
||||
serializer.save(author=self.request.user)
|
||||
Reference in New Issue
Block a user