routes.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357
  1. import sqlite3
  2. from flask import Blueprint, request, jsonify, g, current_app
  3. from .model import predict
  4. import pandas as pd
  5. from flask_sqlalchemy import SQLAlchemy
  6. from . import db # 从 app 包导入 db 实例
  7. from sqlalchemy.engine.reflection import Inspector
  8. from .database_models import Model, ModelParameters
  9. # 创建蓝图 (Blueprint),用于分离路由
  10. bp = Blueprint('routes', __name__)
  11. @bp.route('/tables', methods=['GET'])
  12. def list_tables():
  13. engine = db.engine # 使用 db 实例的 engine
  14. inspector = Inspector.from_engine(engine) # 创建 Inspector 对象
  15. table_names = inspector.get_table_names() # 获取所有表名
  16. return jsonify(table_names) # 以 JSON 形式返回表名列表
  17. @bp.route('/models/<int:model_id>', methods=['GET'])
  18. def get_model(model_id):
  19. try:
  20. model = Model.query.filter_by(ModelID=model_id).first()
  21. if model:
  22. return jsonify({
  23. 'ModelID': model.ModelID,
  24. 'ModelName': model.ModelName,
  25. 'ModelType': model.ModelType,
  26. 'CreatedAt': model.CreatedAt.strftime('%Y-%m-%d %H:%M:%S'),
  27. 'Description': model.Description
  28. })
  29. else:
  30. return jsonify({'message': 'Model not found'}), 404
  31. except Exception as e:
  32. return jsonify({'error': 'Internal server error', 'message': str(e)}), 500
  33. @bp.route('/models', methods=['GET'])
  34. def get_all_models():
  35. try:
  36. models = Model.query.all() # 获取所有模型数据
  37. if models:
  38. result = [
  39. {
  40. 'ModelID': model.ModelID,
  41. 'ModelName': model.ModelName,
  42. 'ModelType': model.ModelType,
  43. 'CreatedAt': model.CreatedAt.strftime('%Y-%m-%d %H:%M:%S'),
  44. 'Description': model.Description
  45. }
  46. for model in models
  47. ]
  48. return jsonify(result)
  49. else:
  50. return jsonify({'message': 'No models found'}), 404
  51. except Exception as e:
  52. return jsonify({'error': 'Internal server error', 'message': str(e)}), 500
  53. @bp.route('/model-parameters', methods=['GET'])
  54. def get_all_model_parameters():
  55. try:
  56. parameters = ModelParameters.query.all() # 获取所有参数数据
  57. if parameters:
  58. result = [
  59. {
  60. 'ParamID': param.ParamID,
  61. 'ModelID': param.ModelID,
  62. 'ParamName': param.ParamName,
  63. 'ParamValue': param.ParamValue
  64. }
  65. for param in parameters
  66. ]
  67. return jsonify(result)
  68. else:
  69. return jsonify({'message': 'No parameters found'}), 404
  70. except Exception as e:
  71. return jsonify({'error': 'Internal server error', 'message': str(e)}), 500
  72. @bp.route('/models/<int:model_id>/parameters', methods=['GET'])
  73. def get_model_parameters(model_id):
  74. try:
  75. model = Model.query.filter_by(ModelID=model_id).first()
  76. if model:
  77. # 获取该模型的所有参数
  78. parameters = [
  79. {
  80. 'ParamID': param.ParamID,
  81. 'ParamName': param.ParamName,
  82. 'ParamValue': param.ParamValue
  83. }
  84. for param in model.parameters
  85. ]
  86. # 返回模型参数信息
  87. return jsonify({
  88. 'ModelID': model.ModelID,
  89. 'ModelName': model.ModelName,
  90. 'ModelType': model.ModelType,
  91. 'CreatedAt': model.CreatedAt.strftime('%Y-%m-%d %H:%M:%S'),
  92. 'Description': model.Description,
  93. 'Parameters': parameters
  94. })
  95. else:
  96. return jsonify({'message': 'Model not found'}), 404
  97. except Exception as e:
  98. return jsonify({'error': 'Internal server error', 'message': str(e)}), 500
  99. @bp.route('/predict', methods=['POST'])
  100. def predict_route():
  101. try:
  102. data = request.get_json()
  103. model_name = data.get('model_name', 'RF_filt') # 提取模型名称
  104. parameters = data.get('parameters', {}) # 提取所有参数
  105. input_data = pd.DataFrame([parameters]) # 转换参数为DataFrame
  106. predictions = predict(input_data, model_name) # 调用预测函数
  107. return jsonify({'predictions': predictions}), 200
  108. except Exception as e:
  109. return jsonify({'error': str(e)}), 400
  110. # 定义添加数据库记录的 API 接口
  111. @bp.route('/add_item', methods=['POST'])
  112. def add_item():
  113. """
  114. 接收 JSON 格式的请求体,包含表名和要插入的数据。
  115. 尝试将数据插入到指定的表中。
  116. :return:
  117. """
  118. db = get_db()
  119. try:
  120. # 确保请求体是JSON格式
  121. data = request.get_json()
  122. if not data:
  123. raise ValueError("No JSON data provided")
  124. table_name = data.get('table')
  125. item_data = data.get('item')
  126. if not table_name or not item_data:
  127. return jsonify({'error': 'Missing table name or item data'}), 400
  128. cur = db.cursor()
  129. # 动态构建 SQL 语句
  130. columns = ', '.join(item_data.keys())
  131. placeholders = ', '.join(['?'] * len(item_data))
  132. sql = f"INSERT INTO {table_name} ({columns}) VALUES ({placeholders})"
  133. cur.execute(sql, tuple(item_data.values()))
  134. db.commit()
  135. # 返回更详细的成功响应
  136. return jsonify({'success': True, 'message': 'Item added successfully'}), 201
  137. except ValueError as e:
  138. return jsonify({'error': str(e)}), 400
  139. except KeyError as e:
  140. return jsonify({'error': f'Missing data field: {e}'}), 400
  141. except sqlite3.IntegrityError as e:
  142. # 处理例如唯一性约束违反等数据库完整性错误
  143. return jsonify({'error': 'Database integrity error', 'details': str(e)}), 409
  144. except sqlite3.Error as e:
  145. # 处理其他数据库错误
  146. return jsonify({'error': 'Database error', 'details': str(e)}), 500
  147. finally:
  148. db.close()
  149. # 定义删除数据库记录的 API 接口
  150. @bp.route('/delete_item', methods=['POST'])
  151. def delete_item():
  152. data = request.get_json()
  153. table_name = data.get('table')
  154. condition = data.get('condition')
  155. # 检查表名和条件是否提供
  156. if not table_name or not condition:
  157. return jsonify({
  158. "success": False,
  159. "message": "缺少表名或条件参数"
  160. }), 400
  161. # 尝试从条件字符串中分离键和值
  162. try:
  163. key, value = condition.split('=')
  164. except ValueError:
  165. return jsonify({
  166. "success": False,
  167. "message": "条件格式错误,应为 'key=value'"
  168. }), 400
  169. db = get_db()
  170. cur = db.cursor()
  171. try:
  172. # 执行删除操作
  173. cur.execute(f"DELETE FROM {table_name} WHERE {key} = ?", (value,))
  174. db.commit()
  175. # 如果没有错误发生,返回成功响应
  176. return jsonify({
  177. "success": True,
  178. "message": "记录删除成功"
  179. }), 200
  180. except sqlite3.Error as e:
  181. # 发生错误,回滚事务
  182. db.rollback()
  183. # 返回失败响应,并包含错误信息
  184. return jsonify({
  185. "success": False,
  186. "message": f"删除失败: {e}"
  187. }), 400
  188. # 定义修改数据库记录的 API 接口
  189. @bp.route('/update_item', methods=['PUT'])
  190. def update_record():
  191. data = request.get_json()
  192. # 检查必要的数据是否提供
  193. if not data or 'table' not in data or 'item' not in data:
  194. return jsonify({
  195. "success": False,
  196. "message": "请求数据不完整"
  197. }), 400
  198. table_name = data['table']
  199. item = data['item']
  200. # 假设 item 的第一个元素是 ID
  201. if not item or next(iter(item.keys())) is None:
  202. return jsonify({
  203. "success": False,
  204. "message": "记录数据为空"
  205. }), 400
  206. # 获取 ID 和其他字段值
  207. id_key = next(iter(item.keys()))
  208. record_id = item[id_key]
  209. updates = {key: value for key, value in item.items() if key != id_key} # 排除 ID
  210. db = get_db()
  211. cur = db.cursor()
  212. try:
  213. record_id = int(record_id) # 确保 ID 是整数
  214. except ValueError:
  215. return jsonify({
  216. "success": False,
  217. "message": "ID 必须是整数"
  218. }), 400
  219. # 准备参数列表,包括更新的值和 ID
  220. parameters = list(updates.values()) + [record_id]
  221. # 执行更新操作
  222. set_clause = ','.join([f"{k} = ?" for k in updates.keys()])
  223. sql = f"UPDATE {table_name} SET {set_clause} WHERE {id_key} = ?"
  224. try:
  225. cur.execute(sql, parameters)
  226. db.commit()
  227. if cur.rowcount == 0:
  228. return jsonify({
  229. "success": False,
  230. "message": "未找到要更新的记录"
  231. }), 404
  232. return jsonify({
  233. "success": True,
  234. "message": "数据更新成功"
  235. }), 200
  236. except sqlite3.Error as e:
  237. db.rollback()
  238. return jsonify({
  239. "success": False,
  240. "message": f"更新失败: {e}"
  241. }), 400
  242. # 定义查询数据库记录的 API 接口
  243. @bp.route('/search/record', methods=['GET'])
  244. def sql_search():
  245. """
  246. 接收 JSON 格式的请求体,包含表名和要查询的 ID。
  247. 尝试查询指定 ID 的记录并返回结果。
  248. :return:
  249. """
  250. try:
  251. data = request.get_json()
  252. # 表名
  253. sql_table = data['table']
  254. # 要搜索的 ID
  255. Id = data['id']
  256. # 连接到数据库
  257. db = get_db()
  258. cur = db.cursor()
  259. # 构造查询语句
  260. sql = f"SELECT * FROM {sql_table} WHERE id = ?"
  261. # 执行查询
  262. cur.execute(sql, (Id,))
  263. # 获取查询结果
  264. rows = cur.fetchall()
  265. column_names = [desc[0] for desc in cur.description]
  266. # 检查是否有结果
  267. if not rows:
  268. return jsonify({'error': '未查找到对应数据。'}), 400
  269. # 构造响应数据
  270. results = []
  271. for row in rows:
  272. result = {column_names[i]: row[i] for i in range(len(row))}
  273. results.append(result)
  274. # 关闭游标和数据库连接
  275. cur.close()
  276. db.close()
  277. # 返回 JSON 响应
  278. return jsonify(results), 200
  279. except sqlite3.Error as e:
  280. # 如果发生数据库错误,返回错误信息
  281. return jsonify({'error': str(e)}), 400
  282. except KeyError as e:
  283. # 如果请求数据中缺少必要的键,返回错误信息
  284. return jsonify({'error': f'缺少必要的数据字段: {e}'}), 400
  285. # 定义提供数据库列表,用于展示表格的 API 接口
  286. @bp.route('/tables', methods=['POST'])
  287. def get_table():
  288. data = request.get_json()
  289. table_name = data.get('table')
  290. if not table_name:
  291. return jsonify({'error': '需要表名'}), 400
  292. db = get_db()
  293. try:
  294. cur = db.cursor()
  295. cur.execute(f"SELECT * FROM {table_name}")
  296. rows = cur.fetchall()
  297. if not rows:
  298. return jsonify({'error': '表为空或不存在'}), 400
  299. headers = [description[0] for description in cur.description]
  300. return jsonify(rows=rows, headers=headers), 200
  301. except sqlite3.Error as e:
  302. return jsonify({'error': str(e)}), 400
  303. finally:
  304. db.close()