team.py 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339
  1. from datetime import datetime
  2. from typing import TYPE_CHECKING, Generic, TypeVar
  3. from core_domain import (
  4. TeamDefinitionContract,
  5. TeamMemberContract,
  6. TeamRunContract,
  7. TeamRunStatus,
  8. TeamStatus,
  9. TeamConfigContract,
  10. )
  11. from core_shared import JSONValue
  12. from pydantic import BaseModel, Field
  13. if TYPE_CHECKING:
  14. from app.db.models import TeamDefinition, TeamRun, TeamConfig
  15. T = TypeVar("T")
  16. class ApiErrorResponse(BaseModel):
  17. errorType: str
  18. message: str
  19. details: dict[str, JSONValue] = Field(default_factory=dict)
  20. class ApiResponse(BaseModel, Generic[T]):
  21. success: bool = True
  22. data: T | None = None
  23. error: ApiErrorResponse | None = None
  24. requestId: str
  25. serverTime: datetime
  26. class PageRequest(BaseModel):
  27. page: int = Field(default=1, ge=1)
  28. pageSize: int = Field(default=20, ge=1, le=200)
  29. keyword: str | None = None
  30. @property
  31. def offset(self) -> int:
  32. return (self.page - 1) * self.pageSize
  33. class PageResult(BaseModel, Generic[T]):
  34. items: list[T]
  35. total: int
  36. page: int
  37. pageSize: int
  38. hasMore: bool
  39. @classmethod
  40. def from_items(
  41. cls,
  42. *,
  43. items: list[T],
  44. total: int,
  45. page: int,
  46. page_size: int) -> "PageResult[T]":
  47. return cls(
  48. items=items,
  49. total=total,
  50. page=page,
  51. pageSize=page_size,
  52. hasMore=page * page_size < total)
  53. class TeamCreateRequest(BaseModel):
  54. code: str
  55. name: str
  56. description: str | None = None
  57. team_type: str = "collaborative"
  58. owner_user_id: str | None = None
  59. metadata_json: dict[str, JSONValue] = Field(default_factory=dict)
  60. class TeamStatusUpdateRequest(BaseModel):
  61. status: TeamStatus
  62. class TeamResponse(TeamDefinitionContract):
  63. @classmethod
  64. def from_entity(cls, entity: "TeamDefinition") -> "TeamResponse":
  65. return cls.model_validate(entity, from_attributes=True)
  66. class TeamConfigCreateRequest(BaseModel):
  67. team_id: str
  68. coordination_mode: str = "supervisor"
  69. objective: str | None = None
  70. member_refs: list[TeamMemberContract] = Field(default_factory=list)
  71. policy_json: dict[str, JSONValue] = Field(default_factory=dict)
  72. class TeamConfigResponse(TeamConfigContract):
  73. @classmethod
  74. def from_entity(cls, entity: "TeamConfig") -> "TeamConfigResponse":
  75. return cls.model_validate(entity, from_attributes=True)
  76. class TeamRunCreateRequest(BaseModel):
  77. team_id: str
  78. team_config_id: str | None = None
  79. session_id: str | None = None
  80. input_text: str | None = None
  81. input_json: dict[str, JSONValue] | None = None
  82. class TeamRunStatusUpdateRequest(BaseModel):
  83. status: TeamRunStatus
  84. worker_key: str | None = None
  85. output_text: str | None = None
  86. output_json: dict[str, JSONValue] | None = None
  87. error_code: str | None = None
  88. error_message: str | None = None
  89. class TeamRunExecuteRequest(BaseModel):
  90. worker_key: str | None = None
  91. dry_run: bool = True
  92. class TeamRunResponse(TeamRunContract):
  93. @classmethod
  94. def from_entity(cls, entity: "TeamRun") -> "TeamRunResponse":
  95. return cls.model_validate(entity, from_attributes=True)
  96. class TeamRunExecuteResponse(BaseModel):
  97. run: TeamRunResponse
  98. member_run_count: int = 0
  99. dry_run: bool = True
  100. class TeamWorkerExecuteNextRequest(BaseModel):
  101. worker_key: str
  102. lease_seconds: int | None = Field(default=None, gt=0)
  103. dry_run: bool | None = None
  104. class TeamWorkerExecuteNextResponse(BaseModel):
  105. run: TeamRunResponse
  106. member_run_count: int = 0
  107. dry_run: bool = True
  108. released_lease_count: int = 0
  109. class TeamDto(BaseModel):
  110. id: str
  111. name: str
  112. description: str | None = None
  113. teamType: str
  114. status: TeamStatus
  115. ownerUserId: str | None = None
  116. metadata: dict[str, JSONValue] = Field(default_factory=dict)
  117. createdTime: datetime
  118. @classmethod
  119. def from_entity(cls, entity: "TeamDefinition") -> "TeamDto":
  120. return cls(
  121. id=entity.id,
  122. name=entity.name,
  123. description=entity.description,
  124. teamType=entity.team_type,
  125. status=entity.status,
  126. ownerUserId=entity.owner_user_id,
  127. metadata=entity.metadata_json or {},
  128. createdTime=entity.created_time)
  129. class TeamListRequestDto(PageRequest):
  130. status: TeamStatus | None = None
  131. class TeamCreateRequestDto(BaseModel):
  132. name: str
  133. description: str | None = None
  134. teamType: str = "collaborative"
  135. ownerUserId: str | None = None
  136. metadata: dict[str, JSONValue] = Field(default_factory=dict)
  137. class TeamDetailRequestDto(BaseModel):
  138. teamId: str
  139. class TeamUpdateRequestDto(BaseModel):
  140. teamId: str
  141. name: str | None = None
  142. description: str | None = None
  143. teamType: str | None = None
  144. status: TeamStatus | None = None
  145. ownerUserId: str | None = None
  146. metadata: dict[str, JSONValue] | None = None
  147. class TeamDeleteRequestDto(BaseModel):
  148. teamId: str
  149. class TeamConfigDto(BaseModel):
  150. id: str
  151. teamId: str
  152. coordinationMode: str
  153. objective: str | None = None
  154. memberRefs: list[dict[str, JSONValue]]
  155. policy: dict[str, JSONValue]
  156. createdTime: datetime
  157. @classmethod
  158. def from_entity(cls, entity: "TeamConfig") -> "TeamConfigDto":
  159. return cls(
  160. id=entity.id,
  161. teamId=entity.team_id,
  162. coordinationMode=entity.coordination_mode,
  163. objective=entity.objective,
  164. memberRefs=entity.member_refs_json,
  165. policy=entity.policy_json,
  166. createdTime=entity.created_time)
  167. class TeamConfigListRequestDto(PageRequest):
  168. teamId: str | None = None
  169. class TeamConfigCreateRequestDto(BaseModel):
  170. teamId: str
  171. coordinationMode: str = "supervisor"
  172. objective: str | None = None
  173. memberRefs: list[dict[str, JSONValue]] = Field(default_factory=list)
  174. policy: dict[str, JSONValue] = Field(default_factory=dict)
  175. class TeamConfigDetailRequestDto(BaseModel):
  176. configId: str
  177. class TeamConfigUpdateRequestDto(BaseModel):
  178. configId: str
  179. coordinationMode: str | None = None
  180. objective: str | None = None
  181. memberRefs: list[dict[str, JSONValue]] | None = None
  182. policy: dict[str, JSONValue] | None = None
  183. class TeamConfigDeleteRequestDto(BaseModel):
  184. configId: str
  185. class TeamRunDto(BaseModel):
  186. id: str
  187. teamId: str
  188. teamConfigId: str
  189. sessionId: str | None = None
  190. inputText: str | None = None
  191. inputJson: dict[str, JSONValue] | None = None
  192. outputText: str | None = None
  193. outputJson: dict[str, JSONValue] | None = None
  194. status: TeamRunStatus
  195. workerKey: str | None = None
  196. queuedTime: datetime | None = None
  197. leaseExpireTime: datetime | None = None
  198. startedTime: datetime | None = None
  199. finishedTime: datetime | None = None
  200. errorCode: str | None = None
  201. errorMessage: str | None = None
  202. createdTime: datetime
  203. @classmethod
  204. def from_entity(cls, entity: "TeamRun") -> "TeamRunDto":
  205. return cls(
  206. id=entity.id,
  207. teamId=entity.team_id,
  208. teamConfigId=entity.team_config_id,
  209. sessionId=entity.session_id,
  210. inputText=entity.input_text,
  211. inputJson=entity.input_json,
  212. outputText=entity.output_text,
  213. outputJson=entity.output_json,
  214. status=entity.status,
  215. workerKey=entity.worker_key,
  216. queuedTime=entity.queued_time,
  217. leaseExpireTime=entity.lease_expire_time,
  218. startedTime=entity.started_time,
  219. finishedTime=entity.finished_time,
  220. errorCode=entity.error_code,
  221. errorMessage=entity.error_message,
  222. createdTime=entity.created_time)
  223. class TeamRunListRequestDto(PageRequest):
  224. teamId: str | None = None
  225. sessionId: str | None = None
  226. status: TeamRunStatus | None = None
  227. class TeamRunCreateRequestDto(BaseModel):
  228. teamId: str
  229. teamConfigId: str | None = None
  230. sessionId: str | None = None
  231. inputText: str | None = None
  232. inputJson: dict[str, JSONValue] | None = None
  233. class TeamRunDetailRequestDto(BaseModel):
  234. teamRunId: str
  235. class TeamRunStatusUpdateRequestDto(BaseModel):
  236. teamRunId: str
  237. status: TeamRunStatus
  238. workerKey: str | None = None
  239. outputText: str | None = None
  240. outputJson: dict[str, JSONValue] | None = None
  241. errorCode: str | None = None
  242. errorMessage: str | None = None
  243. class TeamRunExecuteRequestDto(BaseModel):
  244. teamRunId: str
  245. workerKey: str | None = None
  246. dryRun: bool = True
  247. class TeamRunExecuteData(BaseModel):
  248. run: TeamRunDto
  249. memberRunCount: int = 0
  250. dryRun: bool = True
  251. class TeamRunDeleteRequestDto(BaseModel):
  252. teamRunId: str
  253. class DeleteData(BaseModel):
  254. deleted: bool
  255. teamId: str | None = None
  256. configId: str | None = None
  257. teamRunId: str | None = None