| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156 |
- # -*- coding: utf-8 -*-
- from datetime import datetime
- from pydantic import BaseModel, ConfigDict, Field, field_validator, model_validator
- from urllib.parse import urlparse
- from fastapi import Query
- class ResourceItemSchema(BaseModel):
- """资源项目模型"""
- model_config = ConfigDict(from_attributes=True)
-
- name: str = Field(..., description="文件名")
- file_url: str = Field(..., description="文件URL路径")
- relative_path: str = Field(..., description="相对路径")
- is_file: bool = Field(..., description="是否为文件")
- is_dir: bool = Field(..., description="是否为目录")
- size: int | None = Field(None, description="文件大小(字节)")
- created_time: datetime | None = Field(None, description="创建时间")
- modified_time: datetime | None = Field(None, description="修改时间")
- is_hidden: bool = Field(False, description="是否为隐藏文件")
- @field_validator('file_url')
- @classmethod
- def _validate_file_url(cls, v: str) -> str:
- v = v.strip()
- parsed = urlparse(v)
- if parsed.scheme not in ('http', 'https'):
- raise ValueError('文件URL必须为 http/https')
- return v
- @field_validator('relative_path')
- @classmethod
- def _validate_relative_path(cls, v: str) -> str:
- v = v.strip()
- if '..' in v or v.startswith('\\'):
- raise ValueError('相对路径包含不安全字符')
- return v
- @model_validator(mode='after')
- def _validate_flags(self):
- if self.is_file and self.is_dir:
- raise ValueError('不能同时为文件和目录')
- if not self.is_file and not self.is_dir:
- raise ValueError('必须是文件或目录之一')
- # 根据名称自动修正隐藏标记
- self.is_hidden = self.name.startswith('.')
- return self
- class ResourceDirectorySchema(BaseModel):
- """资源目录模型"""
- model_config = ConfigDict(from_attributes=True)
-
- path: str = Field(..., description="目录路径")
- name: str = Field(..., description="目录名称")
- items: list[ResourceItemSchema] = Field(default_factory=list, description="目录项")
- total_files: int = Field(0, description="文件总数")
- total_dirs: int = Field(0, description="目录总数")
- total_size: int = Field(0, description="总大小")
- class ResourceUploadSchema(BaseModel):
- """资源上传响应模型"""
- model_config = ConfigDict(from_attributes=True)
-
- filename: str = Field(..., description="文件名")
- file_url: str = Field(..., description="访问URL")
- file_size: int = Field(..., description="文件大小")
- upload_time: datetime = Field(..., description="上传时间")
- class ResourceMoveSchema(BaseModel):
- """资源移动模型"""
- model_config = ConfigDict(from_attributes=True)
-
- source_path: str = Field(..., description="源路径")
- target_path: str = Field(..., description="目标路径")
- overwrite: bool = Field(False, description="是否覆盖")
-
- @field_validator('source_path', 'target_path')
- @classmethod
- def validate_paths(cls, value: str):
- if not value or len(value.strip()) == 0:
- raise ValueError("路径不能为空")
- return value.strip()
- class ResourceCopySchema(ResourceMoveSchema):
- """资源复制模型"""
- pass
- class ResourceRenameSchema(BaseModel):
- """资源重命名模型"""
- model_config = ConfigDict(from_attributes=True)
-
- old_path: str = Field(..., description="原路径")
- new_name: str = Field(..., description="新名称")
-
- @field_validator('old_path', 'new_name')
- @classmethod
- def validate_inputs(cls, value: str):
- if not value or len(value.strip()) == 0:
- raise ValueError("参数不能为空")
- return value.strip()
- @field_validator('new_name')
- @classmethod
- def _validate_new_name(cls, v: str) -> str:
- v = v.strip()
- if '..' in v or '/' in v or '\\' in v:
- raise ValueError('新名称包含不安全字符')
- return v
- class ResourceCreateDirSchema(BaseModel):
- """创建目录模型"""
- model_config = ConfigDict(from_attributes=True)
-
- parent_path: str = Field(..., description="父目录路径")
- dir_name: str = Field(..., description="目录名称", max_length=255)
-
- @field_validator('parent_path', 'dir_name')
- @classmethod
- def validate_inputs(cls, value: str, info):
- # 对于parent_path允许为空字符串(表示根目录)或 '/',其他情况必须非空
- if info.field_name == 'parent_path':
- # 允许空字符串或 '/' 表示根目录
- if value is None:
- raise ValueError("参数不能为空")
- # 对于parent_path仍然严格检查路径遍历
- if '..' in value or value.startswith('\\'):
- raise ValueError("参数包含不安全字符")
- else: # 对于dir_name仍然严格检查
- if not value or len(value.strip()) == 0:
- raise ValueError("参数不能为空")
- if '..' in value or value.startswith('/') or value.startswith('\\'):
- raise ValueError("参数包含不安全字符")
- return value.strip()
- class ResourceSearchQueryParam:
- """资源搜索查询参数"""
- def __init__(
- self,
- name: str | None = Query(None, description="搜索关键词"),
- path: str | None = Query(None, description="目录路径"),
- ) -> None:
-
- # 模糊查询字段
- self.name = ("like", name) if name else None
-
- # 精确查询字段
- self.path = path
|