API伺服器的設計
Table of contents
背景
整體流程
API選擇策略
- GPT、perplexity各大AI都建議使用Flask,不過此處還是補充一下決策的背景。
- node.js:在格隔柵資料處理過程過於繁瑣、不予考慮。
- streamlet
- python家族之一
- 有2個模組可以做伺服器,Gunicorn:
st.run(app, host="0.0.0.0", port=8080)
,獨立伺服器:st.streamlit(app, run_url="streamlit")
- 維運html,則使用
st.markdown()
或st.write()
,列印鑲嵌在程式碼內的html內容。不適用太過複雜的html。
- Flask、Django、Streamlit三者比較如下
框架 | 特性與能力 | 優勢 | 劣勢 |
---|---|---|---|
Flask | 輕量、靈活、微框架 | 易學易用,非常適合中小型應用 | 只有基本功能,還需要額外的庫和工具 |
Django | 進階、全端框架 | 快速開發、健壯、可擴展的架構,非常適合大型、複雜的應用 | 陡峭的學習曲線,靈活性不如 Flask |
Streamlit | 用於建立資料應用程式的開源框架 | 即時資料視覺化,非常適合創建顯示和操作資料的應用程式 | 不是專門為建立API而設計的 |
其他方案也詳列如下
框架 | 特性與能力 | 優勢 | 劣勢 |
---|---|---|---|
FastAPI | 現代、高效能Web框架 | 效能快、記憶體佔用小、基於Python類型提示 | 仍然比較新,不像Flask和Django那樣廣泛使用 |
Falcon | 輕量級高效能Web框架 | 快速、有效率、易於使用,非常適合建立RESTful API | 微框架,需要額外的程式庫和工具 |
Pyramid | 靈活的開源Web框架 | 應用廣泛,基於WSGI標準 | 全端框架,陡峭的學習曲線 |
Tornado | 可擴充、非阻塞的網路框架 | 處理大量並發連接,非常適合高效能API | 微框架,需要額外的函式庫和工具 |
兩個圖檔並存服務的考量
matplotlib.pyplot.contour
是產生dxf圖檔必經(最有效率)的過程,輸出檔案只是附加產品- 原本序列的關係,因地形數據處理的效率提高了,似乎也沒有必要維持序列,獨立運作也並無不可。
程式說明
這段程式碼建立了一個使用 Flask 框架的 API 服務,其中包含兩個端點 /api/v1/get_dxf
和 /api/v1/get_cntr
。這兩個端點分別用於生成 DXF 文件和 PNG 圖像,並將其返回給客戶端。具體的工作流程如下:
導入必要的庫
Flask
用於創建 web 應用。BytesIO
用於處理內存中的文件。cntr
和dxf
分別從mem2cntr
和mem2dxf
模組中導入,用於生成 PNG 圖像和 DXF 文件。
創建 Flask 應用
app = Flask(__name__)
初始化 Flask 應用。
靜態文件的端點
serve_html
函數返回index.html
文件。- html文件內容詳見index.html的設計
生成 DXF 文件的端點
get_dxf
函數接收 POST 請求中的 JSON 數據,提取西南和東北角的經緯度,- 並調用
dxf
函數生成 DXF 文件(詳mem2dxf.py)。 - 生成的文件以
BytesIO()
附件形式返回。
生成 PNG 圖像的端點
get_cntr
函數接收 POST 請求中的 JSON 數據,提取西南和東北角的經緯度,- 調用
cntr
函數生成 PNG 圖像。 - 生成的圖像也以
BytesIO()
附件形式返回。
主程序入口
- 當程序以腳本形式運行時,啟動 Flask 服務。
- ip及端口在此設定
- 啟動偵錯模型
呼叫方式
- API程式可以直接使用curl指令從後端主機呼叫,可以不需要前端程式,在測試階段可以用
curl
來得到api程式的結果。 - 如以下呼叫
cntr
程式的指令
url=http://devp.sinotech-eng.com:5000/api/v1/get_file
curl -X POST $url -H "Content-Type: application/json" -d '{"sw_lat": 22.507545744146224, "sw_lon": 120.61295698396863, "ne_lat": 22.535073497331727, "ne_lon": 120.64468000957278}' --output a.png
- 為串連前端程式,一般使用js指令
fetch
來連接API伺服器,詳見前端網頁的設計#_save方法
代碼說明
以下是完整的代碼:
from flask import Flask, request, jsonify, send_file, send_from_directory
import pandas as pd
from io import BytesIO
from mem2cntr import cntr, rd_mem
from mem2dxf import dxf
app = Flask(__name__)
# 靜態文件的端點
@app.route('/')
def serve_html():
return send_from_directory('.', 'index.html')
# 生成 DXF 文件的端點
@app.route('/api/v1/get_dxf', methods=['POST'])
def get_dxf():
try:
data = request.json
print("Received data:", data) # 調試輸出
sw_lat = data.get('sw_lat')
sw_lon = data.get('sw_lon')
ne_lat = data.get('ne_lat')
ne_lon = data.get('ne_lon')
# 檢查是否傳入了所有必要的數據
if not all([sw_lat, sw_lon, ne_lat, ne_lon]):
return jsonify({"error": "缺少必要的經緯度參數"}), 400
# 將 DataFrame 寫入內存中的 CSV 文件
fname, output = dxf((sw_lat, sw_lon), (ne_lat, ne_lon))
if fname == 'LL not right!':
return jsonify({"error": "經緯度參數超過範圍"}), 400
return send_file(output, mimetype='application/dxf', download_name=fname, as_attachment=True)
except KeyError as e:
return jsonify({"error": f"缺少必要的字段: {str(e)}"}), 400
except Exception as e:
print("Exception occurred:", str(e)) # 調試輸出
return jsonify({"error": str(e)}), 500
@app.route('/api/v1/get_cntr', methods=['POST'])
def get_cntr():
try:
data = request.json
print("Received data:", data) # 調試輸出
sw_lat = data.get('sw_lat')
sw_lon = data.get('sw_lon')
ne_lat = data.get('ne_lat')
ne_lon = data.get('ne_lon')
# 檢查是否傳入了所有必要的數據
if not all([sw_lat, sw_lon, ne_lat, ne_lon]):
return jsonify({"error": "缺少必要的經緯度參數"}), 400
# 將 DataFrame 寫入內存中的 CSV 文件
fname, output = cntr((sw_lat, sw_lon), (ne_lat, ne_lon))
if fname == 'LL not right!':
return jsonify({"error": "經緯度參數超過範圍"}), 400
return send_file(output, mimetype='image/png', download_name=fname, as_attachment=True)
except KeyError as e:
return jsonify({"error": f"缺少必要的字段: {str(e)}"}), 400
except Exception as e:
print("Exception occurred:", str(e)) # 調試輸出
return jsonify({"error": str(e)}), 500
# 主程式:開啟偵錯、IP、端口
if __name__ == '__main__':
print("Starting Flask server...")
app.run(debug=True, host='devp.sinotech-eng.com', port=5000)
此應用程序提供了一個簡單的網頁界面,並且可以通過 API 調用來生成並下載 DXF 文件和 PNG 圖像。確保 mem2cntr
和 mem2dxf
模組正確地被導入並且運行正常。
運轉維護
app.py紀錄
- app.py增添下列指令
import logging
from datetime import datetime
app = Flask(__name__)
# 設定日誌配置
logging.basicConfig(
level=logging.INFO,
format='%(asctime)s %(levelname)s: %(message)s',
datefmt='%Y-%m-%d %H:%M:%S',
handlers=[
logging.FileHandler('user_activity.log'),
logging.StreamHandler()
]
)
@app.route('/log_console_output', methods=['POST'])
def log_console_output():
# 獲取前端傳來的console輸出
console_output = request.get_json().get('console_output')
logging.info(f"Console output: {console_output}")
error_output = request.get_json().get('errort')
logging.info(f"Error output: {error_output}")
bound_output = request.get_json().get('Saved bounds')
logging.info(f"Bound output: {bound_output}")
return jsonify({'status': 'success'})
...
print("Received data:", data) # 调试输出
sw_lat = data.get('sw_lat')
sw_lon = data.get('sw_lon')
ne_lat = data.get('ne_lat')
ne_lon = data.get('ne_lon')
with open('user_activity.log', 'a') as log_file:
log_file.write(f"Received data: {data}\n")
index.html啟動紀錄
...
<script>
window.addEventListener('error', function(event) {
// 獲取錯誤資訊
var errorMessage = event.message;
var errorStack = event.error.stack;
// 使用AJAX將錯誤資訊傳送到Flask後端
fetch('/log_console_output', {
method: 'POST',
headers: {
'Content-Type': 'application/json'
},
body: JSON.stringify({
console_output: `Error: ${errorMessage}\nStack: ${errorStack}`
})
})
.then(response => {
console.log('Console output logged successfully');
})
.catch(error => {
console.error('Error logging console output:', error);
});
});
</script>
...
定期檢查與啟動
- 上班日每天7時進行檢查,如果沒有運轉,則予以啟動
- 如果前一天有人執行,則將紀錄檔案重新更名。
$ crontab -l|grep ck_up
0 7 * * 1-5 /nas2/kuang/MyPrograms/CADNA-A/ck_up.cs >& /dev/null 2>&1
$ cat ck_up.cs
#!/bin/bash
#default running at DEVP.sinotech-eng.com:5000
cd /nas2/kuang/MyPrograms/CADNA-A
n=$(ps -ef|grep app.py|wc -l)
if [ $n -lt 3 ];then
~/.conda/envs/pyn_env/bin/python app.py
echo excuted
fi
if [ -e "user_activity.log" ];then mv user_activity.log user_activity.log.$(date -d "yesterday" +%Y%m%d);fi