"""Handlers responsible for serving preview assets dynamically.""" from __future__ import annotations import logging import mimetypes import urllib.parse from pathlib import Path from aiohttp import web from ...config import config as global_config logger = logging.getLogger(__name__) _CHUNK_SIZE = 1024 * 1024 # 1 MB — balance between streaming iteration overhead and per-chunk memory # Video file extensions that bypass native sendfile on Windows # to avoid IOCP/ProactorEventLoop crashes during client disconnect. _VIDEO_EXTENSIONS = frozenset({".mp4", ".webm", ".mov", ".avi", ".mkv"}) class PreviewHandler: """Serve preview assets for the active library at request time.""" def __init__(self, *, config=global_config) -> None: self._config = config async def serve_preview(self, request: web.Request) -> web.StreamResponse: """Return the preview file referenced by the encoded ``path`` query.""" raw_path = request.query.get("path", "") if not raw_path: raise web.HTTPBadRequest(text="Missing 'path' query parameter") try: decoded_path = urllib.parse.unquote(raw_path) except Exception as exc: # pragma: no cover - defensive guard logger.debug("Failed to decode preview path %s: %s", raw_path, exc) raise web.HTTPBadRequest(text="Invalid preview path encoding") from exc normalized = decoded_path.replace("\\", "/") if not self._config.is_preview_path_allowed(normalized): raise web.HTTPForbidden(text="Preview path is not within an allowed directory") candidate = Path(normalized) try: resolved = candidate.expanduser().resolve(strict=False) except Exception as exc: logger.debug("Failed to resolve preview path %s: %s", normalized, exc) raise web.HTTPBadRequest(text="Unable to resolve preview path") from exc if not resolved.is_file(): logger.debug("Preview file not found at %s", str(resolved)) raise web.HTTPNotFound(text="Preview file not found") # aiohttp's FileResponse handles range requests, content headers, and # uses kernel sendfile (zero-copy DMA) on Linux/macOS. On Windows it # uses IOCP-based _sendfile_native which can crash when the client # disconnects mid-transfer during fast scrolling. The _stream_file() # fallback is kept for a future compat toggle. # # Set explicit Cache-Control so the browser can cache video (and image) # previews across VirtualScroller recycling cycles. Without this, # Chrome does not cache 206 Partial Content responses for