schema.py 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. # -*- coding: utf-8 -*-
  2. from fastapi import Query
  3. from pydantic import BaseModel, ConfigDict, Field, EmailStr, field_validator
  4. from urllib.parse import urlparse
  5. from app.core.validator import DateTimeStr, mobile_validator
  6. from app.core.base_schema import BaseSchema, CommonSchema, UserBySchema
  7. from app.core.validator import DateTimeStr
  8. from app.api.v1.module_system.menu.schema import MenuOutSchema
  9. from app.api.v1.module_system.role.schema import RoleOutSchema
  10. class CurrentUserUpdateSchema(BaseModel):
  11. """基础用户信息"""
  12. name: str | None = Field(default=None, max_length=32, description="名称")
  13. mobile: str | None = Field(default=None, description="手机号")
  14. email: EmailStr | None = Field(default=None, description="邮箱")
  15. gender: str | None = Field(default=None, description="性别")
  16. avatar: str | None = Field(default=None, description="头像")
  17. @field_validator("mobile")
  18. @classmethod
  19. def validate_mobile(cls, value: str | None):
  20. return mobile_validator(value)
  21. @field_validator("avatar")
  22. @classmethod
  23. def validate_avatar(cls, value: str | None):
  24. if not value:
  25. return value
  26. parsed = urlparse(value)
  27. if parsed.scheme in ("http", "https") and parsed.netloc:
  28. return value
  29. raise ValueError("头像地址需为有效的HTTP/HTTPS URL")
  30. class UserRegisterSchema(BaseModel):
  31. """注册"""
  32. name: str | None = Field(default=None, max_length=32, description="名称")
  33. mobile: str | None = Field(default=None, description="手机号")
  34. username: str = Field(..., max_length=32, description="账号")
  35. password: str = Field(..., max_length=128, description="密码哈希值")
  36. role_ids: list[int] | None = Field(default=[1], description='角色ID')
  37. created_id: int | None = Field(default=1, description='创建人ID')
  38. description: str | None = Field(default=None, max_length=255, description="备注")
  39. @field_validator("mobile")
  40. @classmethod
  41. def validate_mobile(cls, value: str | None):
  42. return mobile_validator(value)
  43. @field_validator("username")
  44. @classmethod
  45. def validate_username(cls, value: str):
  46. v = value.strip()
  47. if not v:
  48. raise ValueError("账号不能为空")
  49. # 字母开头,允许字母数字_.-
  50. import re
  51. if not re.match(r"^[A-Za-z][A-Za-z0-9_.-]{2,31}$", v):
  52. raise ValueError("账号需字母开头,3-32位,仅含字母/数字/_ . -")
  53. return v
  54. class UserForgetPasswordSchema(BaseModel):
  55. """忘记密码"""
  56. username: str = Field(..., max_length=32, description="用户名")
  57. new_password: str = Field(..., max_length=128, description="新密码")
  58. mobile: str | None = Field(default=None, description="手机号")
  59. @field_validator("mobile")
  60. @classmethod
  61. def validate_mobile(cls, value: str | None):
  62. return mobile_validator(value)
  63. class UserChangePasswordSchema(BaseModel):
  64. """修改密码"""
  65. old_password: str = Field(..., max_length=128, description="旧密码")
  66. new_password: str = Field(..., max_length=128, description="新密码")
  67. class ResetPasswordSchema(BaseModel):
  68. """重置密码"""
  69. id: int = Field(..., description="主键ID")
  70. password: str = Field(..., min_length=6, max_length=128, description="新密码")
  71. class UserCreateSchema(CurrentUserUpdateSchema):
  72. """新增"""
  73. model_config = ConfigDict(from_attributes=True)
  74. username: str | None = Field(default=None, max_length=32, description="用户名")
  75. password: str | None = Field(default=None, max_length=128, description="密码哈希值")
  76. status: str = Field(default="0", description="是否可用")
  77. description: str | None = Field(default=None, max_length=255, description="备注")
  78. is_superuser: bool | None = Field(default=False, description="是否超管")
  79. dept_id: int | None = Field(default=None, description='部门ID')
  80. role_ids: list[int] | None = Field(default=[], description='角色ID')
  81. position_ids: list[int] | None = Field(default=[], description='岗位ID')
  82. class UserUpdateSchema(UserCreateSchema):
  83. """更新"""
  84. model_config = ConfigDict(from_attributes=True)
  85. last_login: DateTimeStr | None = Field(default=None, description="最后登录时间")
  86. class UserOutSchema(UserUpdateSchema, BaseSchema, UserBySchema):
  87. """响应"""
  88. model_config = ConfigDict(arbitrary_types_allowed=True, from_attributes=True)
  89. gitee_login: str | None = Field(default=None, max_length=32, description="Gitee登录")
  90. github_login: str | None = Field(default=None, max_length=32, description="Github登录")
  91. wx_login: str | None = Field(default=None, max_length=32, description="微信登录")
  92. qq_login: str | None = Field(default=None, max_length=32, description="QQ登录")
  93. dept_name: str | None = Field(default=None, description='部门名称')
  94. dept: CommonSchema | None = Field(default=None, description='部门')
  95. positions: list[CommonSchema] | None = Field(default=[], description='岗位')
  96. roles: list[RoleOutSchema] | None = Field(default=[], description='角色')
  97. menus: list[MenuOutSchema] | None = Field(default=[], description='菜单')
  98. class UserQueryParam:
  99. """用户管理查询参数"""
  100. def __init__(
  101. self,
  102. username: str | None = Query(None, description="用户名"),
  103. name: str | None = Query(None, description="名称"),
  104. mobile: str | None = Query(None, description="手机号", pattern=r'^1[3-9]\d{9}$'),
  105. email: str | None = Query(None, description="邮箱", pattern=r'^[a-zA-Z0-9_.+-]+@[a-zA-Z0-9-]+\.[a-zA-Z0-9-.]+$'),
  106. dept_id: int | None = Query(None, description="部门ID"),
  107. status: str | None = Query(None, description="是否可用"),
  108. created_time: list[DateTimeStr] | None = Query(None, description="创建时间范围", examples=["2025-01-01 00:00:00", "2025-12-31 23:59:59"]),
  109. updated_time: list[DateTimeStr] | None = Query(None, description="更新时间范围", examples=["2025-01-01 00:00:00", "2025-12-31 23:59:59"]),
  110. created_id: int | None = Query(None, description="创建人"),
  111. updated_id: int | None = Query(None, description="更新人"),
  112. ) -> None:
  113. # 模糊查询字段
  114. self.username = ("like", username)
  115. self.name = ("like", name)
  116. self.mobile = ("like", mobile)
  117. self.email = ("like", email)
  118. # 精确查询字段
  119. self.dept_id = dept_id
  120. self.created_id = created_id
  121. self.updated_id = updated_id
  122. self.status = status
  123. # 时间范围查询
  124. if created_time and len(created_time) == 2:
  125. self.created_time = ("between", (created_time[0], created_time[1]))
  126. if updated_time and len(updated_time) == 2:
  127. self.updated_time = ("between", (updated_time[0], updated_time[1]))