routes.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396
  1. from flask import request, jsonify
  2. from app.api.data_interface import bp
  3. from app.models.result import success, failed
  4. from app.core.graph.graph_operations import (
  5. connect_graph,
  6. MyEncoder,
  7. create_or_get_node,
  8. )
  9. from app.core.data_interface import interface
  10. from app.core.meta_data import translate_and_parse, get_formatted_time
  11. from app.core.llm import code_generate_standard
  12. import json
  13. @bp.route('/data/standard/add', methods=['POST'])
  14. def data_standard_add():
  15. try:
  16. receiver = request.get_json()
  17. name_zh = receiver['name_zh']
  18. name_en = translate_and_parse(name_zh)
  19. receiver['name_en'] = name_en[0]
  20. receiver['create_time'] = get_formatted_time()
  21. receiver['tag'] = json.dumps(receiver['tag'], ensure_ascii=False)
  22. create_or_get_node('data_standard', **receiver)
  23. res = success('', "success")
  24. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  25. except Exception as e:
  26. res = failed(str(e), 500, {})
  27. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  28. @bp.route('/data/standard/detail', methods=['POST'])
  29. def data_standard_detail():
  30. try:
  31. receiver = request.get_json()
  32. nodeid = receiver['id'] # id
  33. cql = """MATCH (n:data_standard) where id(n) = $nodeId
  34. RETURN properties(n) as property"""
  35. # Create a session from the driver returned by connect_graph
  36. with connect_graph().session() as session:
  37. result = session.run(cql, nodeId=nodeid).single()
  38. property = result["property"] if result else {}
  39. if "tag" not in property:
  40. property["tag"] = None
  41. else:
  42. property["tag"] = json.loads(property["tag"])
  43. if "describe" not in property:
  44. property["describe"] = None
  45. res = success(property, "success")
  46. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  47. except Exception as e:
  48. res = failed(str(e), 500, {})
  49. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  50. @bp.route('/data/standard/code', methods=['POST'])
  51. def data_standard_code():
  52. try:
  53. receiver = request.get_json()
  54. input = receiver['input']
  55. describe = receiver['describe']
  56. output = receiver['output']
  57. relation = {
  58. "input_params": input,
  59. "output_params": output,
  60. }
  61. result = code_generate_standard(describe, relation)
  62. res = success(result, "success")
  63. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  64. except Exception as e:
  65. res = failed(str(e), 500, {})
  66. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  67. @bp.route('/data/standard/update', methods=['POST'])
  68. def data_standard_update():
  69. try:
  70. receiver = request.get_json()
  71. name_zh = receiver['name_zh']
  72. name_en = translate_and_parse(name_zh)
  73. receiver['name_en'] = name_en[0]
  74. receiver['create_time'] = get_formatted_time()
  75. create_or_get_node('data_standard', **receiver)
  76. res = success('', "success")
  77. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  78. except Exception as e:
  79. res = failed(str(e), 500, {})
  80. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  81. @bp.route('/data/standard/list', methods=['POST'])
  82. def data_standard_list():
  83. try:
  84. receiver = request.get_json()
  85. page = int(receiver.get('current', 1))
  86. page_size = int(receiver.get('size', 10))
  87. name_en_filter = receiver.get('name_en', None)
  88. name_zh_filter = receiver.get('name_zh', None)
  89. category = receiver.get('category', None)
  90. time = receiver.get('time', None)
  91. skip_count = (page - 1) * page_size
  92. data, total = interface.standard_list(
  93. skip_count,
  94. page_size,
  95. name_en_filter,
  96. name_zh_filter,
  97. category,
  98. time,
  99. )
  100. response_data = {
  101. 'records': data,
  102. 'total': total,
  103. 'size': page_size,
  104. 'current': page,
  105. }
  106. res = success(response_data, "success")
  107. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  108. except Exception as e:
  109. res = failed(str(e), 500, {})
  110. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  111. @bp.route('/data/standard/graph/all', methods=['POST'])
  112. def data_standard_graph_all():
  113. try:
  114. receiver = request.get_json()
  115. nodeid = receiver['id']
  116. type = receiver['type'] # kinship/impact/all
  117. if type == 'kinship':
  118. result = interface.standard_kinship_graph(nodeid)
  119. elif type == 'impact':
  120. result = interface.standard_impact_graph(nodeid)
  121. else:
  122. result = interface.standard_all_graph(nodeid)
  123. return json.dumps(
  124. success(result, "success"),
  125. ensure_ascii=False,
  126. cls=MyEncoder,
  127. )
  128. except Exception as e:
  129. res = failed(str(e), 500, {})
  130. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  131. @bp.route('/data/label/add', methods=['POST'])
  132. def data_label_add():
  133. try:
  134. receiver = request.get_json()
  135. name_zh = receiver['name_zh']
  136. name_en = translate_and_parse(name_zh)
  137. receiver['name_en'] = name_en[0]
  138. receiver['create_time'] = get_formatted_time()
  139. create_or_get_node('DataLabel', **receiver)
  140. res = success('', "success")
  141. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  142. except Exception as e:
  143. res = failed(str(e), 500, {})
  144. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  145. @bp.route('/data/label/detail', methods=['POST'])
  146. def data_label_detail():
  147. try:
  148. receiver = request.get_json()
  149. nodeid = receiver['id'] # id
  150. cql = """MATCH (n:DataLabel) where id(n) = $nodeId
  151. RETURN properties(n) as property"""
  152. with connect_graph().session() as session:
  153. result = session.run(cql, nodeId=nodeid).single()
  154. property = result["property"] if result else {}
  155. if "describe" not in property:
  156. property["describe"] = None
  157. res = success(property, "success")
  158. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  159. except Exception as e:
  160. res = failed(str(e), 500, {})
  161. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  162. @bp.route('/data/label/list', methods=['POST'])
  163. def data_label_list():
  164. try:
  165. receiver = request.get_json()
  166. page = int(receiver.get('current', 1))
  167. page_size = int(receiver.get('size', 10))
  168. name_en_filter = receiver.get('name_en', None)
  169. name_zh_filter = receiver.get('name_zh', None)
  170. category = receiver.get('category', None)
  171. group = receiver.get('group', None)
  172. skip_count = (page - 1) * page_size
  173. data, total = interface.label_list(
  174. skip_count,
  175. page_size,
  176. name_en_filter,
  177. name_zh_filter,
  178. category,
  179. group,
  180. )
  181. response_data = {
  182. 'records': data,
  183. 'total': total,
  184. 'size': page_size,
  185. 'current': page,
  186. }
  187. res = success(response_data, "success")
  188. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  189. except Exception as e:
  190. res = failed(str(e), 500, {})
  191. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  192. @bp.route('/labellist', methods=['POST'])
  193. def interface_label_list():
  194. """获取 DataLabel 列表(支持多条件 category_filter 过滤)"""
  195. try:
  196. receiver = request.get_json() or {}
  197. page = int(receiver.get('current', 1))
  198. page_size = int(receiver.get('size', 10))
  199. name_en_filter = receiver.get('name_en')
  200. name_zh_filter = receiver.get('name_zh')
  201. category_filter = receiver.get('category_filter')
  202. group_filter = receiver.get('group')
  203. skip_count = (page - 1) * page_size
  204. data, total = interface.label_list(
  205. skip_count,
  206. page_size,
  207. name_en_filter,
  208. name_zh_filter,
  209. category_filter,
  210. group_filter,
  211. )
  212. # 只保留 id, name_zh, name_en 三个字段
  213. records = [
  214. {
  215. 'id': item.get('id'),
  216. 'name_zh': item.get('name_zh'),
  217. 'name_en': item.get('name_en'),
  218. }
  219. for item in data
  220. ]
  221. response_data = {
  222. 'records': records,
  223. 'total': total,
  224. 'size': page_size,
  225. 'current': page,
  226. }
  227. res = success(response_data, "success")
  228. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  229. except Exception as e:
  230. res = failed(str(e), 500, {})
  231. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  232. @bp.route('/graphall', methods=['POST'])
  233. def interface_graph_all():
  234. """获取完整关系图谱"""
  235. try:
  236. receiver = request.get_json() or {}
  237. domain_id = receiver.get("id")
  238. include_meta = receiver.get("meta", True)
  239. if domain_id is None:
  240. return jsonify(failed("节点ID不能为空"))
  241. try:
  242. domain_id = int(domain_id)
  243. except (ValueError, TypeError):
  244. return jsonify(failed(f"节点ID必须为整数, 收到的是: {domain_id}"))
  245. graph_data = interface.graph_all(domain_id, include_meta)
  246. return jsonify(success(graph_data))
  247. except Exception as e:
  248. return jsonify(failed("获取图谱失败", error=str(e)))
  249. @bp.route('/data/label/dynamic/identify', methods=['POST'])
  250. def data_label_dynamic_identify():
  251. try:
  252. receiver = request.get_json()
  253. name_filter = receiver.get('content', None)
  254. data = interface.dynamic_label_list(name_filter)
  255. res = success(data, "success")
  256. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  257. except Exception as e:
  258. res = failed(str(e), 500, {})
  259. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  260. @bp.route('/data/label/graph/all', methods=['POST'])
  261. def data_label_graph():
  262. try:
  263. receiver = request.get_json()
  264. nodeid = receiver['id']
  265. type = receiver['type'] # kinship/impact/all
  266. if type == 'kinship':
  267. result = interface.label_kinship_graph(nodeid)
  268. elif type == 'impact':
  269. result = interface.label_impact_graph(nodeid)
  270. else:
  271. result = interface.label_kinship_graph(nodeid)
  272. return json.dumps(
  273. success(result, "success"),
  274. ensure_ascii=False,
  275. cls=MyEncoder,
  276. )
  277. except Exception as e:
  278. res = failed(str(e), 500, {})
  279. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  280. @bp.route('/metric/label/standard/delete', methods=['POST'])
  281. def metric_label_standard_delete():
  282. try:
  283. receiver = request.get_json()
  284. sourceid = receiver['sourceid']
  285. targetid = receiver['targetid']
  286. cql = """
  287. MATCH (source)-[r]-(target)
  288. WHERE id(source) = $sourceid AND id(target) = $targetid
  289. DELETE r
  290. """
  291. with connect_graph().session() as session:
  292. session.run(cql, sourceid=sourceid, targetid=targetid)
  293. res = success("", "success")
  294. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  295. except Exception as e:
  296. res = failed(str(e), 500, {})
  297. return json.dumps(res, ensure_ascii=False, cls=MyEncoder)
  298. @bp.route('/data/label/delete', methods=['POST'])
  299. def data_label_delete():
  300. """Delete data label node"""
  301. try:
  302. receiver = request.get_json()
  303. node_id = receiver.get('id') if receiver else None
  304. if not node_id:
  305. return jsonify(failed("node id is required", 400, {}))
  306. try:
  307. node_id = int(node_id)
  308. except (ValueError, TypeError):
  309. return jsonify(failed("node id must be an integer", 400, {}))
  310. delete_result = interface.node_delete(node_id)
  311. message = delete_result.get("message") or ""
  312. if delete_result.get("success"):
  313. res = success(
  314. {
  315. "id": node_id,
  316. "message": message,
  317. },
  318. "delete success",
  319. )
  320. else:
  321. res = failed(
  322. message,
  323. 500,
  324. {
  325. "id": node_id,
  326. "message": message,
  327. },
  328. )
  329. return jsonify(res)
  330. except Exception as e:
  331. return jsonify(failed(f"delete failed: {str(e)}", 500, {}))