| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153 |
- from typing import Literal
- from fastapi import Query
- from pydantic import BaseModel, ConfigDict, Field, model_validator
- from app.common.enums import QueueEnum
- from app.core.base_schema import BaseSchema
- from app.core.validator import DateTimeStr, menu_request_validator
- class MenuCreateSchema(BaseModel):
- """菜单创建模型"""
- name: str = Field(..., max_length=50, description="菜单名称")
- type: int = Field(..., ge=1, le=4, description="菜单类型(1:目录 2:菜单 3:按钮 4:外链)")
- order: int = Field(..., ge=1, description="显示顺序")
- permission: str | None = Field(default=None, max_length=100, description="权限标识")
- icon: str | None = Field(default=None, max_length=100, description="菜单图标")
- route_name: str | None = Field(default=None, max_length=100, description="路由名称")
- route_path: str | None = Field(default=None, max_length=200, description="路由地址")
- component_path: str | None = Field(default=None, max_length=255, description="组件路径")
- redirect: str | None = Field(default=None, max_length=200, description="重定向地址")
- hidden: bool = Field(default=False, description="是否隐藏(True:是 False:否)")
- keep_alive: bool = Field(default=True, description="是否缓存(True:是 False:否)")
- always_show: bool = Field(default=False, description="是否始终显示(True:是 False:否)")
- title: str | None = Field(default=None, max_length=50, description="菜单标题")
- params: list[dict[str, str]] | None = Field(
- default=None,
- description="路由参数,格式为[{key: string, value: string}]",
- )
- affix: bool = Field(default=False, description="是否固定标签页(True:是 False:否)")
- parent_id: int | None = Field(default=None, ge=1, description="父菜单ID")
- status: str = Field(default="0", description="是否启用(0:启用 1:禁用)")
- description: str | None = Field(default=None, max_length=255, description="描述")
- @model_validator(mode="before")
- @classmethod
- def _normalize(cls, values):
- if isinstance(values, dict):
- # 字符串去空格
- for k in [
- "name",
- "icon",
- "permission",
- "route_name",
- "route_path",
- "component_path",
- "redirect",
- "title",
- "description",
- ]:
- if k in values and isinstance(values[k], str):
- values[k] = (
- values[k].strip() or None if values[k].strip() == "" else values[k].strip()
- )
- # 父ID转整型
- if "parent_id" in values and isinstance(values["parent_id"], str):
- try:
- values["parent_id"] = int(values["parent_id"].strip())
- except Exception:
- pass
- # 路由名/路径规范
- if "route_path" in values and isinstance(values["route_path"], str):
- rp = values["route_path"]
- if rp and not rp.startswith("/"):
- raise ValueError("路由路径需以 / 开头")
- if "component_path" in values and isinstance(values["component_path"], str):
- cp = values["component_path"]
- if cp and cp.startswith("/"):
- raise ValueError("组件路径不能以 / 开头")
- return values
- @model_validator(mode="after")
- def validate_fields(self):
- """
- 统一校验菜单请求字段(委托到 `menu_request_validator`)。
- 返回:
- - MenuCreateSchema: 校验后的同一实例。
- 异常:
- - CustomException: 字段不满足菜单类型约束时抛出。
- """
- return menu_request_validator(self)
- class MenuUpdateSchema(MenuCreateSchema):
- """菜单更新模型"""
- parent_name: str | None = Field(default=None, max_length=50, description="父菜单名称")
- class MenuOutSchema(MenuCreateSchema, BaseSchema):
- """菜单响应模型"""
- model_config = ConfigDict(from_attributes=True)
- parent_name: str | None = Field(default=None, max_length=50, description="父菜单名称")
- class MenuQueryParam:
- """菜单管理查询参数"""
- def __init__(
- self,
- name: str | None = Query(None, description="菜单名称"),
- route_path: str | None = Query(None, description="路由地址"),
- component_path: str | None = Query(None, description="组件路径"),
- type: Literal[1, 2, 3, 4] | None = Query(
- None, description="菜单类型(1:目录 2:菜单 3:按钮 4:外链)"
- ),
- permission: str | None = Query(None, description="权限标识"),
- description: str | None = Query(None, description="描述"),
- status: str | None = Query(None, description="是否启用"),
- created_time: list[DateTimeStr] | None = Query(
- None,
- description="创建时间范围",
- examples=["2025-01-01 00:00:00", "2025-12-31 23:59:59"],
- ),
- updated_time: list[DateTimeStr] | None = Query(
- None,
- description="更新时间范围",
- examples=["2025-01-01 00:00:00", "2025-12-31 23:59:59"],
- ),
- created_id: int | None = Query(None, description="创建人"),
- updated_id: int | None = Query(None, description="更新人"),
- ) -> None:
- # 模糊查询字段
- self.name = (QueueEnum.like.value, name)
- self.route_path = (QueueEnum.like.value, route_path)
- self.component_path = (QueueEnum.like.value, component_path)
- self.permission = (QueueEnum.like.value, permission)
- # 精确查询字段
- self.type = type
- # 模糊查询字段
- if description:
- self.description = (QueueEnum.like.value, description)
- # 精确查询字段
- if status:
- self.status = (QueueEnum.eq.value, status)
- # 时间范围查询
- if created_time and len(created_time) == 2:
- self.created_time = (QueueEnum.between.value, (created_time[0], created_time[1]))
- if updated_time and len(updated_time) == 2:
- self.updated_time = (QueueEnum.between.value, (updated_time[0], updated_time[1]))
- # 关联查询字段
- if created_id:
- self.created_id = (QueueEnum.eq.value, created_id)
- if updated_id:
- self.updated_id = (QueueEnum.eq.value, updated_id)
|