feat(music-lab): background.py — Pexels Video API + orchestrator video_loop 분기

This commit is contained in:
2026-05-09 13:13:42 +09:00
parent f0c0c18beb
commit e754fb30f5
3 changed files with 131 additions and 1 deletions

View File

@@ -0,0 +1,60 @@
"""Pexels Video API로 background loop 영상 받아오기 (video_loop 모드용)."""
import os
import logging
import httpx
from . import storage
logger = logging.getLogger("music-lab.background")
TIMEOUT_S = 60
async def fetch_video_loop(pipeline_id: int, keyword: str) -> dict:
"""Pexels Video API → 720p HD mp4 다운로드 → /app/data/videos/{id}/loop.mp4 저장.
반환: {"path": str | None, "used_fallback": bool, "error": str | None}
"""
api_key = os.getenv("PEXELS_API_KEY", "")
if not api_key:
return {"path": None, "used_fallback": True, "error": "PEXELS_API_KEY 미설정"}
out_dir = storage.pipeline_dir(pipeline_id)
out_path = os.path.join(out_dir, "loop.mp4")
try:
async with httpx.AsyncClient(timeout=TIMEOUT_S) as client:
resp = await client.get(
"https://api.pexels.com/videos/search",
headers={"Authorization": api_key},
params={"query": keyword or "ambient calm", "per_page": 5,
"orientation": "landscape"},
)
resp.raise_for_status()
data = resp.json()
videos = data.get("videos", [])
if not videos:
return {"path": None, "used_fallback": True,
"error": f"Pexels 결과 없음: {keyword}"}
# 720p/1080p HD 우선, 없으면 첫 번째 video file
chosen = None
for v in videos:
for f in v.get("video_files", []):
if f.get("quality") == "hd" and f.get("width") in (1280, 1920):
chosen = f
break
if chosen:
break
if not chosen:
chosen = videos[0]["video_files"][0]
video_url = chosen["link"]
vid_resp = await client.get(video_url)
vid_resp.raise_for_status()
with open(out_path, "wb") as f:
f.write(vid_resp.content)
return {"path": out_path, "used_fallback": False, "error": None}
except (httpx.HTTPError, httpx.TimeoutException, KeyError, ValueError, OSError) as e:
logger.warning("Pexels video fetch 실패: %s", e)
return {"path": None, "used_fallback": True, "error": str(e)}