API伺服器的設計

Table of contents

背景

整體流程

pngs/2024-12-10-17-36-50.png

  • 這支程式(app.py)接收前端html leaflet元件的呼叫、傳入bounds座標值,引用cntrdxf2支函式進行檔案切割、製圖,最後將結果檔回傳給前端,儲存在使用者本機的下載目錄。

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 用於處理內存中的文件。
  • cntrdxf 分別從 mem2cntrmem2dxf 模組中導入,用於生成 PNG 圖像和 DXF 文件。

創建 Flask 應用

  • app = Flask(__name__) 初始化 Flask 應用。

靜態文件的端點

生成 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

代碼說明

以下是完整的代碼:

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 圖像。確保 mem2cntrmem2dxf 模組正確地被導入並且運行正常。

運轉維護

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