schema.py 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119
  1. from fastapi import Query
  2. from pydantic import (
  3. BaseModel,
  4. ConfigDict,
  5. Field,
  6. field_validator,
  7. model_validator,
  8. )
  9. from app.common.enums import QueueEnum
  10. from app.core.base_schema import BaseSchema, UserBySchema
  11. from app.core.validator import DateTimeStr, datetime_validator
  12. class NodeCreateSchema(BaseModel):
  13. """
  14. 节点创建/编辑时只设置节点基本信息,节点参数在执行时设置
  15. """
  16. name: str = Field(..., max_length=64, description="任务名称")
  17. func: str | None = Field(default=None, description="代码块")
  18. args: str | None = Field(default=None, description="位置参数")
  19. kwargs: str | None = Field(default=None, description="关键字参数")
  20. coalesce: bool | None = Field(default=False, description="是否合并运行:是否在多个运行时间到期时仅运行作业一次")
  21. max_instances: int | None = Field(default=1, ge=1, description="最大实例数:允许的最大并发执行实例数")
  22. jobstore: str | None = Field(default="default", max_length=64, description="任务存储")
  23. executor: str | None = Field(default="default", max_length=64, description="任务执行器:将运行此作业的执行程序的名称",)
  24. start_date: str | None = Field(default=None, description="开始时间")
  25. end_date: str | None = Field(default=None, description="结束时间")
  26. code: str | None = Field(default=None, description="节点编码")
  27. @model_validator(mode="after")
  28. def _validate_func(self):
  29. if not self.func or not self.func.strip():
  30. raise ValueError("必须提供代码块(func)")
  31. return self
  32. class NodeUpdateSchema(NodeCreateSchema):
  33. """节点更新模型"""
  34. class NodeOutSchema(NodeCreateSchema, BaseSchema, UserBySchema):
  35. """节点响应模型"""
  36. trigger: str | None = Field(default=None, description="触发器")
  37. trigger_args: str | None = Field(default=None, description="触发器参数")
  38. model_config = ConfigDict(from_attributes=True)
  39. class NodeQueryParam:
  40. """节点查询参数"""
  41. def __init__(
  42. self,
  43. name: str | None = Query(None, description="节点名称"),
  44. status: str | None = Query(None, description="状态: 启动,停止"),
  45. created_time: list[DateTimeStr] | None = Query(
  46. None,
  47. description="创建时间范围",
  48. examples=["2025-01-01 00:00:00", "2025-12-31 23:59:59"],
  49. ),
  50. updated_time: list[DateTimeStr] | None = Query(
  51. None,
  52. description="更新时间范围",
  53. examples=["2025-01-01 00:00:00", "2025-12-31 23:59:59"],
  54. ),
  55. created_id: int | None = Query(None, description="创建人"),
  56. updated_id: int | None = Query(None, description="更新人"),
  57. ) -> None:
  58. self.name = (QueueEnum.like.value, name)
  59. self.created_id = (QueueEnum.eq.value, created_id)
  60. self.updated_id = (QueueEnum.eq.value, updated_id)
  61. self.status = (QueueEnum.eq.value, status)
  62. if created_time and len(created_time) == 2:
  63. self.created_time = (QueueEnum.between.value, (created_time[0], created_time[1]))
  64. if updated_time and len(updated_time) == 2:
  65. self.updated_time = (QueueEnum.between.value, (updated_time[0], updated_time[1]))
  66. class NodeExecuteSchema(BaseModel):
  67. """节点执行参数"""
  68. trigger: str = Field(default="now", description="触发方式: now/cron/interval/date")
  69. trigger_args: str | None = Field(default=None, description="触发器参数")
  70. start_date: str | None = Field(default=None, description="开始时间")
  71. end_date: str | None = Field(default=None, description="结束时间")
  72. @field_validator("trigger")
  73. @classmethod
  74. def _validate_trigger(cls, v: str) -> str:
  75. allowed = {"now", "cron", "interval", "date"}
  76. v = v.strip()
  77. if v not in allowed:
  78. raise ValueError("触发器必须为 now/cron/interval/date")
  79. return v
  80. @model_validator(mode="after")
  81. def _validate_trigger_args(self):
  82. """非立即执行时必须提供触发器参数"""
  83. if self.trigger != "now" and not self.trigger_args:
  84. raise ValueError("非立即执行时必须提供触发器参数")
  85. return self
  86. @model_validator(mode="after")
  87. def _validate_dates(self):
  88. """跨字段校验:结束时间不得早于开始时间。"""
  89. if self.start_date and self.end_date:
  90. try:
  91. start = datetime_validator(self.start_date)
  92. end = datetime_validator(self.end_date)
  93. except Exception:
  94. raise ValueError("时间格式必须为 YYYY-MM-DD HH:MM:SS")
  95. if end < start:
  96. raise ValueError("结束时间不能早于开始时间")
  97. return self