api_server.py 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723
  1. #! /usr/bin/env python
  2. # -*- coding: utf-8 -*-
  3. # vim:fenc=utf-8
  4. import logging
  5. from argparse import ArgumentParser
  6. import werkzeug.exceptions
  7. from flask import Flask, request, jsonify
  8. from sqlalchemy.orm import sessionmaker
  9. from pqai_agent import configs
  10. from pqai_agent import logging_service, chat_service, prompt_templates
  11. from pqai_agent.data_models.agent_configuration import AgentConfiguration
  12. from pqai_agent.data_models.service_module import ServiceModule
  13. from pqai_agent.history_dialogue_service import HistoryDialogueService
  14. from pqai_agent.user_manager import MySQLUserManager, MySQLUserRelationManager
  15. from pqai_agent.utils.db_utils import create_ai_agent_db_engine
  16. from pqai_agent.utils.prompt_utils import format_agent_profile, format_user_profile
  17. from pqai_agent_server.const import AgentApiConst
  18. from pqai_agent_server.const.status_enum import TestTaskStatus
  19. from pqai_agent_server.dataset_service import DatasetService
  20. from pqai_agent_server.models import MySQLSessionManager
  21. from pqai_agent_server.task_server import TaskManager
  22. from pqai_agent_server.utils import (
  23. run_extractor_prompt,
  24. run_chat_prompt,
  25. run_response_type_prompt,
  26. )
  27. from pqai_agent_server.utils import wrap_response, quit_human_intervention_status
  28. app = Flask('agent_api_server')
  29. logger = logging_service.logger
  30. const = AgentApiConst()
  31. @app.route('/api/listStaffs', methods=['GET'])
  32. def list_staffs():
  33. staff_data = app.user_relation_manager.list_staffs()
  34. return wrap_response(200, data=staff_data)
  35. @app.route('/api/getStaffProfile', methods=['GET'])
  36. def get_staff_profile():
  37. staff_id = request.args['staff_id']
  38. profile = app.user_manager.get_staff_profile(staff_id)
  39. if not profile:
  40. return wrap_response(404, msg='staff not found')
  41. else:
  42. return wrap_response(200, data=profile)
  43. @app.route('/api/getUserProfile', methods=['GET'])
  44. def get_user_profile():
  45. user_id = request.args['user_id']
  46. profile = app.user_manager.get_user_profile(user_id)
  47. if not profile:
  48. resp = {
  49. 'code': 404,
  50. 'msg': 'user not found'
  51. }
  52. else:
  53. resp = {
  54. 'code': 200,
  55. 'msg': 'success',
  56. 'data': profile
  57. }
  58. return jsonify(resp)
  59. @app.route('/api/listUsers', methods=['GET'])
  60. def list_users():
  61. user_name = request.args.get('user_name', None)
  62. user_union_id = request.args.get('user_union_id', None)
  63. if not user_name and not user_union_id:
  64. resp = {
  65. 'code': 400,
  66. 'msg': 'user_name or user_union_id is required'
  67. }
  68. return jsonify(resp)
  69. data = app.user_manager.list_users(user_name=user_name, user_union_id=user_union_id)
  70. return jsonify({'code': 200, 'data': data})
  71. @app.route('/api/getDialogueHistory', methods=['GET'])
  72. def get_dialogue_history():
  73. staff_id = request.args['staff_id']
  74. user_id = request.args['user_id']
  75. recent_minutes = int(request.args.get('recent_minutes', 1440))
  76. dialogue_history = app.history_dialogue_service.get_dialogue_history(staff_id, user_id, recent_minutes)
  77. return jsonify({'code': 200, 'data': dialogue_history})
  78. @app.route('/api/listModels', methods=['GET'])
  79. def list_models():
  80. models = [
  81. {
  82. 'model_type': 'openai_compatible',
  83. 'model_name': chat_service.VOLCENGINE_MODEL_DEEPSEEK_V3,
  84. 'display_name': 'DeepSeek V3 on 火山'
  85. },
  86. {
  87. 'model_type': 'openai_compatible',
  88. 'model_name': chat_service.VOLCENGINE_MODEL_DOUBAO_PRO_32K,
  89. 'display_name': '豆包Pro 32K'
  90. },
  91. {
  92. 'model_type': 'openai_compatible',
  93. 'model_name': chat_service.VOLCENGINE_MODEL_DOUBAO_PRO_1_5,
  94. 'display_name': '豆包Pro 1.5'
  95. },
  96. {
  97. 'model_type': 'openai_compatible',
  98. 'model_name': chat_service.VOLCENGINE_BOT_DEEPSEEK_V3_SEARCH,
  99. 'display_name': 'DeepSeek V3联网 on 火山'
  100. },
  101. {
  102. 'model_type': 'openai_compatible',
  103. 'model_name': chat_service.VOLCENGINE_MODEL_DOUBAO_1_5_VISION_PRO,
  104. 'display_name': '豆包1.5视觉理解Pro'
  105. },
  106. ]
  107. return wrap_response(200, data=models)
  108. @app.route('/api/listScenes', methods=['GET'])
  109. def list_scenes():
  110. scenes = [
  111. {'scene': 'greeting', 'display_name': '问候'},
  112. {'scene': 'chitchat', 'display_name': '闲聊'},
  113. {'scene': 'profile_extractor', 'display_name': '画像提取'},
  114. {'scene': 'response_type_detector', 'display_name': '回复模态判断'},
  115. {'scene': 'custom_debugging', 'display_name': '自定义调试场景'}
  116. ]
  117. return wrap_response(200, data=scenes)
  118. @app.route('/api/getBasePrompt', methods=['GET'])
  119. def get_base_prompt():
  120. scene = request.args['scene']
  121. prompt_map = {
  122. 'greeting': prompt_templates.GENERAL_GREETING_PROMPT,
  123. 'chitchat': prompt_templates.CHITCHAT_PROMPT_COZE,
  124. 'profile_extractor': prompt_templates.USER_PROFILE_EXTRACT_PROMPT_V2,
  125. 'response_type_detector': prompt_templates.RESPONSE_TYPE_DETECT_PROMPT,
  126. 'custom_debugging': '',
  127. }
  128. model_map = {
  129. 'greeting': chat_service.VOLCENGINE_MODEL_DOUBAO_PRO_32K,
  130. 'chitchat': chat_service.VOLCENGINE_MODEL_DOUBAO_PRO_32K,
  131. 'profile_extractor': chat_service.VOLCENGINE_MODEL_DOUBAO_PRO_1_5,
  132. 'response_type_detector': chat_service.VOLCENGINE_MODEL_DOUBAO_PRO_1_5,
  133. 'custom_debugging': chat_service.VOLCENGINE_BOT_DEEPSEEK_V3_SEARCH
  134. }
  135. if scene not in prompt_map:
  136. return wrap_response(404, msg='scene not found')
  137. data = {
  138. 'model_name': model_map[scene],
  139. 'content': prompt_map[scene]
  140. }
  141. return wrap_response(200, data=data)
  142. @app.route('/api/runPrompt', methods=['POST'])
  143. def run_prompt():
  144. try:
  145. req_data = request.json
  146. logger.debug(req_data)
  147. scene = req_data['scene']
  148. if scene == 'profile_extractor':
  149. response = run_extractor_prompt(req_data)
  150. return wrap_response(200, data=response)
  151. elif scene == 'response_type_detector':
  152. response = run_response_type_prompt(req_data)
  153. return wrap_response(200, data=response.choices[0].message.content)
  154. else:
  155. response = run_chat_prompt(req_data)
  156. return wrap_response(200, data=response.choices[0].message.content)
  157. except Exception as e:
  158. logger.error(e)
  159. return wrap_response(500, msg='Error: {}'.format(e))
  160. @app.route('/api/formatForPrompt', methods=['POST'])
  161. def format_data_for_prompt():
  162. try:
  163. req_data = request.json
  164. content = req_data['content']
  165. format_type = req_data['format_type']
  166. if format_type == 'staff_profile':
  167. if not isinstance(content, dict):
  168. return wrap_response(400, msg='staff_profile should be a dict')
  169. response = format_agent_profile(content)
  170. elif format_type == 'user_profile':
  171. if not isinstance(content, dict):
  172. return wrap_response(400, msg='user_profile should be a dict')
  173. response = format_user_profile(content)
  174. elif format_type == 'dialogue':
  175. if not isinstance(content, list):
  176. return wrap_response(400, msg='dialogue should be a list')
  177. from pqai_agent_server.utils.prompt_util import format_dialogue_history
  178. response = format_dialogue_history(content)
  179. else:
  180. return wrap_response(400, msg='Invalid format_type')
  181. return wrap_response(200, data=response)
  182. except Exception as e:
  183. logger.error(e)
  184. return wrap_response(500, msg='Error: {}'.format(e))
  185. @app.route("/api/healthCheck", methods=["GET"])
  186. def health_check():
  187. return wrap_response(200, msg="OK")
  188. @app.route("/api/getStaffSessionSummary", methods=["GET"])
  189. def get_staff_session_summary():
  190. staff_id = request.args.get("staff_id")
  191. status = request.args.get("status", const.DEFAULT_STAFF_STATUS)
  192. page_id = request.args.get("page_id", const.DEFAULT_PAGE_ID)
  193. page_size = request.args.get("page_size", const.DEFAULT_PAGE_SIZE)
  194. # check params
  195. try:
  196. page_id = int(page_id)
  197. page_size = int(page_size)
  198. status = int(status)
  199. except Exception as e:
  200. return wrap_response(404, msg="Invalid parameter: {}".format(e))
  201. staff_session_summary = app.session_manager.get_staff_sessions_summary(
  202. staff_id, page_id, page_size, status
  203. )
  204. if not staff_session_summary:
  205. return wrap_response(404, msg="staff not found")
  206. else:
  207. return wrap_response(200, data=staff_session_summary)
  208. @app.route("/api/getStaffSessionList", methods=["GET"])
  209. def get_staff_session_list():
  210. staff_id = request.args.get("staff_id")
  211. if not staff_id:
  212. return wrap_response(404, msg="staff_id is required")
  213. page_size = request.args.get("page_size", const.DEFAULT_PAGE_SIZE)
  214. page_id = request.args.get("page_id", const.DEFAULT_PAGE_ID)
  215. # check params
  216. try:
  217. page_id = int(page_id)
  218. page_size = int(page_size)
  219. except Exception as e:
  220. return wrap_response(404, msg="Invalid parameter: {}".format(e))
  221. staff_session_list = app.session_manager.get_staff_session_list(staff_id, page_id, page_size)
  222. if not staff_session_list:
  223. return wrap_response(404, msg="staff not found")
  224. return wrap_response(200, data=staff_session_list)
  225. @app.route("/api/getStaffList", methods=["GET"])
  226. def get_staff_list():
  227. page_size = request.args.get("page_size", const.DEFAULT_PAGE_SIZE)
  228. page_id = request.args.get("page_id", const.DEFAULT_PAGE_ID)
  229. # check params
  230. try:
  231. page_id = int(page_id)
  232. page_size = int(page_size)
  233. except Exception as e:
  234. return wrap_response(404, msg="Invalid parameter: {}".format(e))
  235. staff_list = app.user_manager.get_staff_list(page_id, page_size)
  236. if not staff_list:
  237. return wrap_response(404, msg="staff not found")
  238. return wrap_response(200, data=staff_list)
  239. @app.route("/api/getConversationList", methods=["GET"])
  240. def get_conversation_list():
  241. """
  242. 获取staff && user 私聊对话列表
  243. :return:
  244. """
  245. staff_id = request.args.get("staff_id")
  246. user_id = request.args.get("user_id")
  247. if not staff_id or not user_id:
  248. return wrap_response(404, msg="staff_id and user_id are required")
  249. page = request.args.get("page")
  250. response = app.session_manager.get_conversation_list(staff_id, user_id, page, const.DEFAULT_CONVERSATION_SIZE)
  251. return wrap_response(200, data=response)
  252. @app.route("/api/sendMessage", methods=["POST"])
  253. def send_message():
  254. return wrap_response(200, msg="暂不实现功能")
  255. @app.route("/api/quitHumanInterventionStatus", methods=["POST"])
  256. def quit_human_interventions_status():
  257. """
  258. 退出人工介入状态
  259. :return:
  260. """
  261. req_data = request.json
  262. staff_id = req_data["staff_id"]
  263. user_id = req_data["user_id"]
  264. if not user_id or not staff_id:
  265. return wrap_response(404, msg="user_id and staff_id are required")
  266. response = quit_human_intervention_status(user_id, staff_id)
  267. return wrap_response(200, data=response)
  268. ## Agent管理接口
  269. @app.route("/api/getNativeAgentList", methods=["GET"])
  270. def get_native_agent_list():
  271. """
  272. 获取所有的Agent列表
  273. :return:
  274. """
  275. page = request.args.get('page', 1)
  276. page_size = request.args.get('page_size', 50)
  277. create_user = request.args.get('create_user', None)
  278. update_user = request.args.get('update_user', None)
  279. offset = (int(page) - 1) * int(page_size)
  280. with app.session_maker() as session:
  281. query = session.query(AgentConfiguration) \
  282. .filter(AgentConfiguration.is_delete == 0)
  283. if create_user:
  284. query = query.filter(AgentConfiguration.create_user == create_user)
  285. if update_user:
  286. query = query.filter(AgentConfiguration.update_user == update_user)
  287. query = query.offset(offset).limit(int(page_size))
  288. data = query.all()
  289. ret_data = [
  290. {
  291. 'id': agent.id,
  292. 'name': agent.name,
  293. 'display_name': agent.display_name,
  294. 'type': agent.type,
  295. 'execution_model': agent.execution_model,
  296. 'create_time': agent.create_time.strftime('%Y-%m-%d %H:%M:%S'),
  297. 'update_time': agent.update_time.strftime('%Y-%m-%d %H:%M:%S')
  298. }
  299. for agent in data
  300. ]
  301. return wrap_response(200, data=ret_data)
  302. @app.route("/api/getNativeAgentConfiguration", methods=["GET"])
  303. def get_native_agent_configuration():
  304. """
  305. 获取指定Agent的配置
  306. :return:
  307. """
  308. agent_id = request.args.get('agent_id')
  309. if not agent_id:
  310. return wrap_response(404, msg='agent_id is required')
  311. with app.session_maker() as session:
  312. agent = session.query(AgentConfiguration).filter(AgentConfiguration.id == agent_id).first()
  313. if not agent:
  314. return wrap_response(404, msg='Agent not found')
  315. data = {
  316. 'id': agent.id,
  317. 'name': agent.name,
  318. 'display_name': agent.display_name,
  319. 'type': agent.type,
  320. 'execution_model': agent.execution_model,
  321. 'system_prompt': agent.system_prompt,
  322. 'task_prompt': agent.task_prompt,
  323. 'tools': agent.tools,
  324. 'sub_agents': agent.sub_agents,
  325. 'extra_params': agent.extra_params,
  326. 'create_time': agent.create_time.strftime('%Y-%m-%d %H:%M:%S'),
  327. 'update_time': agent.update_time.strftime('%Y-%m-%d %H:%M:%S')
  328. }
  329. return wrap_response(200, data=data)
  330. @app.route("/api/saveNativeAgentConfiguration", methods=["POST"])
  331. def save_native_agent_configuration():
  332. """
  333. 保存Agent配置
  334. :return:
  335. """
  336. req_data = request.json
  337. agent_id = req_data.get('agent_id', None)
  338. name = req_data.get('name')
  339. display_name = req_data.get('display_name', None)
  340. type_ = req_data.get('type', 0)
  341. execution_model = req_data.get('execution_model', None)
  342. system_prompt = req_data.get('system_prompt', None)
  343. task_prompt = req_data.get('task_prompt', None)
  344. tools = req_data.get('tools', [])
  345. sub_agents = req_data.get('sub_agents', [])
  346. extra_params = req_data.get('extra_params', {})
  347. if not name:
  348. return wrap_response(400, msg='name is required')
  349. with app.session_maker() as session:
  350. if agent_id:
  351. agent_id = int(agent_id)
  352. agent = session.query(AgentConfiguration).filter(AgentConfiguration.id == agent_id).first()
  353. if not agent:
  354. return wrap_response(404, msg='Agent not found')
  355. agent.name = name
  356. agent.display_name = display_name
  357. agent.type = type_
  358. agent.execution_model = execution_model
  359. agent.system_prompt = system_prompt
  360. agent.task_prompt = task_prompt
  361. agent.tools = tools
  362. agent.sub_agents = sub_agents
  363. agent.extra_params = extra_params
  364. else:
  365. agent = AgentConfiguration(
  366. name=name,
  367. display_name=display_name,
  368. type=type_,
  369. execution_model=execution_model,
  370. system_prompt=system_prompt,
  371. task_prompt=task_prompt,
  372. tools=tools,
  373. sub_agents=sub_agents,
  374. extra_params=extra_params
  375. )
  376. session.add(agent)
  377. session.commit()
  378. return wrap_response(200, msg='Agent configuration saved successfully', data={'id': agent.id})
  379. @app.route("/api/getModuleList", methods=["GET"])
  380. def get_module_list():
  381. """
  382. 获取所有的模块列表
  383. :return:
  384. """
  385. with app.session_maker() as session:
  386. query = session.query(ServiceModule) \
  387. .filter(ServiceModule.is_delete == 0)
  388. data = query.all()
  389. ret_data = [
  390. {
  391. 'id': module.id,
  392. 'name': module.name,
  393. 'display_name': module.display_name,
  394. 'default_agent_type': module.default_agent_type,
  395. 'default_agent_id': module.default_agent_id,
  396. 'create_time': module.create_time.strftime('%Y-%m-%d %H:%M:%S'),
  397. 'update_time': module.update_time.strftime('%Y-%m-%d %H:%M:%S')
  398. }
  399. for module in data
  400. ]
  401. return wrap_response(200, data=ret_data)
  402. @app.route("/api/getModuleConfiguration", methods=["GET"])
  403. def get_module_configuration():
  404. """
  405. 获取指定模块的配置
  406. :return:
  407. """
  408. module_id = request.args.get('module_id')
  409. if not module_id:
  410. return wrap_response(404, msg='module_id is required')
  411. with app.session_maker() as session:
  412. module = session.query(ServiceModule).filter(ServiceModule.id == module_id).first()
  413. if not module:
  414. return wrap_response(404, msg='Module not found')
  415. data = {
  416. 'id': module.id,
  417. 'name': module.name,
  418. 'display_name': module.display_name,
  419. 'default_agent_type': module.default_agent_type,
  420. 'default_agent_id': module.default_agent_id,
  421. 'create_time': module.create_time.strftime('%Y-%m-%d %H:%M:%S'),
  422. 'updated_time': module.updated_time.strftime('%Y-%m-%d %H:%M:%S')
  423. }
  424. return wrap_response(200, data=data)
  425. @app.route("/api/saveModuleConfiguration", methods=["POST"])
  426. def save_module_configuration():
  427. """
  428. 保存模块配置
  429. :return:
  430. """
  431. req_data = request.json
  432. module_id = req_data.get('module_id', None)
  433. name = req_data.get('name')
  434. display_name = req_data.get('display_name', None)
  435. default_agent_type = req_data.get('default_agent_type', 0)
  436. default_agent_id = req_data.get('default_agent_id', None)
  437. if not name:
  438. return wrap_response(400, msg='name is required')
  439. with app.session_maker() as session:
  440. if module_id:
  441. module_id = int(module_id)
  442. module = session.query(ServiceModule).filter(ServiceModule.id == module_id).first()
  443. if not module:
  444. return wrap_response(404, msg='Module not found')
  445. module.name = name
  446. module.display_name = display_name
  447. module.default_agent_type = default_agent_type
  448. module.default_agent_id = default_agent_id
  449. else:
  450. module = ServiceModule(
  451. name=name,
  452. display_name=display_name,
  453. default_agent_type=default_agent_type,
  454. default_agent_id=default_agent_id
  455. )
  456. session.add(module)
  457. session.commit()
  458. return wrap_response(200, msg='Module configuration saved successfully', data={'id': module.id})
  459. @app.route("/api/getTestTaskList", methods=["GET"])
  460. def get_test_task_list():
  461. """
  462. 获取单元测试任务列表
  463. :return:
  464. """
  465. page_num = request.args.get("pageNum", const.DEFAULT_PAGE_ID)
  466. page_size = request.args.get("pageSize", const.DEFAULT_PAGE_SIZE)
  467. try:
  468. page_num = int(page_num)
  469. page_size = int(page_size)
  470. except Exception as e:
  471. return wrap_response(404, msg="Invalid parameter: {}".format(e))
  472. response = app.task_manager.get_test_task_list(page_num, page_size)
  473. return wrap_response(200, data=response)
  474. @app.route("/api/getTestTaskConversations", methods=["GET"])
  475. def get_test_task_conversations():
  476. """
  477. 获取单元测试对话任务列表
  478. :return:
  479. """
  480. task_id = request.args.get("taskId", None)
  481. if not task_id:
  482. return wrap_response(404, msg='task_id is required')
  483. page_num = request.args.get("pageNum", const.DEFAULT_PAGE_ID)
  484. page_size = request.args.get("pageSize", const.DEFAULT_PAGE_SIZE)
  485. try:
  486. page_num = int(page_num)
  487. page_size = int(page_size)
  488. except Exception as e:
  489. return wrap_response(404, msg="Invalid parameter: {}".format(e))
  490. response = app.task_manager.get_test_task_conversations(int(task_id), page_num, page_size)
  491. return wrap_response(200, data=response)
  492. @app.route("/api/createTestTask", methods=["POST"])
  493. def create_test_task():
  494. """
  495. 创建单元测试任务
  496. :return:
  497. """
  498. req_data = request.json
  499. agent_id = req_data.get('agentId', None)
  500. module_id = req_data.get('moduleId', None)
  501. evaluate_type = req_data.get('evaluateType', None)
  502. if not agent_id:
  503. return wrap_response(404, msg='agent id is required')
  504. if not module_id:
  505. return wrap_response(404, msg='module id is required')
  506. if not evaluate_type:
  507. return wrap_response(404, msg='evaluate_type id is required')
  508. app.task_manager.create_task(agent_id, module_id, evaluate_type)
  509. return wrap_response(200)
  510. @app.route("/api/stopTestTask", methods=["POST"])
  511. def stop_test_task():
  512. """
  513. 停止单元测试任务
  514. :return:
  515. """
  516. req_data = request.json
  517. task_id = req_data.get('taskId', None)
  518. if not task_id:
  519. return wrap_response(400, msg='task id is required')
  520. task = app.task_manager.get_task(task_id)
  521. if task.status not in (TestTaskStatus.NOT_STARTED.value, TestTaskStatus.IN_PROGRESS.value):
  522. return wrap_response(400, msg='task status is invalid')
  523. app.task_manager.cancel_task(task_id)
  524. return wrap_response(200)
  525. @app.route("/api/resumeTestTask", methods=["POST"])
  526. def resume_test_task():
  527. """
  528. 恢复停止的单元测试任务
  529. :return:
  530. """
  531. req_data = request.json
  532. task_id = req_data.get('taskId', None)
  533. if not task_id:
  534. return wrap_response(400, msg='task id is required')
  535. task = app.task_manager.get_task(task_id)
  536. if task.status != TestTaskStatus.CANCELLED.value:
  537. return wrap_response(400, msg='task status is invalid')
  538. app.task_manager.resume_task(task_id)
  539. return wrap_response(200)
  540. @app.route("/api/getDatasetList", methods=["GET"])
  541. def get_dataset_list():
  542. """
  543. 获取数据集列表
  544. :return:
  545. """
  546. page_num = request.args.get("pageNum", const.DEFAULT_PAGE_ID)
  547. page_size = request.args.get("pageSize", const.DEFAULT_PAGE_SIZE)
  548. try:
  549. page_num = int(page_num)
  550. page_size = int(page_size)
  551. except Exception as e:
  552. return wrap_response(404, msg="Invalid parameter: {}".format(e))
  553. response = app.dataset_service.get_dataset_list(page_num, page_size)
  554. return wrap_response(200, data=response)
  555. @app.route("/api/getConversationDataList", methods=["GET"])
  556. def get_conversation_data_list():
  557. """
  558. 获取对话列表
  559. :return:
  560. """
  561. dataset_id = request.args.get("datasetId", None)
  562. if not dataset_id:
  563. return wrap_response(404, msg='dataset_id is required')
  564. page_num = request.args.get("pageNum", const.DEFAULT_PAGE_ID)
  565. page_size = request.args.get("pageSize", const.DEFAULT_PAGE_SIZE)
  566. try:
  567. page_num = int(page_num)
  568. page_size = int(page_size)
  569. except Exception as e:
  570. return wrap_response(404, msg="Invalid parameter: {}".format(e))
  571. response = app.dataset_service.get_conversation_data_list(int(dataset_id), page_num, page_size)
  572. return wrap_response(200, data=response)
  573. @app.errorhandler(werkzeug.exceptions.BadRequest)
  574. def handle_bad_request(e):
  575. logger.error(e)
  576. return wrap_response(400, msg='Bad Request: {}'.format(e.description))
  577. if __name__ == '__main__':
  578. parser = ArgumentParser()
  579. parser.add_argument('--prod', action='store_true')
  580. parser.add_argument('--host', default='127.0.0.1')
  581. parser.add_argument('--port', type=int, default=8083)
  582. parser.add_argument('--log-level', default='INFO')
  583. args = parser.parse_args()
  584. config = configs.get()
  585. logging_level = logging.getLevelName(args.log_level)
  586. logging_service.setup_root_logger(level=logging_level, logfile_name='agent_api_server.log')
  587. # set db config
  588. agent_db_config = config['database']['ai_agent']
  589. growth_db_config = config['database']['growth']
  590. user_db_config = config['storage']['user']
  591. staff_db_config = config['storage']['staff']
  592. agent_state_db_config = config['storage']['agent_state']
  593. chat_history_db_config = config['storage']['chat_history']
  594. # init user manager
  595. user_manager = MySQLUserManager(agent_db_config, growth_db_config, staff_db_config['table'])
  596. app.user_manager = user_manager
  597. # init session manager
  598. session_manager = MySQLSessionManager(
  599. db_config=agent_db_config,
  600. staff_table=staff_db_config['table'],
  601. user_table=user_db_config['table'],
  602. agent_state_table=agent_state_db_config['table'],
  603. chat_history_table=chat_history_db_config['table']
  604. )
  605. app.session_manager = session_manager
  606. agent_db_engine = create_ai_agent_db_engine()
  607. app.session_maker = sessionmaker(bind=agent_db_engine)
  608. dataset_service = DatasetService(session_maker=sessionmaker(bind=agent_db_engine))
  609. app.dataset_service = dataset_service
  610. task_manager = TaskManager(session_maker=sessionmaker(bind=agent_db_engine), dataset_service=dataset_service)
  611. app.task_manager = task_manager
  612. task_manager.recover_tasks()
  613. wecom_db_config = config['storage']['user_relation']
  614. user_relation_manager = MySQLUserRelationManager(
  615. agent_db_config, growth_db_config,
  616. config['storage']['staff']['table'],
  617. user_db_config['table'],
  618. wecom_db_config['table']['staff'],
  619. wecom_db_config['table']['relation'],
  620. wecom_db_config['table']['user']
  621. )
  622. app.user_relation_manager = user_relation_manager
  623. app.history_dialogue_service = HistoryDialogueService(
  624. config['storage']['history_dialogue']['api_base_url']
  625. )
  626. app.run(debug=not args.prod, host=args.host, port=args.port)