api_db.py 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286
  1. import os
  2. import sqlite3
  3. from flask import Flask, jsonify, request
  4. from flask import g
  5. from flask_cors import CORS
  6. # 定义应用的状态码文档
  7. """
  8. 状态码 200 成功
  9. 状态码 400 失败
  10. """
  11. app = Flask(__name__)
  12. # 设置数据库文件和上传文件夹的路径
  13. DATABASE = 'SoilAcidification.db'
  14. UPLOAD_FOLDER = 'uploads'
  15. app.config['UPLOAD_FOLDER'] = UPLOAD_FOLDER
  16. # 确保上传文件夹存在,如果不存在则创建
  17. os.makedirs(app.config['UPLOAD_FOLDER'], exist_ok=True)
  18. # 跨源资源共享(CORS)配置,允许跨域请求
  19. CORS(app)
  20. # 创建一个数据库连接池
  21. def get_db():
  22. db = getattr(g, '_database', None)
  23. if db is None:
  24. db = g._database = sqlite3.connect(DATABASE)
  25. return db
  26. # 请求后钩子,为响应添加内容类型头
  27. @app.after_request
  28. def add_headers(response):
  29. response.headers['Content-Type'] = 'application/json; charset=utf-8'
  30. return response
  31. @app.route('/')
  32. def index():
  33. return 'Hello, World!'
  34. # 定义添加数据库记录的 API 接口
  35. @app.route('/add_item', methods=['POST'])
  36. def add_item():
  37. """
  38. 接收 JSON 格式的请求体,包含表名和要插入的数据。
  39. 尝试将数据插入到指定的表中。
  40. :return:
  41. """
  42. db = get_db()
  43. try:
  44. # 确保请求体是JSON格式
  45. data = request.get_json()
  46. if not data:
  47. raise ValueError("No JSON data provided")
  48. table_name = data.get('table')
  49. item_data = data.get('item')
  50. if not table_name or not item_data:
  51. return jsonify({'error': 'Missing table name or item data'}), 400
  52. cur = db.cursor()
  53. # 动态构建 SQL 语句
  54. columns = ', '.join(item_data.keys())
  55. placeholders = ', '.join(['?'] * len(item_data))
  56. sql = f"INSERT INTO {table_name} ({columns}) VALUES ({placeholders})"
  57. cur.execute(sql, tuple(item_data.values()))
  58. db.commit()
  59. # 返回更详细的成功响应
  60. return jsonify({'success': True, 'message': 'Item added successfully'}), 201
  61. except ValueError as e:
  62. return jsonify({'error': str(e)}), 400
  63. except KeyError as e:
  64. return jsonify({'error': f'Missing data field: {e}'}), 400
  65. except sqlite3.IntegrityError as e:
  66. # 处理例如唯一性约束违反等数据库完整性错误
  67. return jsonify({'error': 'Database integrity error', 'details': str(e)}), 409
  68. except sqlite3.Error as e:
  69. # 处理其他数据库错误
  70. return jsonify({'error': 'Database error', 'details': str(e)}), 500
  71. finally:
  72. db.close()
  73. # 定义删除数据库记录的 API 接口
  74. @app.route('/delete_item', methods=['DELETE'])
  75. def delete_item():
  76. data = request.get_json()
  77. table_name = data.get('table')
  78. condition = data.get('condition')
  79. # 检查表名和条件是否提供
  80. if not table_name or not condition:
  81. return jsonify({
  82. "success": False,
  83. "message": "缺少表名或条件参数"
  84. }), 400
  85. # 尝试从条件字符串中分离键和值
  86. try:
  87. key, value = condition.split('=')
  88. except ValueError:
  89. return jsonify({
  90. "success": False,
  91. "message": "条件格式错误,应为 'key=value'"
  92. }), 400
  93. db = get_db()
  94. cur = db.cursor()
  95. try:
  96. # 执行删除操作
  97. cur.execute(f"DELETE FROM {table_name} WHERE {key} = ?", (value,))
  98. db.commit()
  99. # 如果没有错误发生,返回成功响应
  100. return jsonify({
  101. "success": True,
  102. "message": "记录删除成功"
  103. }), 200
  104. except sqlite3.Error as e:
  105. # 发生错误,回滚事务
  106. db.rollback()
  107. # 返回失败响应,并包含错误信息
  108. return jsonify({
  109. "success": False,
  110. "message": f"删除失败: {e}"
  111. }), 400
  112. # 定义修改数据库记录的 API 接口
  113. @app.route('/update_item', methods=['PUT'])
  114. def update_record():
  115. data = request.get_json()
  116. # 检查必要的数据是否提供
  117. if not data or 'table' not in data or 'item' not in data:
  118. return jsonify({
  119. "success": False,
  120. "message": "请求数据不完整"
  121. }), 400
  122. table_name = data['table']
  123. item = data['item']
  124. # 假设 item 的第一个元素是 ID
  125. if not item or next(iter(item.keys())) is None:
  126. return jsonify({
  127. "success": False,
  128. "message": "记录数据为空"
  129. }), 400
  130. # 获取 ID 和其他字段值
  131. id_key = next(iter(item.keys()))
  132. record_id = item[id_key]
  133. updates = {key: value for key, value in item.items() if key != id_key} # 排除 ID
  134. db = get_db()
  135. cur = db.cursor()
  136. try:
  137. record_id = int(record_id) # 确保 ID 是整数
  138. except ValueError:
  139. return jsonify({
  140. "success": False,
  141. "message": "ID 必须是整数"
  142. }), 400
  143. # 准备参数列表,包括更新的值和 ID
  144. parameters = list(updates.values()) + [record_id]
  145. # 执行更新操作
  146. set_clause = ','.join([f"{k} = ?" for k in updates.keys()])
  147. sql = f"UPDATE {table_name} SET {set_clause} WHERE {id_key} = ?"
  148. try:
  149. cur.execute(sql, parameters)
  150. db.commit()
  151. if cur.rowcount == 0:
  152. return jsonify({
  153. "success": False,
  154. "message": "未找到要更新的记录"
  155. }), 404
  156. return jsonify({
  157. "success": True,
  158. "message": "数据更新成功"
  159. }), 200
  160. except sqlite3.Error as e:
  161. db.rollback()
  162. return jsonify({
  163. "success": False,
  164. "message": f"更新失败: {e}"
  165. }), 400
  166. # 定义查询数据库记录的 API 接口
  167. @app.route('/search/record', methods=['GET'])
  168. def sql_search():
  169. """
  170. 接收 JSON 格式的请求体,包含表名和要查询的 ID。
  171. 尝试查询指定 ID 的记录并返回结果。
  172. :return:
  173. """
  174. try:
  175. data = request.get_json()
  176. # 表名
  177. sql_table = data['table']
  178. # 要搜索的 ID
  179. Id = data['id']
  180. # 连接到数据库
  181. db = get_db()
  182. cur = db.cursor()
  183. # 构造查询语句
  184. sql = f"SELECT * FROM {sql_table} WHERE id = ?"
  185. # 执行查询
  186. cur.execute(sql, (Id,))
  187. # 获取查询结果
  188. rows = cur.fetchall()
  189. column_names = [desc[0] for desc in cur.description]
  190. # 检查是否有结果
  191. if not rows:
  192. return jsonify({'error': '未查找到对应数据。'}), 400
  193. # 构造响应数据
  194. results = []
  195. for row in rows:
  196. result = {column_names[i]: row[i] for i in range(len(row))}
  197. results.append(result)
  198. # 关闭游标和数据库连接
  199. cur.close()
  200. db.close()
  201. # 返回 JSON 响应
  202. return jsonify(results), 200
  203. except sqlite3.Error as e:
  204. # 如果发生数据库错误,返回错误信息
  205. return jsonify({'error': str(e)}), 400
  206. except KeyError as e:
  207. # 如果请求数据中缺少必要的键,返回错误信息
  208. return jsonify({'error': f'缺少必要的数据字段: {e}'}), 400
  209. # 定义提供数据库列表,用于展示表格的 API 接口
  210. @app.route('/tables', methods=['POST'])
  211. def get_table():
  212. data = request.get_json()
  213. table_name = data.get('table')
  214. if not table_name:
  215. return jsonify({'error': '需要表名'}), 400
  216. db = get_db()
  217. try:
  218. cur = db.cursor()
  219. cur.execute(f"SELECT * FROM {table_name}")
  220. rows = cur.fetchall()
  221. if not rows:
  222. return jsonify({'error': '表为空或不存在'}), 400
  223. headers = [description[0] for description in cur.description]
  224. return jsonify(rows=rows, headers=headers), 200
  225. except sqlite3.Error as e:
  226. return jsonify({'error': str(e)}), 400
  227. finally:
  228. db.close()
  229. if __name__ == '__main__':
  230. app.run(debug=True)