feat: Initialize frontend with React, Vite, and Tailwind CSS
- Set up main entry point for React application. - Create About, Home, NotFound, Privacy, and Terms pages with SEO support. - Implement API service for file uploads and task management. - Add global styles using Tailwind CSS. - Create utility functions for SEO and text processing. - Configure Vite for development and production builds. - Set up Nginx configuration for serving frontend and backend. - Add scripts for cleanup of expired files and sitemap generation. - Implement deployment script for production environment.
This commit is contained in:
70
backend/app/routes/video.py
Normal file
70
backend/app/routes/video.py
Normal file
@@ -0,0 +1,70 @@
|
||||
"""Video processing routes."""
|
||||
from flask import Blueprint, request, jsonify
|
||||
|
||||
from app.extensions import limiter
|
||||
from app.utils.file_validator import validate_file, FileValidationError
|
||||
from app.utils.sanitizer import generate_safe_path
|
||||
from app.tasks.video_tasks import create_gif_task
|
||||
|
||||
video_bp = Blueprint("video", __name__)
|
||||
|
||||
ALLOWED_VIDEO_TYPES = ["mp4", "webm"]
|
||||
|
||||
|
||||
@video_bp.route("/to-gif", methods=["POST"])
|
||||
@limiter.limit("5/minute")
|
||||
def video_to_gif_route():
|
||||
"""
|
||||
Convert a video clip to an animated GIF.
|
||||
|
||||
Accepts: multipart/form-data with:
|
||||
- 'file': Video file (MP4, WebM, max 50MB)
|
||||
- 'start_time' (optional): Start time in seconds (default: 0)
|
||||
- 'duration' (optional): Duration in seconds, max 15 (default: 5)
|
||||
- 'fps' (optional): Frames per second, max 20 (default: 10)
|
||||
- 'width' (optional): Output width, max 640 (default: 480)
|
||||
Returns: JSON with task_id for polling
|
||||
"""
|
||||
if "file" not in request.files:
|
||||
return jsonify({"error": "No file provided."}), 400
|
||||
|
||||
file = request.files["file"]
|
||||
|
||||
# Parse and validate parameters
|
||||
try:
|
||||
start_time = float(request.form.get("start_time", 0))
|
||||
duration = float(request.form.get("duration", 5))
|
||||
fps = int(request.form.get("fps", 10))
|
||||
width = int(request.form.get("width", 480))
|
||||
except (ValueError, TypeError):
|
||||
return jsonify({"error": "Invalid parameters. Must be numeric."}), 400
|
||||
|
||||
# Enforce limits
|
||||
if start_time < 0:
|
||||
return jsonify({"error": "Start time cannot be negative."}), 400
|
||||
if duration <= 0 or duration > 15:
|
||||
return jsonify({"error": "Duration must be between 0.5 and 15 seconds."}), 400
|
||||
if fps < 1 or fps > 20:
|
||||
return jsonify({"error": "FPS must be between 1 and 20."}), 400
|
||||
if width < 100 or width > 640:
|
||||
return jsonify({"error": "Width must be between 100 and 640 pixels."}), 400
|
||||
|
||||
try:
|
||||
original_filename, ext = validate_file(file, allowed_types=ALLOWED_VIDEO_TYPES)
|
||||
except FileValidationError as e:
|
||||
return jsonify({"error": e.message}), e.code
|
||||
|
||||
# Save file
|
||||
task_id, input_path = generate_safe_path(ext, folder_type="upload")
|
||||
file.save(input_path)
|
||||
|
||||
# Dispatch task
|
||||
task = create_gif_task.delay(
|
||||
input_path, task_id, original_filename,
|
||||
start_time, duration, fps, width,
|
||||
)
|
||||
|
||||
return jsonify({
|
||||
"task_id": task.id,
|
||||
"message": "GIF creation started. Poll /api/tasks/{task_id}/status for progress.",
|
||||
}), 202
|
||||
Reference in New Issue
Block a user