Video pipeline (script→assets→storyboard→video→stitch): - robust split_script_into_segments (4 non-empty scenes), scene-aware storyboard/video prompts - link VideoSegment→ScriptSegment + storyboard-frame reference image (graceful text fallback) - idempotent poll_video_segment (no double-charge on repeated polling) - threaded export (no Celery worker needed) + poll-export endpoint - run_export_job rewritten to filter_complex: per-clip trim, xfade transitions, subtitle burn-in (Pillow PNG overlay; this ffmpeg lacks libass), BGM mix - upload-video-segment / upload-bgm / save-timeline endpoints - serializers embed asset preview URLs (beat assets pagination); Pillow added to requirements Also includes prior uncommitted backend work: account preferences/sessions, billing trend, product/asset endpoints, accounts 0002 migration. Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
89 lines
3.1 KiB
Python
89 lines
3.1 KiB
Python
from rest_framework import serializers
|
|
|
|
from apps.billing.models import CreditAccount
|
|
|
|
from .models import LoginSession, Team, TeamMember, User, UserPreference
|
|
|
|
|
|
class UserSerializer(serializers.ModelSerializer):
|
|
class Meta:
|
|
model = User
|
|
fields = ["id", "username", "first_name", "last_name", "email", "phone", "avatar_url", "status"]
|
|
read_only_fields = ["id", "status"]
|
|
|
|
|
|
class UserPreferenceSerializer(serializers.ModelSerializer):
|
|
class Meta:
|
|
model = UserPreference
|
|
fields = ["notify", "two_factor_enabled", "creation_defaults", "display", "updated_at"]
|
|
read_only_fields = ["updated_at"]
|
|
|
|
|
|
class LoginSessionSerializer(serializers.ModelSerializer):
|
|
is_current = serializers.SerializerMethodField()
|
|
|
|
class Meta:
|
|
model = LoginSession
|
|
fields = ["id", "user_agent", "ip_address", "last_seen_at", "created_at", "is_current"]
|
|
read_only_fields = fields
|
|
|
|
def get_is_current(self, obj) -> bool:
|
|
ctx = self.context or {}
|
|
return bool(obj.ip_address and obj.ip_address == ctx.get("current_ip") and obj.user_agent == ctx.get("current_ua"))
|
|
|
|
|
|
class TeamSerializer(serializers.ModelSerializer):
|
|
class Meta:
|
|
model = Team
|
|
fields = ["id", "name", "status", "owner", "created_at", "updated_at"]
|
|
read_only_fields = ["id", "status", "owner", "created_at", "updated_at"]
|
|
|
|
|
|
class TeamMemberSerializer(serializers.ModelSerializer):
|
|
user = UserSerializer(read_only=True)
|
|
|
|
class Meta:
|
|
model = TeamMember
|
|
fields = ["id", "team", "user", "role", "status", "monthly_credit_limit"]
|
|
read_only_fields = ["id", "team", "user", "status"]
|
|
|
|
|
|
class RegisterSerializer(serializers.Serializer):
|
|
username = serializers.CharField(max_length=150)
|
|
password = serializers.CharField(min_length=8, write_only=True)
|
|
email = serializers.EmailField(required=False, allow_blank=True)
|
|
team_name = serializers.CharField(max_length=128, required=False, allow_blank=True)
|
|
|
|
def validate_username(self, value):
|
|
if User.objects.filter(username=value).exists():
|
|
raise serializers.ValidationError("username already exists")
|
|
return value
|
|
|
|
def create(self, validated_data):
|
|
from decimal import Decimal
|
|
|
|
from django.conf import settings
|
|
from django.db import transaction
|
|
|
|
with transaction.atomic():
|
|
user = User.objects.create_user(
|
|
username=validated_data["username"],
|
|
password=validated_data["password"],
|
|
email=validated_data.get("email", ""),
|
|
)
|
|
team = Team.objects.create(
|
|
name=validated_data.get("team_name") or f"{user.username}'s Team",
|
|
owner=user,
|
|
)
|
|
TeamMember.objects.create(team=team, user=user, role=TeamMember.Role.OWNER)
|
|
CreditAccount.objects.create(
|
|
team=team,
|
|
balance=Decimal(str(settings.DEFAULT_TRIAL_CREDITS)),
|
|
)
|
|
return {"user": user, "team": team}
|
|
|
|
|
|
class LoginSerializer(serializers.Serializer):
|
|
username = serializers.CharField()
|
|
password = serializers.CharField(write_only=True)
|