- PATCH /jobs/{job_id} endpoint for updating title and cost_tracker_project_id
- cost_tracker_project_id exposed on JobResponse (GET /jobs/{id})
- Inline project ID field in QCDetail and FinalDetail — saved via PATCH
- "AI Cost Dashboard" link in UserList header
- cost_tracker_project_id added to Job type and JobUpdateRequest schema
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
149 lines
3.8 KiB
Python
149 lines
3.8 KiB
Python
from typing import Any, Literal, Optional, Union
|
|
|
|
from pydantic import BaseModel
|
|
|
|
from ..models.job import (
|
|
AccessibleVideoProgressItem,
|
|
JobStatus,
|
|
LangOutput,
|
|
RequestedOutputs,
|
|
Review,
|
|
TTSPreferences,
|
|
)
|
|
from ..schemas.accessible_video import AccessibleVideoMethod
|
|
|
|
|
|
class JobResponse(BaseModel):
|
|
id: str
|
|
client_id: Optional[str] = None # ID of the user who created the job
|
|
title: str
|
|
status: JobStatus
|
|
source: dict[str, Any]
|
|
requested_outputs: RequestedOutputs
|
|
review: Review
|
|
outputs: Optional[dict[str, LangOutput]] = None
|
|
accessible_video_progress: Optional[dict[str, AccessibleVideoProgressItem]] = None
|
|
created_at: Optional[str] = None
|
|
updated_at: Optional[str] = None
|
|
created_by_name: Optional[str] = None # User's full_name who created the job
|
|
cost_tracker_project_id: Optional[str] = None
|
|
|
|
|
|
class JobListResponse(BaseModel):
|
|
jobs: list[JobResponse]
|
|
total: int
|
|
page: int
|
|
size: int
|
|
|
|
|
|
class JobCreateRequest(BaseModel):
|
|
title: str
|
|
requested_outputs: RequestedOutputs
|
|
|
|
|
|
class JobUpdateRequest(BaseModel):
|
|
title: Optional[str] = None
|
|
review_notes: Optional[str] = None
|
|
cost_tracker_project_id: Optional[str] = None
|
|
|
|
|
|
class ApproveEnglishRequest(BaseModel):
|
|
notes: Optional[str] = None
|
|
|
|
|
|
class ApproveSourceRequest(BaseModel):
|
|
"""Request to approve source language content (works for any language)"""
|
|
notes: Optional[str] = None
|
|
tts_preferences: Optional[TTSPreferences] = None # Override TTS voice settings
|
|
accessible_video_method: Optional[AccessibleVideoMethod] = None # User-selected method for accessible video
|
|
|
|
|
|
class UpdateTTSPreferencesRequest(BaseModel):
|
|
"""Request to update TTS preferences and regenerate all TTS segments"""
|
|
tts_preferences: TTSPreferences
|
|
|
|
|
|
class RejectJobRequest(BaseModel):
|
|
notes: str
|
|
|
|
|
|
class CompleteJobRequest(BaseModel):
|
|
notes: Optional[str] = None
|
|
|
|
|
|
class VttUpdateRequest(BaseModel):
|
|
captions_vtt: Optional[str] = None
|
|
audio_description_vtt: Optional[str] = None
|
|
language: Optional[str] = None # If None, defaults to source language
|
|
|
|
|
|
class VttTimingAdjustRequest(BaseModel):
|
|
offset_seconds: float
|
|
language: str = "en"
|
|
adjust_captions: bool = True
|
|
adjust_audio_description: bool = True
|
|
|
|
|
|
class JobDownloadsResponse(BaseModel):
|
|
downloads: dict[str, Union[dict[str, str], str]] # language -> {file_type: signed_url} OR source_video -> signed_url
|
|
|
|
|
|
class VttContentResponse(BaseModel):
|
|
captions_vtt: Optional[str] = None
|
|
audio_description_vtt: Optional[str] = None
|
|
retimed_captions_vtt: Optional[str] = None # Re-timed captions for accessible videos
|
|
|
|
|
|
class AssetValidationResponse(BaseModel):
|
|
is_valid: bool
|
|
errors: list[str]
|
|
warnings: list[str] = []
|
|
|
|
|
|
class JobDeleteResponse(BaseModel):
|
|
message: str
|
|
|
|
|
|
class BulkDeleteRequest(BaseModel):
|
|
job_ids: list[str]
|
|
|
|
|
|
class BulkDeleteResponse(BaseModel):
|
|
deleted_count: int
|
|
total_requested: int
|
|
errors: list[str]
|
|
|
|
|
|
class BulkApproveRequest(BaseModel):
|
|
"""Request to bulk approve multiple jobs with optional settings"""
|
|
job_ids: list[str]
|
|
notes: Optional[str] = None
|
|
accessible_video_method: Optional[AccessibleVideoMethod] = None # Method for accessible video
|
|
tts_preferences: Optional[TTSPreferences] = None
|
|
|
|
|
|
class BulkApproveResponse(BaseModel):
|
|
"""Response for bulk approval operation"""
|
|
approved_count: int
|
|
total_requested: int
|
|
errors: list[str]
|
|
|
|
|
|
class ReturnToQCRequest(BaseModel):
|
|
notes: str
|
|
|
|
|
|
class BulkReturnToQCRequest(BaseModel):
|
|
job_ids: list[str]
|
|
notes: str
|
|
|
|
|
|
class BulkReturnToQCResponse(BaseModel):
|
|
returned_count: int
|
|
total_requested: int
|
|
errors: list[str]
|
|
|
|
|
|
class BulkDownloadRequest(BaseModel):
|
|
"""Request to download multiple jobs as a single zip file"""
|
|
job_ids: list[str]
|