routes.py 60 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945
  1. from flask import jsonify, request, make_response, Blueprint, current_app, send_file
  2. from app.api.data_parse import bp
  3. from app.core.data_parse.parse_system import (
  4. update_business_card,
  5. get_business_cards,
  6. update_business_card_status,
  7. create_talent_tag,
  8. get_talent_tag_list,
  9. update_talent_tag,
  10. delete_talent_tag,
  11. query_neo4j_graph,
  12. talent_get_tags,
  13. talent_update_tags,
  14. get_business_card,
  15. search_business_cards_by_mobile,
  16. get_duplicate_records,
  17. process_duplicate_record,
  18. get_duplicate_record_detail,
  19. fix_broken_duplicate_records
  20. )
  21. # 导入解析任务相关函数
  22. from app.core.data_parse.parse_task import (
  23. get_parse_tasks,
  24. get_parse_task_detail
  25. )
  26. # 导入酒店管理相关函数
  27. from app.core.data_parse.hotel_management import (
  28. get_hotel_positions_list,
  29. add_hotel_positions,
  30. update_hotel_positions,
  31. query_hotel_positions,
  32. delete_hotel_positions,
  33. get_hotel_group_brands_list,
  34. add_hotel_group_brands,
  35. update_hotel_group_brands,
  36. query_hotel_group_brands,
  37. delete_hotel_group_brands
  38. )
  39. # 导入新的名片图片解析函数和添加名片函数
  40. from app.core.data_parse.parse_card import process_business_card_image, add_business_card, delete_business_card
  41. # 导入网页文本解析函数
  42. from app.core.data_parse.parse_web import process_webpage_with_QWen, add_webpage_talent
  43. from app.config.config import DevelopmentConfig, ProductionConfig
  44. import logging
  45. import boto3
  46. from botocore.config import Config
  47. from botocore.exceptions import ClientError
  48. from io import BytesIO
  49. import base64
  50. import os
  51. import urllib.parse
  52. from minio import Minio
  53. # Define logger
  54. logger = logging.getLogger(__name__)
  55. # For failure responses
  56. def failed(message, code=500):
  57. return {
  58. 'success': False,
  59. 'message': message,
  60. 'data': None
  61. }, code
  62. # 根据环境选择配置
  63. if os.environ.get('FLASK_ENV') == 'production':
  64. config = ProductionConfig()
  65. else:
  66. config = DevelopmentConfig()
  67. # 使用配置变量
  68. minio_url = f"{'https' if config.MINIO_SECURE else 'http'}://{config.MINIO_HOST}"
  69. minio_access_key = config.MINIO_USER
  70. minio_secret_key = config.MINIO_PASSWORD
  71. minio_bucket = config.MINIO_BUCKET
  72. use_ssl = config.MINIO_SECURE
  73. def get_minio_client():
  74. """获取 MinIO 客户端实例"""
  75. return Minio(
  76. '192.168.3.143:9000',
  77. access_key=config.MINIO_USER,
  78. secret_key=config.MINIO_PASSWORD,
  79. secure=config.MINIO_SECURE
  80. )
  81. # 名片解析接口
  82. @bp.route('/business-card-parse', methods=['POST'])
  83. def parse_business_card_route():
  84. """
  85. 解析名片图片并提取信息的API接口(仅解析,不保存到数据库)
  86. 请求参数:
  87. - image: 名片图片文件 (multipart/form-data)
  88. 返回:
  89. - JSON: 包含提取的名片信息和处理状态
  90. 注意:此接口仅负责图片解析和信息提取,不会将数据保存到数据库
  91. """
  92. # 检查是否上传了文件
  93. if 'image' not in request.files:
  94. return jsonify({
  95. 'success': False,
  96. 'message': '未上传图片',
  97. 'data': None
  98. }), 400
  99. image_file = request.files['image']
  100. # 检查文件是否为空
  101. if image_file.filename == '':
  102. return jsonify({
  103. 'success': False,
  104. 'message': '未选择文件',
  105. 'data': None
  106. }), 400
  107. # 检查文件类型是否为图片
  108. if not image_file.content_type.startswith('image/'):
  109. return jsonify({
  110. 'success': False,
  111. 'message': '上传的文件不是图片',
  112. 'data': None
  113. }), 400
  114. # 处理名片图片
  115. result = process_business_card_image(image_file)
  116. if result['success']:
  117. return jsonify(result), 200
  118. else:
  119. return jsonify(result), 500
  120. # 添加名片记录接口
  121. @bp.route('/add-business-card', methods=['POST'])
  122. def add_business_card_route():
  123. """
  124. 添加名片记录的API接口(解析图片并保存到数据库)
  125. 请求参数:
  126. - card_data: 名片信息数据 (JSON格式,可以通过form-data或JSON body传递)
  127. - image: 名片图片文件 (multipart/form-data,可选)
  128. 返回:
  129. - JSON: 包含保存结果和处理状态
  130. 注意:此接口负责业务逻辑处理,包括重复检查、MinIO上传和数据库保存
  131. """
  132. try:
  133. # 获取名片数据 - 支持两种方式
  134. card_data = None
  135. # 方式1:通过JSON body传递
  136. if request.is_json:
  137. card_data = request.get_json()
  138. # 方式2:通过form-data传递card_data字段
  139. elif 'card_data' in request.form:
  140. import json
  141. try:
  142. card_data = json.loads(request.form['card_data'])
  143. except json.JSONDecodeError:
  144. return jsonify({
  145. 'success': False,
  146. 'message': 'card_data格式错误,必须是有效的JSON字符串',
  147. 'data': None
  148. }), 400
  149. # 检查是否提供了名片数据
  150. if not card_data:
  151. return jsonify({
  152. 'success': False,
  153. 'message': '未提供名片数据,请通过JSON body或form-data的card_data字段传递',
  154. 'data': None
  155. }), 400
  156. # 获取可选的图片文件
  157. image_file = None
  158. if 'image' in request.files:
  159. image_file = request.files['image']
  160. # 检查文件是否为空
  161. if image_file.filename == '':
  162. image_file = None
  163. # 检查文件类型是否为图片
  164. elif not image_file.content_type.startswith('image/'):
  165. return jsonify({
  166. 'success': False,
  167. 'message': '上传的文件不是图片',
  168. 'data': None
  169. }), 400
  170. # 调用业务逻辑函数处理名片数据
  171. result = add_business_card(card_data, image_file)
  172. # 根据处理结果设置HTTP状态码
  173. if result['success']:
  174. if result['code'] == 200:
  175. status_code = 200
  176. elif result['code'] == 202:
  177. status_code = 202 # Accepted - 创建成功但有疑似重复记录
  178. else:
  179. status_code = 200
  180. else:
  181. if result['code'] == 400:
  182. status_code = 400
  183. else:
  184. status_code = 500
  185. return jsonify(result), status_code
  186. except Exception as e:
  187. logger.error(f"添加名片记录失败: {str(e)}")
  188. return jsonify({
  189. 'success': False,
  190. 'message': f'添加名片记录失败: {str(e)}',
  191. 'data': None
  192. }), 500
  193. # 更新名片信息接口
  194. @bp.route('/business-cards/<int:card_id>', methods=['PUT'])
  195. def update_business_card_route(card_id):
  196. """
  197. 更新名片信息的API接口
  198. 路径参数:
  199. - card_id: 名片记录ID
  200. 请求参数:
  201. - JSON格式的名片信息
  202. 返回:
  203. - JSON: 包含更新后的名片信息和处理状态
  204. """
  205. # 获取请求数据
  206. data = request.json
  207. if not data:
  208. return jsonify({
  209. 'success': False,
  210. 'message': '请求数据为空',
  211. 'data': None
  212. }), 400
  213. # 调用业务逻辑函数处理更新
  214. result = update_business_card(card_id, data)
  215. # 根据处理结果设置HTTP状态码
  216. status_code = 200 if result['success'] else 500
  217. if 'not found' in result.get('message', '').lower() or '未找到' in result.get('message', ''):
  218. status_code = 404
  219. return jsonify(result), status_code
  220. # 获取所有名片记录的API接口
  221. @bp.route('/get-business-cards', methods=['GET'])
  222. def get_business_cards_route():
  223. """
  224. 获取所有名片记录的API接口
  225. 返回:
  226. - JSON: 包含名片记录列表和处理状态
  227. """
  228. # 调用业务逻辑函数获取名片列表
  229. result = get_business_cards()
  230. # 根据处理结果设置HTTP状态码
  231. status_code = 200 if result['success'] else 500
  232. return jsonify(result), status_code
  233. @bp.route('/update-business-cards/<int:card_id>/status', methods=['PUT'])
  234. def update_business_card_status_route(card_id):
  235. """
  236. 更新名片状态的API接口
  237. 路径参数:
  238. - card_id: 名片记录ID
  239. 请求参数:
  240. - JSON格式,包含status字段
  241. 返回:
  242. - JSON: 包含更新后的名片信息和处理状态
  243. """
  244. # 获取请求数据
  245. data = request.json
  246. if not data or 'status' not in data:
  247. return jsonify({
  248. 'success': False,
  249. 'message': '请求数据为空或缺少status字段',
  250. 'data': None
  251. }), 400
  252. status = data['status']
  253. # 调用业务逻辑函数处理状态更新
  254. result = update_business_card_status(card_id, status)
  255. # 根据处理结果设置HTTP状态码
  256. status_code = 200 if result['success'] else 500
  257. if 'not found' in result.get('message', '').lower() or '未找到' in result.get('message', ''):
  258. status_code = 404
  259. return jsonify(result), status_code
  260. # 从MinIO获取名片图片的API接口
  261. @bp.route('/business-cards/image/<path:image_path>', methods=['GET'])
  262. def get_business_card_image(image_path):
  263. """
  264. 从MinIO获取名片图片的API接口
  265. 路径参数:
  266. - image_path: MinIO中的图片路径
  267. 返回:
  268. - 图片数据流
  269. """
  270. try:
  271. # 记录下载请求信息,便于调试
  272. logger.info(f"获取名片图片请求: {image_path}")
  273. # 获取 MinIO 客户端
  274. minio_client = get_minio_client()
  275. if not minio_client:
  276. return jsonify(failed("MinIO客户端初始化失败")), 500
  277. try:
  278. # 使用正确的MinIO客户端方法
  279. data = minio_client.get_object(minio_bucket, image_path)
  280. # 创建内存文件流
  281. file_stream = BytesIO(data.read())
  282. # 获取文件名
  283. file_name = image_path.split('/')[-1]
  284. # 返回文件
  285. return send_file(
  286. file_stream,
  287. as_attachment=False, # 设置为False,让浏览器直接显示图片
  288. download_name=file_name,
  289. mimetype='image/jpeg' # 根据实际图片类型设置
  290. )
  291. except Exception as e:
  292. logger.error(f"MinIO获取文件失败: {str(e)}")
  293. return jsonify(failed(f"文件获取失败: {str(e)}")), 404
  294. except Exception as e:
  295. logger.error(f"文件下载失败: {str(e)}")
  296. return jsonify(failed(str(e))), 500
  297. finally:
  298. # 确保关闭数据流
  299. if 'data' in locals():
  300. data.close()
  301. # 创建人才标签接口
  302. @bp.route('/create-talent-tag', methods=['POST'])
  303. def create_talent_tag_route():
  304. """
  305. 创建人才标签的API接口
  306. 请求参数:
  307. - JSON格式,包含以下字段:
  308. - name: 标签名称
  309. - category: 标签分类
  310. - description: 标签描述
  311. - status: 启用状态,默认为'active'
  312. 返回:
  313. - JSON: 包含创建结果和标签信息
  314. """
  315. try:
  316. # 获取请求数据
  317. data = request.get_json()
  318. if not data:
  319. return jsonify({
  320. 'success': False,
  321. 'message': '请求数据为空',
  322. 'data': None
  323. }), 400
  324. # 验证必要字段
  325. if 'name' not in data or not data['name']:
  326. return jsonify({
  327. 'success': False,
  328. 'message': '标签名称不能为空',
  329. 'data': None
  330. }), 400
  331. # 处理分类字段,如果未提供则设置默认值
  332. if 'category' not in data or not data['category']:
  333. data['category'] = '未分类'
  334. # 调用业务逻辑函数处理创建
  335. result = create_talent_tag(data)
  336. # 根据处理结果设置HTTP状态码
  337. status_code = 200 if result['success'] else 500
  338. return jsonify(result), status_code
  339. except Exception as e:
  340. logger.error(f"创建人才标签失败: {str(e)}")
  341. return jsonify({
  342. 'success': False,
  343. 'message': f'创建人才标签失败: {str(e)}',
  344. 'data': None
  345. }), 500
  346. # 获取人才标签列表接口
  347. @bp.route('/get-talent-tag-list', methods=['GET'])
  348. def get_talent_tag_list_route():
  349. """
  350. 获取人才标签列表的API接口
  351. 返回:
  352. - JSON: 包含人才标签列表和处理状态
  353. """
  354. try:
  355. # 调用业务逻辑函数获取人才标签列表
  356. result = get_talent_tag_list()
  357. # 根据处理结果设置HTTP状态码
  358. status_code = 200 if result['success'] else 500
  359. return jsonify(result), status_code
  360. except Exception as e:
  361. logger.error(f"获取人才标签列表失败: {str(e)}")
  362. return jsonify({
  363. 'success': False,
  364. 'message': f'获取人才标签列表失败: {str(e)}',
  365. 'data': []
  366. }), 500
  367. # 更新人才标签接口
  368. @bp.route('/update-talent-tag/<int:tag_id>', methods=['PUT'])
  369. def update_talent_tag_route(tag_id):
  370. """
  371. 更新人才标签的API接口
  372. 路径参数:
  373. - tag_id: 标签节点ID
  374. 请求参数:
  375. - JSON格式,可能包含以下字段:
  376. - name: 标签名称
  377. - category: 标签分类
  378. - description: 标签描述
  379. - status: 启用状态
  380. 返回:
  381. - JSON: 包含更新结果和标签信息
  382. """
  383. try:
  384. # 获取请求数据
  385. data = request.get_json()
  386. if not data:
  387. return jsonify({
  388. 'success': False,
  389. 'message': '请求数据为空',
  390. 'data': None
  391. }), 400
  392. # 调用业务逻辑函数处理更新
  393. result = update_talent_tag(tag_id, data)
  394. # 根据处理结果设置HTTP状态码
  395. if not result['success']:
  396. if result['code'] == 404:
  397. status_code = 404
  398. elif result['code'] == 400:
  399. status_code = 400
  400. else:
  401. status_code = 500
  402. else:
  403. status_code = 200
  404. return jsonify(result), status_code
  405. except Exception as e:
  406. logger.error(f"更新人才标签失败: {str(e)}")
  407. return jsonify({
  408. 'success': False,
  409. 'message': f'更新人才标签失败: {str(e)}',
  410. 'data': None
  411. }), 500
  412. # 删除人才标签接口
  413. @bp.route('/delete-talent-tag/<int:tag_id>', methods=['DELETE'])
  414. def delete_talent_tag_route(tag_id):
  415. """
  416. 删除人才标签的API接口
  417. 路径参数:
  418. - tag_id: 标签节点ID
  419. 返回:
  420. - JSON: 包含删除结果和被删除的标签信息
  421. """
  422. try:
  423. # 调用业务逻辑函数执行删除
  424. result = delete_talent_tag(tag_id)
  425. # 根据处理结果设置HTTP状态码
  426. if not result['success']:
  427. if result['code'] == 404:
  428. status_code = 404
  429. else:
  430. status_code = 500
  431. else:
  432. status_code = 200
  433. return jsonify(result), status_code
  434. except Exception as e:
  435. logger.error(f"删除人才标签失败: {str(e)}")
  436. return jsonify({
  437. 'success': False,
  438. 'message': f'删除人才标签失败: {str(e)}',
  439. 'data': None
  440. }), 500
  441. @bp.route('/query-kg', methods=['POST'])
  442. def query_kg():
  443. """
  444. 查询知识图谱API接口
  445. 请求参数:
  446. - query_requirement: 查询需求描述(JSON格式)
  447. 返回:
  448. - JSON: 包含查询结果和处理状态
  449. """
  450. try:
  451. # 获取请求数据
  452. data = request.json
  453. if not data or 'query_requirement' not in data:
  454. return jsonify({
  455. 'code': 400,
  456. 'success': False,
  457. 'message': '请求数据为空或缺少query_requirement字段',
  458. 'data': []
  459. }), 400
  460. query_requirement = data['query_requirement']
  461. # 调用业务逻辑函数执行查询
  462. result = query_neo4j_graph(query_requirement)
  463. # 根据处理结果设置HTTP状态码
  464. status_code = 200 if result['success'] else 500
  465. return jsonify(result), status_code
  466. except Exception as e:
  467. logger.error(f"查询知识图谱失败: {str(e)}")
  468. return jsonify({
  469. 'code': 500,
  470. 'success': False,
  471. 'message': f"查询知识图谱失败: {str(e)}",
  472. 'data': []
  473. }), 500
  474. @bp.route('/talent-get-tags/<int:talent_id>', methods=['GET'])
  475. def talent_get_tags_route(talent_id):
  476. """
  477. 获取人才标签的API接口
  478. 路径参数:
  479. - talent_id: 人才节点ID
  480. 返回:
  481. - JSON: 包含人才关联的标签列表和处理状态
  482. """
  483. try:
  484. # 调用业务逻辑函数获取人才标签
  485. result = talent_get_tags(talent_id)
  486. # 根据处理结果设置HTTP状态码
  487. status_code = 200 if result['success'] else 500
  488. return jsonify(result), status_code
  489. except Exception as e:
  490. logger.error(f"获取人才标签失败: {str(e)}")
  491. return jsonify({
  492. 'code': 500,
  493. 'success': False,
  494. 'message': f"获取人才标签失败: {str(e)}",
  495. 'data': []
  496. }), 500
  497. @bp.route('/talent-update-tags', methods=['POST'])
  498. def talent_update_tags_route():
  499. """
  500. 更新人才标签关系的API接口
  501. 请求参数:
  502. - JSON数组,包含talent和tag字段的对象列表
  503. 例如: [
  504. {"talent": 12345, "tag": "市场营销"},
  505. {"talent": 12345, "tag": "酒店管理"}
  506. ]
  507. 返回:
  508. - JSON: 包含更新结果的状态信息
  509. """
  510. try:
  511. # 获取请求数据
  512. data = request.json
  513. if not data:
  514. return jsonify({
  515. 'code': 400,
  516. 'success': False,
  517. 'message': '请求数据为空',
  518. 'data': None
  519. }), 400
  520. # 调用业务逻辑函数处理标签关系更新
  521. result = talent_update_tags(data)
  522. # 根据处理结果设置HTTP状态码
  523. if result['code'] == 200:
  524. status_code = 200
  525. elif result['code'] == 206:
  526. status_code = 206 # Partial Content
  527. elif result['code'] == 400:
  528. status_code = 400 # Bad Request
  529. elif result['code'] == 404:
  530. status_code = 404 # Not Found
  531. else:
  532. status_code = 500 # Internal Server Error
  533. return jsonify(result), status_code
  534. except Exception as e:
  535. logger.error(f"更新人才标签关系失败: {str(e)}")
  536. return jsonify({
  537. 'code': 500,
  538. 'success': False,
  539. 'message': f"更新人才标签关系失败: {str(e)}",
  540. 'data': None
  541. }), 500
  542. # 测试MinIO连接
  543. def test_minio_connection():
  544. """测试MinIO连接是否正常"""
  545. try:
  546. client = get_minio_client()
  547. if client.bucket_exists(minio_bucket):
  548. return {
  549. 'success': True,
  550. 'message': f'连接MinIO服务器成功,存储桶 {minio_bucket} 存在',
  551. 'config': {
  552. 'host': config.MINIO_HOST,
  553. 'bucket': minio_bucket,
  554. 'secure': use_ssl
  555. }
  556. }
  557. else:
  558. return {
  559. 'success': False,
  560. 'message': f'连接MinIO服务器成功,但存储桶 {minio_bucket} 不存在',
  561. 'config': {
  562. 'host': config.MINIO_HOST,
  563. 'bucket': minio_bucket,
  564. 'secure': use_ssl
  565. }
  566. }
  567. except Exception as e:
  568. return {
  569. 'success': False,
  570. 'message': f'连接MinIO服务器失败: {str(e)}',
  571. 'config': {
  572. 'host': config.MINIO_HOST,
  573. 'bucket': minio_bucket,
  574. 'secure': use_ssl
  575. }
  576. }
  577. # MinIO测试接口
  578. @bp.route('/test-minio-connection', methods=['GET'])
  579. def test_minio_connection_route():
  580. """
  581. 测试MinIO连接的API接口
  582. 返回:
  583. - JSON: 包含连接测试结果
  584. """
  585. try:
  586. result = test_minio_connection()
  587. status_code = 200 if result['success'] else 500
  588. return jsonify(result), status_code
  589. except Exception as e:
  590. logger.error(f"测试MinIO连接失败: {str(e)}")
  591. return jsonify({
  592. 'success': False,
  593. 'message': f'测试MinIO连接失败: {str(e)}',
  594. 'config': {
  595. 'host': config.MINIO_HOST,
  596. 'bucket': minio_bucket,
  597. 'secure': use_ssl
  598. }
  599. }), 500
  600. # 获取单个名片记录的API接口
  601. @bp.route('/get-business-card/<int:card_id>', methods=['GET'])
  602. def get_business_card_route(card_id):
  603. """
  604. 获取单个名片记录的API接口
  605. 路径参数:
  606. - card_id: 名片记录ID
  607. 返回:
  608. - JSON: 包含名片记录信息和处理状态
  609. """
  610. # 调用业务逻辑函数获取名片记录
  611. result = get_business_card(card_id)
  612. # 根据处理结果设置HTTP状态码
  613. if not result['success']:
  614. if result['code'] == 404:
  615. status_code = 404
  616. else:
  617. status_code = 500
  618. else:
  619. status_code = 200
  620. return jsonify(result), status_code
  621. @bp.route('/search-business-cards-by-mobile', methods=['GET'])
  622. def search_business_cards_by_mobile_route():
  623. """
  624. 根据手机号码搜索名片记录的API接口
  625. 查询参数:
  626. - mobile: 要搜索的手机号码
  627. 返回:
  628. - JSON: 包含搜索到的名片记录列表和处理状态
  629. 示例:
  630. GET /search-business-cards-by-mobile?mobile=13800138000
  631. """
  632. try:
  633. # 获取查询参数
  634. mobile_number = request.args.get('mobile', '').strip()
  635. if not mobile_number:
  636. return jsonify({
  637. 'success': False,
  638. 'message': '请提供要搜索的手机号码',
  639. 'data': []
  640. }), 400
  641. # 调用业务逻辑函数搜索名片记录
  642. result = search_business_cards_by_mobile(mobile_number)
  643. # 根据处理结果设置HTTP状态码
  644. if result['code'] == 200:
  645. status_code = 200
  646. elif result['code'] == 400:
  647. status_code = 400
  648. else:
  649. status_code = 500
  650. return jsonify(result), status_code
  651. except Exception as e:
  652. # 处理未预期的异常
  653. error_msg = f"根据手机号码搜索名片时发生错误: {str(e)}"
  654. logger.error(error_msg, exc_info=True)
  655. return jsonify({
  656. 'success': False,
  657. 'message': error_msg,
  658. 'data': []
  659. }), 500
  660. @bp.route('/get-hotel-positions-list', methods=['GET'])
  661. def get_hotel_positions_list_route():
  662. """
  663. 获取酒店职位数据表全部记录的API接口
  664. 返回:
  665. - JSON: 包含酒店职位记录列表和处理状态
  666. """
  667. try:
  668. # 调用业务逻辑函数获取酒店职位列表
  669. result = get_hotel_positions_list()
  670. # 根据处理结果设置HTTP状态码
  671. status_code = 200 if result['success'] else 500
  672. return jsonify(result), status_code
  673. except Exception as e:
  674. # 处理未预期的异常
  675. error_msg = f"获取酒店职位列表时发生错误: {str(e)}"
  676. logger.error(error_msg, exc_info=True)
  677. return jsonify({
  678. 'success': False,
  679. 'message': error_msg,
  680. 'data': [],
  681. 'count': 0
  682. }), 500
  683. @bp.route('/add-hotel-positions', methods=['POST'])
  684. def add_hotel_positions_route():
  685. """
  686. 新增酒店职位数据表记录的API接口
  687. 请求参数:
  688. - JSON格式,包含以下字段:
  689. - department_zh: 部门中文名称 (必填)
  690. - department_en: 部门英文名称 (必填)
  691. - position_zh: 职位中文名称 (必填)
  692. - position_en: 职位英文名称 (必填)
  693. - position_abbr: 职位英文缩写 (可选)
  694. - level_zh: 职级中文名称 (必填)
  695. - level_en: 职级英文名称 (必填)
  696. - created_by: 创建者 (可选)
  697. - updated_by: 更新者 (可选)
  698. - status: 状态 (可选)
  699. 返回:
  700. - JSON: 包含创建结果和职位信息
  701. """
  702. try:
  703. # 获取请求数据
  704. data = request.get_json()
  705. if not data:
  706. return jsonify({
  707. 'success': False,
  708. 'message': '请求数据为空',
  709. 'data': None
  710. }), 400
  711. # 调用业务逻辑函数处理创建
  712. result = add_hotel_positions(data)
  713. # 根据处理结果设置HTTP状态码
  714. if result['code'] == 200:
  715. status_code = 201 # Created
  716. elif result['code'] == 400:
  717. status_code = 400 # Bad Request
  718. elif result['code'] == 409:
  719. status_code = 409 # Conflict
  720. else:
  721. status_code = 500 # Internal Server Error
  722. return jsonify(result), status_code
  723. except Exception as e:
  724. # 处理未预期的异常
  725. error_msg = f"创建酒店职位记录时发生错误: {str(e)}"
  726. logger.error(error_msg, exc_info=True)
  727. return jsonify({
  728. 'success': False,
  729. 'message': error_msg,
  730. 'data': None
  731. }), 500
  732. @bp.route('/update-hotel-positions/<int:position_id>', methods=['PUT'])
  733. def update_hotel_positions_route(position_id):
  734. """
  735. 修改酒店职位数据表记录的API接口
  736. 路径参数:
  737. - position_id: 职位记录ID
  738. 请求参数:
  739. - JSON格式,可能包含以下字段:
  740. - department_zh: 部门中文名称
  741. - department_en: 部门英文名称
  742. - position_zh: 职位中文名称
  743. - position_en: 职位英文名称
  744. - position_abbr: 职位英文缩写
  745. - level_zh: 职级中文名称
  746. - level_en: 职级英文名称
  747. - updated_by: 更新者
  748. - status: 状态
  749. 返回:
  750. - JSON: 包含更新结果和职位信息
  751. """
  752. try:
  753. # 获取请求数据
  754. data = request.get_json()
  755. if not data:
  756. return jsonify({
  757. 'success': False,
  758. 'message': '请求数据为空',
  759. 'data': None
  760. }), 400
  761. # 调用业务逻辑函数处理更新
  762. result = update_hotel_positions(position_id, data)
  763. # 根据处理结果设置HTTP状态码
  764. if result['code'] == 200:
  765. status_code = 200 # OK
  766. elif result['code'] == 400:
  767. status_code = 400 # Bad Request
  768. elif result['code'] == 404:
  769. status_code = 404 # Not Found
  770. elif result['code'] == 409:
  771. status_code = 409 # Conflict
  772. else:
  773. status_code = 500 # Internal Server Error
  774. return jsonify(result), status_code
  775. except Exception as e:
  776. # 处理未预期的异常
  777. error_msg = f"更新酒店职位记录时发生错误: {str(e)}"
  778. logger.error(error_msg, exc_info=True)
  779. return jsonify({
  780. 'success': False,
  781. 'message': error_msg,
  782. 'data': None
  783. }), 500
  784. @bp.route('/query-hotel-positions/<int:position_id>', methods=['GET'])
  785. def query_hotel_positions_route(position_id):
  786. """
  787. 查找指定ID的酒店职位数据表记录的API接口
  788. 路径参数:
  789. - position_id: 职位记录ID
  790. 返回:
  791. - JSON: 包含查找结果和职位信息
  792. """
  793. try:
  794. # 调用业务逻辑函数查找职位记录
  795. result = query_hotel_positions(position_id)
  796. # 根据处理结果设置HTTP状态码
  797. if result['code'] == 200:
  798. status_code = 200 # OK
  799. elif result['code'] == 404:
  800. status_code = 404 # Not Found
  801. else:
  802. status_code = 500 # Internal Server Error
  803. return jsonify(result), status_code
  804. except Exception as e:
  805. # 处理未预期的异常
  806. error_msg = f"查找酒店职位记录时发生错误: {str(e)}"
  807. logger.error(error_msg, exc_info=True)
  808. return jsonify({
  809. 'success': False,
  810. 'message': error_msg,
  811. 'data': None
  812. }), 500
  813. @bp.route('/delete-hotel-positions/<int:position_id>', methods=['DELETE'])
  814. def delete_hotel_positions_route(position_id):
  815. """
  816. 删除指定ID的酒店职位数据表记录的API接口
  817. 路径参数:
  818. - position_id: 职位记录ID
  819. 返回:
  820. - JSON: 包含删除结果和被删除的职位信息
  821. """
  822. try:
  823. # 调用业务逻辑函数删除职位记录
  824. result = delete_hotel_positions(position_id)
  825. # 根据处理结果设置HTTP状态码
  826. if result['code'] == 200:
  827. status_code = 200 # OK
  828. elif result['code'] == 404:
  829. status_code = 404 # Not Found
  830. else:
  831. status_code = 500 # Internal Server Error
  832. return jsonify(result), status_code
  833. except Exception as e:
  834. # 处理未预期的异常
  835. error_msg = f"删除酒店职位记录时发生错误: {str(e)}"
  836. logger.error(error_msg, exc_info=True)
  837. return jsonify({
  838. 'success': False,
  839. 'message': error_msg,
  840. 'data': None
  841. }), 500
  842. @bp.route('/get-hotel-group-brands-list', methods=['GET'])
  843. def get_hotel_group_brands_list_route():
  844. """
  845. 获取酒店集团子品牌数据表全部记录的API接口
  846. 返回:
  847. - JSON: 包含酒店集团品牌记录列表和处理状态
  848. """
  849. try:
  850. # 调用业务逻辑函数获取酒店集团品牌列表
  851. result = get_hotel_group_brands_list()
  852. # 根据处理结果设置HTTP状态码
  853. status_code = 200 if result['success'] else 500
  854. return jsonify(result), status_code
  855. except Exception as e:
  856. # 处理未预期的异常
  857. error_msg = f"获取酒店集团品牌列表时发生错误: {str(e)}"
  858. logger.error(error_msg, exc_info=True)
  859. return jsonify({
  860. 'success': False,
  861. 'message': error_msg,
  862. 'data': [],
  863. 'count': 0
  864. }), 500
  865. @bp.route('/add-hotel-group-brands', methods=['POST'])
  866. def add_hotel_group_brands_route():
  867. """
  868. 新增酒店集团子品牌数据表记录的API接口
  869. 请求参数:
  870. - JSON格式,包含以下字段:
  871. - group_name_en: 集团英文名称 (必填)
  872. - group_name_zh: 集团中文名称 (必填)
  873. - brand_name_en: 品牌英文名称 (必填)
  874. - brand_name_zh: 品牌中文名称 (必填)
  875. - positioning_level_en: 定位级别英文名称 (必填)
  876. - positioning_level_zh: 定位级别中文名称 (必填)
  877. - created_by: 创建者 (可选)
  878. - updated_by: 更新者 (可选)
  879. - status: 状态 (可选)
  880. 返回:
  881. - JSON: 包含创建结果和品牌信息
  882. """
  883. try:
  884. # 获取请求数据
  885. data = request.get_json()
  886. if not data:
  887. return jsonify({
  888. 'success': False,
  889. 'message': '请求数据为空',
  890. 'data': None
  891. }), 400
  892. # 调用业务逻辑函数处理创建
  893. result = add_hotel_group_brands(data)
  894. # 根据处理结果设置HTTP状态码
  895. if result['code'] == 200:
  896. status_code = 201 # Created
  897. elif result['code'] == 400:
  898. status_code = 400 # Bad Request
  899. elif result['code'] == 409:
  900. status_code = 409 # Conflict
  901. else:
  902. status_code = 500 # Internal Server Error
  903. return jsonify(result), status_code
  904. except Exception as e:
  905. # 处理未预期的异常
  906. error_msg = f"创建酒店集团品牌记录时发生错误: {str(e)}"
  907. logger.error(error_msg, exc_info=True)
  908. return jsonify({
  909. 'success': False,
  910. 'message': error_msg,
  911. 'data': None
  912. }), 500
  913. @bp.route('/update-hotel-group-brands/<int:brand_id>', methods=['PUT'])
  914. def update_hotel_group_brands_route(brand_id):
  915. """
  916. 修改酒店集团子品牌数据表记录的API接口
  917. 路径参数:
  918. - brand_id: 品牌记录ID
  919. 请求参数:
  920. - JSON格式,可能包含以下字段:
  921. - group_name_en: 集团英文名称
  922. - group_name_zh: 集团中文名称
  923. - brand_name_en: 品牌英文名称
  924. - brand_name_zh: 品牌中文名称
  925. - positioning_level_en: 定位级别英文名称
  926. - positioning_level_zh: 定位级别中文名称
  927. - updated_by: 更新者
  928. - status: 状态
  929. 返回:
  930. - JSON: 包含更新结果和品牌信息
  931. """
  932. try:
  933. # 获取请求数据
  934. data = request.get_json()
  935. if not data:
  936. return jsonify({
  937. 'success': False,
  938. 'message': '请求数据为空',
  939. 'data': None
  940. }), 400
  941. # 调用业务逻辑函数处理更新
  942. result = update_hotel_group_brands(brand_id, data)
  943. # 根据处理结果设置HTTP状态码
  944. if result['code'] == 200:
  945. status_code = 200 # OK
  946. elif result['code'] == 400:
  947. status_code = 400 # Bad Request
  948. elif result['code'] == 404:
  949. status_code = 404 # Not Found
  950. elif result['code'] == 409:
  951. status_code = 409 # Conflict
  952. else:
  953. status_code = 500 # Internal Server Error
  954. return jsonify(result), status_code
  955. except Exception as e:
  956. # 处理未预期的异常
  957. error_msg = f"更新酒店集团品牌记录时发生错误: {str(e)}"
  958. logger.error(error_msg, exc_info=True)
  959. return jsonify({
  960. 'success': False,
  961. 'message': error_msg,
  962. 'data': None
  963. }), 500
  964. @bp.route('/query-hotel-group-brands/<int:brand_id>', methods=['GET'])
  965. def query_hotel_group_brands_route(brand_id):
  966. """
  967. 查找指定ID的酒店集团子品牌数据表记录的API接口
  968. 路径参数:
  969. - brand_id: 品牌记录ID
  970. 返回:
  971. - JSON: 包含查找结果和品牌信息
  972. """
  973. try:
  974. # 调用业务逻辑函数查找品牌记录
  975. result = query_hotel_group_brands(brand_id)
  976. # 根据处理结果设置HTTP状态码
  977. if result['code'] == 200:
  978. status_code = 200 # OK
  979. elif result['code'] == 404:
  980. status_code = 404 # Not Found
  981. else:
  982. status_code = 500 # Internal Server Error
  983. return jsonify(result), status_code
  984. except Exception as e:
  985. # 处理未预期的异常
  986. error_msg = f"查找酒店集团品牌记录时发生错误: {str(e)}"
  987. logger.error(error_msg, exc_info=True)
  988. return jsonify({
  989. 'success': False,
  990. 'message': error_msg,
  991. 'data': None
  992. }), 500
  993. @bp.route('/delete-hotel-group-brands/<int:brand_id>', methods=['DELETE'])
  994. def delete_hotel_group_brands_route(brand_id):
  995. """
  996. 删除指定ID的酒店集团子品牌数据表记录的API接口
  997. 路径参数:
  998. - brand_id: 品牌记录ID
  999. 返回:
  1000. - JSON: 包含删除结果和被删除的品牌信息
  1001. """
  1002. try:
  1003. # 调用业务逻辑函数删除品牌记录
  1004. result = delete_hotel_group_brands(brand_id)
  1005. # 根据处理结果设置HTTP状态码
  1006. if result['code'] == 200:
  1007. status_code = 200 # OK
  1008. elif result['code'] == 404:
  1009. status_code = 404 # Not Found
  1010. else:
  1011. status_code = 500 # Internal Server Error
  1012. return jsonify(result), status_code
  1013. except Exception as e:
  1014. # 处理未预期的异常
  1015. error_msg = f"删除酒店集团品牌记录时发生错误: {str(e)}"
  1016. logger.error(error_msg, exc_info=True)
  1017. return jsonify({
  1018. 'success': False,
  1019. 'message': error_msg,
  1020. 'data': None
  1021. }), 500
  1022. # ==================================
  1023. # 重复记录处理API接口
  1024. # ==================================
  1025. @bp.route('/get-duplicate-records', methods=['GET'])
  1026. def get_duplicate_records_route():
  1027. """
  1028. 获取重复记录列表的API接口
  1029. 查询参数:
  1030. - status: 可选,筛选特定状态的记录 ('pending', 'processed', 'ignored')
  1031. 返回:
  1032. - JSON: 包含重复记录列表和处理状态
  1033. """
  1034. try:
  1035. # 获取查询参数
  1036. status = request.args.get('status', None)
  1037. # 验证status参数的有效性
  1038. if status and status not in ['pending', 'processed', 'ignored']:
  1039. return jsonify({
  1040. 'success': False,
  1041. 'message': 'status参数无效,必须为 pending、processed 或 ignored',
  1042. 'data': None
  1043. }), 400
  1044. # 调用业务逻辑函数获取重复记录列表
  1045. result = get_duplicate_records(status)
  1046. # 根据处理结果设置HTTP状态码
  1047. status_code = 200 if result['success'] else 500
  1048. return jsonify(result), status_code
  1049. except Exception as e:
  1050. # 处理未预期的异常
  1051. error_msg = f"获取重复记录列表时发生错误: {str(e)}"
  1052. logger.error(error_msg, exc_info=True)
  1053. return jsonify({
  1054. 'success': False,
  1055. 'message': error_msg,
  1056. 'data': [],
  1057. 'count': 0
  1058. }), 500
  1059. @bp.route('/process-duplicate-record/<int:duplicate_id>', methods=['POST'])
  1060. def process_duplicate_record_route(duplicate_id):
  1061. """
  1062. 处理重复记录的API接口
  1063. 路径参数:
  1064. - duplicate_id: 重复记录ID
  1065. 请求参数:
  1066. - JSON格式,包含以下字段:
  1067. - action: 处理动作 (必填) ('merge_to_suspected', 'keep_main', 'ignore')
  1068. - selected_duplicate_id: 当action为'merge_to_suspected'时,选择的疑似重复记录ID (可选)
  1069. - processed_by: 处理人 (可选)
  1070. - notes: 处理备注 (可选)
  1071. 返回:
  1072. - JSON: 包含处理结果和状态信息
  1073. """
  1074. try:
  1075. # 获取请求数据
  1076. data = request.get_json()
  1077. if not data:
  1078. return jsonify({
  1079. 'success': False,
  1080. 'message': '请求数据为空',
  1081. 'data': None
  1082. }), 400
  1083. # 验证必填字段
  1084. action = data.get('action')
  1085. if not action:
  1086. return jsonify({
  1087. 'success': False,
  1088. 'message': '缺少必填字段: action',
  1089. 'data': None
  1090. }), 400
  1091. # 验证action参数的有效性
  1092. if action not in ['merge_to_suspected', 'keep_main', 'ignore']:
  1093. return jsonify({
  1094. 'success': False,
  1095. 'message': 'action参数无效,必须为 merge_to_suspected、keep_main 或 ignore',
  1096. 'data': None
  1097. }), 400
  1098. # 提取其他参数
  1099. selected_duplicate_id = data.get('selected_duplicate_id')
  1100. processed_by = data.get('processed_by')
  1101. notes = data.get('notes')
  1102. # 特殊验证:如果action为merge_to_suspected,必须提供selected_duplicate_id
  1103. if action == 'merge_to_suspected' and not selected_duplicate_id:
  1104. return jsonify({
  1105. 'success': False,
  1106. 'message': '执行merge_to_suspected操作时必须提供selected_duplicate_id',
  1107. 'data': None
  1108. }), 400
  1109. # 调用业务逻辑函数处理重复记录
  1110. result = process_duplicate_record(
  1111. duplicate_id=duplicate_id,
  1112. action=action,
  1113. selected_duplicate_id=selected_duplicate_id,
  1114. processed_by=processed_by,
  1115. notes=notes
  1116. )
  1117. # 根据处理结果设置HTTP状态码
  1118. if result['code'] == 200:
  1119. status_code = 200 # OK
  1120. elif result['code'] == 400:
  1121. status_code = 400 # Bad Request
  1122. elif result['code'] == 404:
  1123. status_code = 404 # Not Found
  1124. else:
  1125. status_code = 500 # Internal Server Error
  1126. return jsonify(result), status_code
  1127. except Exception as e:
  1128. # 处理未预期的异常
  1129. error_msg = f"处理重复记录时发生错误: {str(e)}"
  1130. logger.error(error_msg, exc_info=True)
  1131. return jsonify({
  1132. 'success': False,
  1133. 'message': error_msg,
  1134. 'data': None
  1135. }), 500
  1136. @bp.route('/get-duplicate-record-detail/<int:duplicate_id>', methods=['GET'])
  1137. def get_duplicate_record_detail_route(duplicate_id):
  1138. """
  1139. 获取指定重复记录详细信息的API接口
  1140. 路径参数:
  1141. - duplicate_id: 重复记录ID
  1142. 返回:
  1143. - JSON: 包含重复记录详细信息
  1144. """
  1145. try:
  1146. # 调用业务逻辑函数获取重复记录详情
  1147. result = get_duplicate_record_detail(duplicate_id)
  1148. # 根据处理结果设置HTTP状态码
  1149. if result['code'] == 200:
  1150. status_code = 200 # OK
  1151. elif result['code'] == 404:
  1152. status_code = 404 # Not Found
  1153. else:
  1154. status_code = 500 # Internal Server Error
  1155. return jsonify(result), status_code
  1156. except Exception as e:
  1157. # 处理未预期的异常
  1158. error_msg = f"获取重复记录详情时发生错误: {str(e)}"
  1159. logger.error(error_msg, exc_info=True)
  1160. return jsonify({
  1161. 'success': False,
  1162. 'message': error_msg,
  1163. 'data': None
  1164. }), 500
  1165. # 删除名片记录接口
  1166. @bp.route('/delete-business-card/<int:card_id>', methods=['DELETE'])
  1167. def delete_business_card_route(card_id):
  1168. """
  1169. 删除名片记录的API接口
  1170. 路径参数:
  1171. - card_id: 名片记录ID (必填)
  1172. 功能说明:
  1173. - 删除PostgreSQL数据库中business_cards表的指定记录
  1174. - 删除PostgreSQL数据库中duplicate_business_cards表的相关记录
  1175. - 删除MinIO存储中的名片图片文件
  1176. - 删除Neo4j图数据库中talent节点及其关联关系
  1177. 返回:
  1178. - JSON: 包含删除操作的结果状态和被删除的记录信息
  1179. 状态码:
  1180. - 200: 完全成功删除所有相关数据
  1181. - 206: 部分成功 (PostgreSQL删除成功,但Neo4j删除失败)
  1182. - 404: 未找到指定ID的名片记录
  1183. - 500: 删除操作失败
  1184. """
  1185. try:
  1186. # 验证card_id参数
  1187. if not card_id or card_id <= 0:
  1188. return jsonify({
  1189. 'success': False,
  1190. 'message': '无效的名片记录ID',
  1191. 'data': None
  1192. }), 400
  1193. # 调用删除函数
  1194. result = delete_business_card(card_id)
  1195. # 根据处理结果设置HTTP状态码和返回响应
  1196. if result['success']:
  1197. if result['code'] == 200:
  1198. status_code = 200 # 完全成功
  1199. elif result['code'] == 206:
  1200. status_code = 206 # 部分成功
  1201. else:
  1202. status_code = 200 # 默认成功
  1203. else:
  1204. if result['code'] == 404:
  1205. status_code = 404 # 未找到记录
  1206. elif result['code'] == 400:
  1207. status_code = 400 # 参数错误
  1208. else:
  1209. status_code = 500 # 服务器错误
  1210. return jsonify(result), status_code
  1211. except Exception as e:
  1212. logger.error(f"删除名片记录失败: {str(e)}")
  1213. return jsonify({
  1214. 'success': False,
  1215. 'message': f'删除名片记录失败: {str(e)}',
  1216. 'data': None
  1217. }), 500
  1218. # 修复损坏的重复记录接口
  1219. @bp.route('/fix-broken-duplicate-records', methods=['POST'])
  1220. def fix_broken_duplicate_records_route():
  1221. """
  1222. 修复duplicate_business_cards表中main_card_id为null的损坏记录
  1223. 功能说明:
  1224. - 查找所有main_card_id为null的损坏记录
  1225. - 删除这些损坏的记录以维护数据完整性
  1226. - 返回修复操作的详细结果
  1227. 返回:
  1228. - JSON: 包含修复操作的结果和被删除记录的信息
  1229. 状态码:
  1230. - 200: 修复成功
  1231. - 500: 修复失败
  1232. 注意:
  1233. - 此操作会永久删除损坏的记录
  1234. - 建议在系统维护时执行此操作
  1235. """
  1236. try:
  1237. # 调用修复函数
  1238. result = fix_broken_duplicate_records()
  1239. # 根据结果设置状态码
  1240. if result['success']:
  1241. status_code = 200
  1242. else:
  1243. status_code = 500
  1244. return jsonify(result), status_code
  1245. except Exception as e:
  1246. logger.error(f"修复损坏记录接口调用失败: {str(e)}")
  1247. return jsonify({
  1248. 'success': False,
  1249. 'message': f'修复损坏记录接口调用失败: {str(e)}',
  1250. 'data': None
  1251. }), 500
  1252. # 网页文本解析接口
  1253. @bp.route('/webpage-parse', methods=['POST'])
  1254. def webpage_parse_route():
  1255. """
  1256. 解析网页 Markdown 文本并提取人员信息的API接口
  1257. 请求参数:
  1258. - markdown_text: 网页的 Markdown 格式文本内容 (JSON格式)
  1259. - publish_time: 发布时间,用于career_path中的date字段 (JSON格式)
  1260. 请求体示例:
  1261. {
  1262. "markdown_text": "# 张三\n\n职位:高级经理\n\n公司:XX酒店\n\n![照片](http://example.com/photo.jpg)",
  1263. "publish_time": "2025-01-15"
  1264. }
  1265. 返回:
  1266. - JSON: 包含提取的人员信息和处理状态
  1267. 功能说明:
  1268. - 接收 Markdown 格式的网页文本
  1269. - 进行必要的格式和内容验证
  1270. - 使用 Qwen VL Max 模型提取人员信息
  1271. - 支持提取照片链接 (pic_url)
  1272. - 返回标准化的人员信息数据
  1273. 状态码:
  1274. - 200: 解析成功
  1275. - 400: 请求参数错误
  1276. - 500: 解析失败
  1277. """
  1278. try:
  1279. # 检查请求是否为 JSON 格式
  1280. if not request.is_json:
  1281. return jsonify({
  1282. 'success': False,
  1283. 'message': '请求必须是 JSON 格式',
  1284. 'data': None
  1285. }), 400
  1286. # 获取请求数据
  1287. data = request.get_json()
  1288. # 检查是否提供了 markdown_text 参数
  1289. if 'markdown_text' not in data:
  1290. return jsonify({
  1291. 'success': False,
  1292. 'message': '缺少必填参数: markdown_text',
  1293. 'data': None
  1294. }), 400
  1295. # 检查是否提供了 publish_time 参数
  1296. if 'publish_time' not in data:
  1297. return jsonify({
  1298. 'success': False,
  1299. 'message': '缺少必填参数: publish_time',
  1300. 'data': None
  1301. }), 400
  1302. markdown_text = data['markdown_text']
  1303. publish_time = data['publish_time']
  1304. # 验证 markdown_text 是否为字符串
  1305. if not isinstance(markdown_text, str):
  1306. return jsonify({
  1307. 'success': False,
  1308. 'message': 'markdown_text 必须是字符串类型',
  1309. 'data': None
  1310. }), 400
  1311. # 验证 publish_time 是否为字符串
  1312. if not isinstance(publish_time, str):
  1313. return jsonify({
  1314. 'success': False,
  1315. 'message': 'publish_time 必须是字符串类型',
  1316. 'data': None
  1317. }), 400
  1318. # 验证文本内容不能为空
  1319. if not markdown_text.strip():
  1320. return jsonify({
  1321. 'success': False,
  1322. 'message': 'markdown_text 内容不能为空',
  1323. 'data': None
  1324. }), 400
  1325. # 验证文本长度(防止过长的文本)
  1326. if len(markdown_text) > 50000: # 限制最大50KB
  1327. return jsonify({
  1328. 'success': False,
  1329. 'message': 'markdown_text 内容过长,最大支持50KB',
  1330. 'data': None
  1331. }), 400
  1332. # 基本的 Markdown 格式验证(可选)
  1333. # 检查是否包含一些基本的文本内容
  1334. if len(markdown_text.strip()) < 10:
  1335. return jsonify({
  1336. 'success': False,
  1337. 'message': 'markdown_text 内容过短,无法进行有效解析',
  1338. 'data': None
  1339. }), 400
  1340. # 记录解析请求
  1341. logger.info(f"开始解析网页文本,内容长度: {len(markdown_text)} 字符")
  1342. # 调用网页文本解析函数
  1343. extracted_data = process_webpage_with_QWen(markdown_text, publish_time)
  1344. # 返回成功结果
  1345. return jsonify({
  1346. 'success': True,
  1347. 'message': '网页文本解析成功',
  1348. 'data': extracted_data
  1349. }), 200
  1350. except Exception as e:
  1351. # 记录错误日志
  1352. error_msg = f"网页文本解析失败: {str(e)}"
  1353. logger.error(error_msg, exc_info=True)
  1354. # 返回错误响应
  1355. return jsonify({
  1356. 'success': False,
  1357. 'message': error_msg,
  1358. 'data': None
  1359. }), 500
  1360. # 添加网页人才信息接口
  1361. @bp.route('/add-webpage-talent', methods=['POST'])
  1362. def add_webpage_talent_route():
  1363. """
  1364. 添加网页人才信息的API接口,包括保存网页内容和创建名片记录
  1365. 请求参数:
  1366. - talent_list: 人才信息列表,每个item包含业务卡片格式的数据 (JSON数组)
  1367. - web_md: 网页markdown文本内容 (JSON字符串)
  1368. 请求体示例:
  1369. {
  1370. "talent_list": [
  1371. {
  1372. "name_zh": "张三",
  1373. "name_en": "Zhang San",
  1374. "title_zh": "总经理",
  1375. "title_en": "General Manager",
  1376. "hotel_zh": "北京万豪酒店",
  1377. "hotel_en": "Beijing Marriott Hotel",
  1378. "brand_group": "万豪",
  1379. "mobile": "13800138000",
  1380. "email": "zhangsan@example.com"
  1381. },
  1382. {
  1383. "name_zh": "李四",
  1384. "name_en": "Li Si",
  1385. "title_zh": "市场总监",
  1386. "title_en": "Marketing Director",
  1387. "hotel_zh": "上海希尔顿酒店",
  1388. "hotel_en": "Shanghai Hilton Hotel",
  1389. "brand_group": "希尔顿",
  1390. "mobile": "13900139000",
  1391. "email": "lisi@example.com"
  1392. }
  1393. ],
  1394. "web_md": "# 人事任命公告\n\n**1** 张三被任命为北京万豪酒店总经理...\n\n**2** 李四被任命为上海希尔顿酒店市场总监..."
  1395. }
  1396. 返回:
  1397. - JSON: 包含处理结果,包括成功和失败的记录统计
  1398. 功能说明:
  1399. - 将网页markdown内容保存到MinIO
  1400. - 循环处理talent_list中的每个人才记录
  1401. - 为每个人才创建business_card记录
  1402. - 使用与名片上传相同的重复检查逻辑
  1403. - 在business_card记录的updated_by字段中记录MinIO路径
  1404. 状态码:
  1405. - 200: 所有记录处理成功
  1406. - 206: 部分记录处理成功
  1407. - 400: 请求参数错误
  1408. - 500: 处理失败
  1409. """
  1410. try:
  1411. # 检查请求是否为 JSON 格式
  1412. if not request.is_json:
  1413. return jsonify({
  1414. 'success': False,
  1415. 'message': '请求必须是 JSON 格式',
  1416. 'data': None
  1417. }), 400
  1418. # 获取请求数据
  1419. data = request.get_json()
  1420. # 检查是否提供了 talent_list 参数
  1421. if 'talent_list' not in data:
  1422. return jsonify({
  1423. 'success': False,
  1424. 'message': '缺少必填参数: talent_list',
  1425. 'data': None
  1426. }), 400
  1427. # 检查是否提供了 web_md 参数
  1428. if 'web_md' not in data:
  1429. return jsonify({
  1430. 'success': False,
  1431. 'message': '缺少必填参数: web_md',
  1432. 'data': None
  1433. }), 400
  1434. talent_list = data['talent_list']
  1435. web_md = data['web_md']
  1436. # 验证 talent_list 是否为数组
  1437. if not isinstance(talent_list, list):
  1438. return jsonify({
  1439. 'success': False,
  1440. 'message': 'talent_list 必须是数组类型',
  1441. 'data': None
  1442. }), 400
  1443. # 验证 web_md 是否为字符串
  1444. if not isinstance(web_md, str):
  1445. return jsonify({
  1446. 'success': False,
  1447. 'message': 'web_md 必须是字符串类型',
  1448. 'data': None
  1449. }), 400
  1450. # 验证数组不能为空
  1451. if len(talent_list) == 0:
  1452. return jsonify({
  1453. 'success': False,
  1454. 'message': 'talent_list 不能为空数组',
  1455. 'data': None
  1456. }), 400
  1457. # 验证文本内容不能为空
  1458. if not web_md.strip():
  1459. return jsonify({
  1460. 'success': False,
  1461. 'message': 'web_md 内容不能为空',
  1462. 'data': None
  1463. }), 400
  1464. # 验证文本长度(防止过长的文本)
  1465. if len(web_md) > 100000: # 限制最大100KB
  1466. return jsonify({
  1467. 'success': False,
  1468. 'message': 'web_md 内容过长,最大支持100KB',
  1469. 'data': None
  1470. }), 400
  1471. # 验证数组长度(防止过多记录)
  1472. if len(talent_list) > 50: # 限制最大50条记录
  1473. return jsonify({
  1474. 'success': False,
  1475. 'message': 'talent_list 记录过多,最大支持50条记录',
  1476. 'data': None
  1477. }), 400
  1478. # 基本的数据格式验证
  1479. for index, talent_data in enumerate(talent_list):
  1480. if not isinstance(talent_data, dict):
  1481. return jsonify({
  1482. 'success': False,
  1483. 'message': f'talent_list 第{index + 1}项必须是对象类型',
  1484. 'data': None
  1485. }), 400
  1486. # 检查必要的字段
  1487. if not talent_data.get('name_zh'):
  1488. return jsonify({
  1489. 'success': False,
  1490. 'message': f'talent_list 第{index + 1}项缺少必填字段: name_zh',
  1491. 'data': None
  1492. }), 400
  1493. # 记录处理请求
  1494. logger.info(f"开始处理网页人才信息,人才数量: {len(talent_list)}, 网页内容长度: {len(web_md)} 字符")
  1495. # 调用网页人才处理函数
  1496. result = add_webpage_talent(talent_list, web_md)
  1497. # 根据处理结果设置HTTP状态码
  1498. if result['success']:
  1499. if result['code'] == 200:
  1500. status_code = 200 # 全部成功
  1501. elif result['code'] == 206:
  1502. status_code = 206 # 部分成功
  1503. else:
  1504. status_code = 200 # 默认成功
  1505. else:
  1506. if result['code'] == 400:
  1507. status_code = 400 # 参数错误
  1508. else:
  1509. status_code = 500 # 服务器错误
  1510. return jsonify(result), status_code
  1511. except Exception as e:
  1512. # 记录错误日志
  1513. error_msg = f"添加网页人才信息失败: {str(e)}"
  1514. logger.error(error_msg, exc_info=True)
  1515. # 返回错误响应
  1516. return jsonify({
  1517. 'success': False,
  1518. 'message': error_msg,
  1519. 'data': None
  1520. }), 500
  1521. # 获取解析任务列表接口
  1522. @bp.route('/get-parse-tasks', methods=['GET'])
  1523. def get_parse_tasks_route():
  1524. """
  1525. 获取解析任务列表的API接口,支持分页
  1526. 查询参数:
  1527. - page: 页码,从1开始,默认为1
  1528. - per_page: 每页记录数,默认为10,最大100
  1529. - task_type: 任务类型过滤,可选
  1530. - task_status: 任务状态过滤,可选
  1531. 返回:
  1532. - JSON: 包含解析任务列表和分页信息
  1533. 功能说明:
  1534. - 支持分页查询,每页默认10条记录
  1535. - 支持按任务类型和状态过滤
  1536. - 按创建时间倒序排列
  1537. - 返回总记录数和分页信息
  1538. 状态码:
  1539. - 200: 查询成功
  1540. - 400: 请求参数错误
  1541. - 500: 查询失败
  1542. """
  1543. try:
  1544. # 获取查询参数
  1545. page = request.args.get('page', 1, type=int)
  1546. per_page = request.args.get('per_page', 10, type=int)
  1547. task_type = request.args.get('task_type', type=str)
  1548. task_status = request.args.get('task_status', type=str)
  1549. # 记录请求日志
  1550. logger.info(f"获取解析任务列表请求: page={page}, per_page={per_page}, task_type={task_type}, task_status={task_status}")
  1551. # 调用核心业务逻辑
  1552. result = get_parse_tasks(page, per_page, task_type, task_status)
  1553. # 返回结果
  1554. return jsonify({
  1555. 'success': result['success'],
  1556. 'message': result['message'],
  1557. 'data': result['data']
  1558. }), result['code']
  1559. except Exception as e:
  1560. # 记录错误日志
  1561. error_msg = f"获取解析任务列表接口失败: {str(e)}"
  1562. logger.error(error_msg, exc_info=True)
  1563. # 返回错误响应
  1564. return jsonify({
  1565. 'success': False,
  1566. 'message': error_msg,
  1567. 'data': None
  1568. }), 500
  1569. # 获取解析任务详情接口
  1570. @bp.route('/get-parse-task-detail', methods=['GET'])
  1571. def get_parse_task_detail_route():
  1572. """
  1573. 获取解析任务详情的API接口
  1574. 查询参数:
  1575. - task_name: 任务名称,必填
  1576. 返回:
  1577. - JSON: 包含任务详细信息
  1578. 功能说明:
  1579. - 根据任务名称查询指定任务的详细信息
  1580. - 返回任务的所有字段信息
  1581. - 包含解析结果的完整数据
  1582. 状态码:
  1583. - 200: 查询成功
  1584. - 400: 请求参数错误
  1585. - 404: 任务不存在
  1586. - 500: 查询失败
  1587. """
  1588. try:
  1589. # 获取查询参数
  1590. task_name = request.args.get('task_name', type=str)
  1591. # 参数验证
  1592. if not task_name:
  1593. return jsonify({
  1594. 'success': False,
  1595. 'message': '任务名称参数不能为空',
  1596. 'data': None
  1597. }), 400
  1598. # 记录请求日志
  1599. logger.info(f"获取解析任务详情请求: task_name={task_name}")
  1600. # 调用核心业务逻辑
  1601. result = get_parse_task_detail(task_name)
  1602. # 返回结果
  1603. return jsonify({
  1604. 'success': result['success'],
  1605. 'message': result['message'],
  1606. 'data': result['data']
  1607. }), result['code']
  1608. except Exception as e:
  1609. # 记录错误日志
  1610. error_msg = f"获取解析任务详情接口失败: {str(e)}"
  1611. logger.error(error_msg, exc_info=True)
  1612. # 返回错误响应
  1613. return jsonify({
  1614. 'success': False,
  1615. 'message': error_msg,
  1616. 'data': None
  1617. }), 500