routes.py 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426
  1. from datetime import datetime
  2. from typing import Annotated, TypeVar
  3. from core_domain import ServiceHealth
  4. from core_shared import error_detail
  5. from fastapi import APIRouter, Depends, HTTPException, Query, Request
  6. from sqlalchemy import text
  7. from sqlalchemy.orm import Session
  8. from app.application.services import ToolApplicationService, build_tool_application_service
  9. from app.bootstrap.settings import ToolServiceSettings
  10. from app.db.session import get_db
  11. from app.schemas.tool import (
  12. ApiResponse,
  13. DeleteData,
  14. McpConnectData,
  15. McpConnectRequestDto,
  16. McpDiscoverRequestDto,
  17. PageRequest,
  18. PageResult,
  19. ToolBindingCreateRequest,
  20. ToolBindingCreateRequestDto,
  21. ToolBindingDeleteRequestDto,
  22. ToolBindingDetailRequestDto,
  23. ToolBindingDetailResponse,
  24. ToolBindingDto,
  25. ToolBindingListRequestDto,
  26. ToolBindingResponse,
  27. ToolBindingUpdateRequestDto,
  28. ToolCreateRequest,
  29. ToolCreateRequestDto,
  30. ToolCredentialCreateRequest,
  31. ToolCredentialCreateRequestDto,
  32. ToolCredentialDeleteRequestDto,
  33. ToolCredentialDetailRequestDto,
  34. ToolCredentialDto,
  35. ToolCredentialResponse,
  36. ToolCredentialRevealDto,
  37. ToolCredentialRevealRequestDto,
  38. ToolCredentialRevealResponse,
  39. ToolCredentialUpdateRequestDto,
  40. ToolDeleteRequestDto,
  41. ToolDetailRequestDto,
  42. ToolDto,
  43. ToolResponse,
  44. ToolUpdateRequestDto,
  45. ToolConnectionCreateRequest,
  46. ToolConnectionCreateRequestDto,
  47. ToolConnectionDeleteRequestDto,
  48. ToolConnectionDetailRequestDto,
  49. ToolConnectionDto,
  50. ToolConnectionListRequestDto,
  51. ToolConnectionResponse,
  52. ToolConnectionUpdateRequestDto,
  53. )
  54. router = APIRouter()
  55. DbSession = Annotated[Session, Depends(get_db)]
  56. T = TypeVar("T")
  57. def get_tool_application_service(
  58. request: Request,
  59. db: DbSession) -> ToolApplicationService:
  60. settings: ToolServiceSettings = request.app.state.settings
  61. return build_tool_application_service(db=db, settings=settings)
  62. ToolServiceDep = Annotated[ToolApplicationService, Depends(get_tool_application_service)]
  63. def ok(data: T) -> ApiResponse[T]:
  64. return ApiResponse[T](
  65. data=data,
  66. requestId="",
  67. serverTime=datetime.utcnow())
  68. @router.get("/health", response_model=ServiceHealth)
  69. def health_check(db: DbSession) -> ServiceHealth:
  70. db.execute(text("SELECT 1"))
  71. return ServiceHealth(service="tool-service", status="ok", database="ok")
  72. @router.post("", response_model=ToolResponse)
  73. def create_tool_definition(
  74. payload: ToolCreateRequest,
  75. service: ToolServiceDep) -> ToolResponse:
  76. entity = service.create_tool_definition(payload)
  77. return ToolResponse.from_entity(entity)
  78. @router.get("", response_model=list[ToolResponse])
  79. def list_tool_definitions(
  80. service: ToolServiceDep) -> list[ToolResponse]:
  81. return [ToolResponse.from_entity(item) for item in service.list_tool_definitions()]
  82. @router.post("/bindings", response_model=ToolBindingResponse)
  83. def create_tool_binding(
  84. payload: ToolBindingCreateRequest,
  85. service: ToolServiceDep) -> ToolBindingResponse:
  86. try:
  87. entity = service.create_tool_binding(payload)
  88. except ValueError as exc:
  89. raise HTTPException(status_code=422, detail=error_detail("error.validation", message=str(exc))) from exc
  90. return ToolBindingResponse.from_entity(entity)
  91. @router.get("/bindings", response_model=list[ToolBindingResponse])
  92. def list_tool_bindings(
  93. service: ToolServiceDep,
  94. app_id: str | None = Query(default=None)) -> list[ToolBindingResponse]:
  95. return [
  96. ToolBindingResponse.from_entity(item)
  97. for item in service.list_tool_bindings(app_id=app_id)
  98. ]
  99. @router.get("/bindings/{binding_id}", response_model=ToolBindingDetailResponse)
  100. def get_tool_binding_detail(
  101. binding_id: str,
  102. service: ToolServiceDep) -> ToolBindingDetailResponse:
  103. result = service.get_tool_binding_detail(binding_id=binding_id)
  104. if result is None:
  105. raise HTTPException(status_code=404, detail=error_detail("error.tool_binding.not_found", id=binding_id))
  106. binding, tool_connection, tool_definition = result
  107. return ToolBindingDetailResponse(
  108. binding=ToolBindingResponse.from_entity(binding),
  109. connection=ToolConnectionResponse.from_entity(tool_connection),
  110. tool_definition=ToolResponse.from_entity(tool_definition))
  111. @router.post("/credentials", response_model=ToolCredentialResponse)
  112. def create_tool_credential(
  113. payload: ToolCredentialCreateRequest,
  114. service: ToolServiceDep) -> ToolCredentialResponse:
  115. entity = service.create_tool_credential(payload)
  116. return ToolCredentialResponse.from_entity(entity)
  117. @router.get("/credentials", response_model=list[ToolCredentialResponse])
  118. def list_tool_credentials(
  119. service: ToolServiceDep) -> list[ToolCredentialResponse]:
  120. return [
  121. ToolCredentialResponse.from_entity(item)
  122. for item in service.list_tool_credentials()
  123. ]
  124. @router.post("/credentials/{credential_id}/reveal", response_model=ToolCredentialRevealResponse)
  125. def reveal_tool_credential(
  126. credential_id: str,
  127. service: ToolServiceDep) -> ToolCredentialRevealResponse:
  128. result = service.reveal_tool_credential(
  129. credential_id=credential_id)
  130. if result is None:
  131. raise HTTPException(status_code=404, detail=error_detail("error.tool_credential.not_found", id=credential_id))
  132. credential, secret_json = result
  133. return ToolCredentialRevealResponse(
  134. credential=ToolCredentialResponse.from_entity(credential),
  135. secret_json=secret_json)
  136. @router.post("/list", response_model=ApiResponse[PageResult[ToolDto]])
  137. def list_tools_contract(
  138. payload: PageRequest,
  139. service: ToolServiceDep) -> ApiResponse[PageResult[ToolDto]]:
  140. keyword = (payload.keyword or "").lower().strip()
  141. items = [
  142. item
  143. for item in service.list_tool_definitions()
  144. if not keyword
  145. or keyword in item.name.lower()
  146. or keyword in item.tool_type.lower()
  147. or keyword in (item.description or "").lower()
  148. ]
  149. page_items = items[payload.offset:payload.offset + payload.pageSize]
  150. return ok(
  151. PageResult[ToolDto].from_items(
  152. items=[ToolDto.from_entity(item) for item in page_items],
  153. total=len(items),
  154. page=payload.page,
  155. page_size=payload.pageSize))
  156. @router.post("/create", response_model=ApiResponse[ToolDto])
  157. def create_tool_contract(
  158. payload: ToolCreateRequestDto,
  159. service: ToolServiceDep) -> ApiResponse[ToolDto]:
  160. entity = service.create_tool_definition_from_contract(payload)
  161. return ok(ToolDto.from_entity(entity))
  162. @router.post("/detail", response_model=ApiResponse[ToolDto])
  163. def get_tool_contract(
  164. payload: ToolDetailRequestDto,
  165. service: ToolServiceDep) -> ApiResponse[ToolDto]:
  166. entity = service.get_tool_definition_from_contract(payload)
  167. if entity is None:
  168. raise HTTPException(status_code=404, detail=error_detail("error.tool.not_found", id=payload.toolId))
  169. return ok(ToolDto.from_entity(entity))
  170. @router.post("/update", response_model=ApiResponse[ToolDto])
  171. def update_tool_contract(
  172. payload: ToolUpdateRequestDto,
  173. service: ToolServiceDep) -> ApiResponse[ToolDto]:
  174. entity = service.update_tool_definition_from_contract(payload)
  175. if entity is None:
  176. raise HTTPException(status_code=404, detail=error_detail("error.tool.not_found", id=payload.toolId))
  177. return ok(ToolDto.from_entity(entity))
  178. @router.post("/delete", response_model=ApiResponse[DeleteData])
  179. def delete_tool_contract(
  180. payload: ToolDeleteRequestDto,
  181. service: ToolServiceDep) -> ApiResponse[DeleteData]:
  182. deleted = service.delete_tool_definition_from_contract(payload)
  183. return ok(DeleteData(deleted=deleted, toolId=payload.toolId))
  184. @router.post("/connections/list", response_model=ApiResponse[PageResult[ToolConnectionDto]])
  185. def list_tool_connections_contract(
  186. payload: ToolConnectionListRequestDto,
  187. service: ToolServiceDep) -> ApiResponse[PageResult[ToolConnectionDto]]:
  188. items = service.list_tool_connections(tool_id=payload.toolId)
  189. page_items = items[payload.offset:payload.offset + payload.pageSize]
  190. return ok(
  191. PageResult[ToolConnectionDto].from_items(
  192. items=[ToolConnectionDto.from_entity(item) for item in page_items],
  193. total=len(items),
  194. page=payload.page,
  195. page_size=payload.pageSize))
  196. @router.post("/connections/create", response_model=ApiResponse[ToolConnectionDto])
  197. def create_tool_connection_contract(
  198. payload: ToolConnectionCreateRequestDto,
  199. service: ToolServiceDep) -> ApiResponse[ToolConnectionDto]:
  200. entity = service.create_tool_connection_from_contract(payload)
  201. return ok(ToolConnectionDto.from_entity(entity))
  202. @router.post("/connections/detail", response_model=ApiResponse[ToolConnectionDto])
  203. def get_tool_connection_contract(
  204. payload: ToolConnectionDetailRequestDto,
  205. service: ToolServiceDep) -> ApiResponse[ToolConnectionDto]:
  206. entity = service.get_tool_connection_from_contract(payload)
  207. if entity is None:
  208. raise HTTPException(
  209. status_code=404,
  210. detail=error_detail("error.tool_connection.not_found", id=payload.connectionId))
  211. return ok(ToolConnectionDto.from_entity(entity))
  212. @router.post("/connections/update", response_model=ApiResponse[ToolConnectionDto])
  213. def update_tool_connection_contract(
  214. payload: ToolConnectionUpdateRequestDto,
  215. service: ToolServiceDep) -> ApiResponse[ToolConnectionDto]:
  216. entity = service.update_tool_connection_from_contract(payload)
  217. if entity is None:
  218. raise HTTPException(
  219. status_code=404,
  220. detail=error_detail("error.tool_connection.not_found", id=payload.connectionId))
  221. return ok(ToolConnectionDto.from_entity(entity))
  222. @router.post("/connections/delete", response_model=ApiResponse[DeleteData])
  223. def delete_tool_connection_contract(
  224. payload: ToolConnectionDeleteRequestDto,
  225. service: ToolServiceDep) -> ApiResponse[DeleteData]:
  226. connection_id = payload.connectionId
  227. deleted = service.delete_tool_connection(connection_id=connection_id)
  228. return ok(DeleteData(deleted=deleted, connectionId=connection_id))
  229. @router.post("/mcp/connect", response_model=ApiResponse[McpConnectData])
  230. def connect_mcp_server_contract(
  231. payload: McpConnectRequestDto,
  232. service: ToolServiceDep) -> ApiResponse[McpConnectData]:
  233. return ok(service.connect_mcp_server(payload))
  234. @router.post("/mcp/discover", response_model=ApiResponse[ToolConnectionDto])
  235. def discover_mcp_server_contract(
  236. payload: McpDiscoverRequestDto,
  237. service: ToolServiceDep) -> ApiResponse[ToolConnectionDto]:
  238. connection = service.discover_mcp_connection(payload)
  239. if connection is None:
  240. raise HTTPException(
  241. status_code=404,
  242. detail=error_detail("error.tool_connection.not_found", id=payload.connectionId))
  243. return ok(ToolConnectionDto.from_entity(connection))
  244. @router.post("/bindings/list", response_model=ApiResponse[PageResult[ToolBindingDto]])
  245. def list_tool_bindings_contract(
  246. payload: ToolBindingListRequestDto,
  247. service: ToolServiceDep) -> ApiResponse[PageResult[ToolBindingDto]]:
  248. items = service.list_tool_bindings(app_id=payload.appId)
  249. page_items = items[payload.offset:payload.offset + payload.pageSize]
  250. return ok(
  251. PageResult[ToolBindingDto].from_items(
  252. items=[ToolBindingDto.from_entity(item) for item in page_items],
  253. total=len(items),
  254. page=payload.page,
  255. page_size=payload.pageSize))
  256. @router.post("/bindings/create", response_model=ApiResponse[ToolBindingDto])
  257. def create_tool_binding_contract(
  258. payload: ToolBindingCreateRequestDto,
  259. service: ToolServiceDep) -> ApiResponse[ToolBindingDto]:
  260. try:
  261. entity = service.create_tool_binding_from_contract(payload)
  262. except ValueError as exc:
  263. raise HTTPException(status_code=422, detail=error_detail("error.validation", message=str(exc))) from exc
  264. return ok(ToolBindingDto.from_entity(entity))
  265. @router.post("/bindings/detail", response_model=ApiResponse[ToolBindingDto])
  266. def get_tool_binding_contract(
  267. payload: ToolBindingDetailRequestDto,
  268. service: ToolServiceDep) -> ApiResponse[ToolBindingDto]:
  269. entity = service.get_tool_binding_from_contract(payload)
  270. if entity is None:
  271. raise HTTPException(
  272. status_code=404,
  273. detail=error_detail("error.tool_binding.not_found", id=payload.bindingId))
  274. return ok(ToolBindingDto.from_entity(entity))
  275. @router.post("/bindings/update", response_model=ApiResponse[ToolBindingDto])
  276. def update_tool_binding_contract(
  277. payload: ToolBindingUpdateRequestDto,
  278. service: ToolServiceDep) -> ApiResponse[ToolBindingDto]:
  279. entity = service.update_tool_binding_from_contract(payload)
  280. if entity is None:
  281. raise HTTPException(
  282. status_code=404,
  283. detail=error_detail("error.tool_binding.not_found", id=payload.bindingId))
  284. return ok(ToolBindingDto.from_entity(entity))
  285. @router.post("/bindings/delete", response_model=ApiResponse[DeleteData])
  286. def delete_tool_binding_contract(
  287. payload: ToolBindingDeleteRequestDto,
  288. service: ToolServiceDep) -> ApiResponse[DeleteData]:
  289. deleted = service.delete_tool_binding(payload)
  290. return ok(DeleteData(deleted=deleted, bindingId=payload.bindingId))
  291. @router.post("/credentials/list", response_model=ApiResponse[PageResult[ToolCredentialDto]])
  292. def list_tool_credentials_contract(
  293. payload: PageRequest,
  294. service: ToolServiceDep) -> ApiResponse[PageResult[ToolCredentialDto]]:
  295. keyword = (payload.keyword or "").lower().strip()
  296. items = [
  297. item
  298. for item in service.list_tool_credentials()
  299. if not keyword
  300. or keyword in item.name.lower()
  301. or keyword in item.credential_type.lower()
  302. ]
  303. page_items = items[payload.offset:payload.offset + payload.pageSize]
  304. return ok(
  305. PageResult[ToolCredentialDto].from_items(
  306. items=[ToolCredentialDto.from_entity(item) for item in page_items],
  307. total=len(items),
  308. page=payload.page,
  309. page_size=payload.pageSize))
  310. @router.post("/credentials/create", response_model=ApiResponse[ToolCredentialDto])
  311. def create_tool_credential_contract(
  312. payload: ToolCredentialCreateRequestDto,
  313. service: ToolServiceDep) -> ApiResponse[ToolCredentialDto]:
  314. entity = service.create_tool_credential_from_contract(payload)
  315. return ok(ToolCredentialDto.from_entity(entity))
  316. @router.post("/credentials/detail", response_model=ApiResponse[ToolCredentialDto])
  317. def get_tool_credential_contract(
  318. payload: ToolCredentialDetailRequestDto,
  319. service: ToolServiceDep) -> ApiResponse[ToolCredentialDto]:
  320. entity = service.get_tool_credential_from_contract(payload)
  321. if entity is None:
  322. raise HTTPException(
  323. status_code=404,
  324. detail=error_detail("error.tool_credential.not_found", id=payload.credentialId))
  325. return ok(ToolCredentialDto.from_entity(entity))
  326. @router.post("/credentials/update", response_model=ApiResponse[ToolCredentialDto])
  327. def update_tool_credential_contract(
  328. payload: ToolCredentialUpdateRequestDto,
  329. service: ToolServiceDep) -> ApiResponse[ToolCredentialDto]:
  330. entity = service.update_tool_credential_from_contract(payload)
  331. if entity is None:
  332. raise HTTPException(
  333. status_code=404,
  334. detail=error_detail("error.tool_credential.not_found", id=payload.credentialId))
  335. return ok(ToolCredentialDto.from_entity(entity))
  336. @router.post("/credentials/delete", response_model=ApiResponse[DeleteData])
  337. def delete_tool_credential_contract(
  338. payload: ToolCredentialDeleteRequestDto,
  339. service: ToolServiceDep) -> ApiResponse[DeleteData]:
  340. deleted = service.delete_tool_credential(payload)
  341. return ok(DeleteData(deleted=deleted, credentialId=payload.credentialId))
  342. @router.post("/credentials/reveal", response_model=ApiResponse[ToolCredentialRevealDto])
  343. def reveal_tool_credential_contract(
  344. payload: ToolCredentialRevealRequestDto,
  345. service: ToolServiceDep) -> ApiResponse[ToolCredentialRevealDto]:
  346. result = service.reveal_tool_credential_from_contract(
  347. credential_id=payload.credentialId)
  348. if result is None:
  349. raise HTTPException(
  350. status_code=404,
  351. detail=error_detail("error.tool_credential.not_found", id=payload.credentialId))
  352. return ok(result)