import mimetypes import re from fastapi import HTTPException, UploadFile from fastapi.responses import Response from domain.config.service import ConfigService from domain.virtual_fs.thumbnail import get_or_create_thumb, is_image_filename, is_raw_filename, is_video_filename from .temp_link import VirtualFSTempLinkMixin class VirtualFSRouteMixin(VirtualFSTempLinkMixin): @classmethod async def serve_file(cls, full_path: str, range_header: str | None) -> Response: full_path = cls._normalize_path(full_path) if is_raw_filename(full_path): import io import rawpy from PIL import Image try: raw_data = await cls.read_file(full_path) with rawpy.imread(io.BytesIO(raw_data)) as raw: rgb = raw.postprocess(use_camera_wb=True, output_bps=8) im = Image.fromarray(rgb) buf = io.BytesIO() im.save(buf, "JPEG", quality=90) content = buf.getvalue() return Response(content=content, media_type="image/jpeg") except FileNotFoundError: raise HTTPException(404, detail="File not found") except Exception as exc: raise HTTPException(500, detail=f"RAW file processing failed: {exc}") adapter_instance, adapter_model, root, rel = await cls.resolve_adapter_and_rel(full_path) redirect_response = await cls.maybe_redirect_download(adapter_instance, adapter_model, root, rel) if redirect_response is not None: return redirect_response try: content = await cls.read_file(full_path) except FileNotFoundError: raise HTTPException(404, detail="File not found") if not isinstance(content, (bytes, bytearray)): return Response(content=content, media_type="application/octet-stream") content_length = len(content) content_type = mimetypes.guess_type(full_path)[0] or "application/octet-stream" if range_header: range_match = re.match(r"bytes=(\\d+)-(\\d*)", range_header) if range_match: start = int(range_match.group(1)) end = int(range_match.group(2)) if range_match.group(2) else content_length - 1 start = max(0, min(start, content_length - 1)) end = max(start, min(end, content_length - 1)) chunk = content[start : end + 1] chunk_size = len(chunk) headers = { "Content-Range": f"bytes {start}-{end}/{content_length}", "Accept-Ranges": "bytes", "Content-Length": str(chunk_size), "Content-Type": content_type, } return Response(content=chunk, status_code=206, headers=headers) headers = { "Accept-Ranges": "bytes", "Content-Length": str(content_length), "Content-Type": content_type, } if content_type.startswith("video/"): headers["Cache-Control"] = "public, max-age=3600" return Response(content=content, headers=headers) @classmethod async def get_thumbnail(cls, full_path: str, w: int, h: int, fit: str) -> Response: full_path = cls._normalize_path(full_path) if fit not in ("cover", "contain"): raise HTTPException(400, detail="fit must be cover|contain") adapter, mount, root, rel = await cls.resolve_adapter_and_rel(full_path) if not rel or rel.endswith("/"): raise HTTPException(400, detail="Not a file") if not (is_image_filename(rel) or is_video_filename(rel)): raise HTTPException(404, detail="Not an image or video") data, mime, key = await get_or_create_thumb(adapter, mount.id, root, rel, w, h, fit) # type: ignore headers = { "Cache-Control": "public, max-age=3600", "ETag": key, } return Response(content=data, media_type=mime, headers=headers) @classmethod async def stream_response(cls, full_path: str, range_header: str | None): full_path = cls._normalize_path(full_path) try: return await cls.stream_file(full_path, range_header) except HTTPException: raise except FileNotFoundError: raise HTTPException(404, detail="File not found") except Exception as exc: raise HTTPException(500, detail=f"Stream error: {exc}") @classmethod async def create_temp_link(cls, full_path: str, expires_in: int): full_path = cls._normalize_path(full_path) token = await cls.generate_temp_link_token(full_path, expires_in=expires_in) file_domain = await ConfigService.get("FILE_DOMAIN") if file_domain: file_domain = file_domain.rstrip("/") url = f"{file_domain}/api/fs/public/{token}" else: url = f"/api/fs/public/{token}" return {"token": token, "path": full_path, "url": url} @classmethod async def access_public_file(cls, token: str, range_header: str | None): try: path = await cls.verify_temp_link_token(token) except HTTPException as exc: raise exc try: return await cls.stream_file(path, range_header) except FileNotFoundError: raise HTTPException(404, detail="File not found via token") except Exception as exc: raise HTTPException(500, detail=f"File access error: {exc}") @classmethod async def stat(cls, full_path: str): full_path = cls._normalize_path(full_path) return await cls.stat_file(full_path) @classmethod async def write_uploaded_file(cls, full_path: str, data: bytes): full_path = cls._normalize_path(full_path) await cls.write_file(full_path, data) return {"written": True, "path": full_path, "size": len(data)} @classmethod async def mkdir(cls, path: str): path = cls._normalize_path(path) if not path or path == "/": raise HTTPException(400, detail="Invalid path") await cls.make_dir(path) return {"created": True, "path": path} @classmethod async def move(cls, src: str, dst: str, overwrite: bool): src = cls._normalize_path(src) dst = cls._normalize_path(dst) debug_info = await cls.move_path(src, dst, overwrite=overwrite, return_debug=True, allow_cross=True) queued = bool(debug_info.get("queued")) response = { "moved": not queued, "queued": queued, "src": src, "dst": dst, "overwrite": overwrite, } if queued: response["task_id"] = debug_info.get("task_id") response["task_name"] = debug_info.get("task_name") return response @classmethod async def rename(cls, src: str, dst: str, overwrite: bool): src = cls._normalize_path(src) dst = cls._normalize_path(dst) await cls.rename_path(src, dst, overwrite=overwrite, return_debug=False) return {"renamed": True, "src": src, "dst": dst, "overwrite": overwrite} @classmethod async def copy(cls, src: str, dst: str, overwrite: bool): src = cls._normalize_path(src) dst = cls._normalize_path(dst) debug_info = await cls.copy_path(src, dst, overwrite=overwrite, return_debug=True, allow_cross=True) queued = bool(debug_info.get("queued")) response = { "copied": not queued, "queued": queued, "src": src, "dst": dst, "overwrite": overwrite, } if queued: response["task_id"] = debug_info.get("task_id") response["task_name"] = debug_info.get("task_name") return response @classmethod async def upload_stream_from_upload_file(cls, full_path: str, file: UploadFile, chunk_size: int, overwrite: bool): full_path = cls._normalize_path(full_path) if full_path.endswith("/"): raise HTTPException(400, detail="Path must be a file") adapter, _m, root, rel = await cls.resolve_adapter_and_rel(full_path) exists_func = getattr(adapter, "exists", None) if not overwrite and callable(exists_func): try: if await exists_func(root, rel): raise HTTPException(409, detail="Destination exists") except HTTPException: raise except Exception: pass async def gen(): while True: chunk = await file.read(chunk_size) if not chunk: break yield chunk size = await cls.write_file_stream(full_path, gen(), overwrite=overwrite) return {"uploaded": True, "path": full_path, "size": size, "overwrite": overwrite} @classmethod async def list_directory(cls, full_path: str, page_num: int, page_size: int, sort_by: str, sort_order: str): full_path = cls._normalize_path(full_path) result = await cls.list_virtual_dir(full_path, page_num, page_size, sort_by, sort_order) return { "path": full_path, "entries": result["items"], "pagination": { "total": result["total"], "page": result["page"], "page_size": result["page_size"], "pages": result["pages"], }, } @classmethod async def delete(cls, full_path: str): full_path = cls._normalize_path(full_path) await cls.delete_path(full_path) return {"deleted": True, "path": full_path}