MicroPython
OpenAI
json
在 MicroPython 中利用 HTTP POST 傳送中文如果不注意會出錯, 本文以 OpenAI API 為例。OpenAI 雖然提供有 Python 的官方套件, 不過如果你是要在 MicroPython 中使用 OpeAnAI 的 API, 並不能直接套用, 這時就要回歸到 OpenAI API 最根本的 HTTP Post API 了。
OpenAI 是透過 HTTP Post 提供服務, 以聊天為例, 的是 ChatCompletion 服務:
因此, 最簡單的 OpenAI HTTP Post API 的程式就像是這樣:
執行結果如下:
既然是使用 HTTP Post, 哪麼只要從 requests 模組改成 urequests 模組, 應該就可以原封不動照搬程式了, 我們來試看看:
不過執行後就會看到 OpenAI 伺服器端回覆 400 錯誤:
同樣的程式, 搬到 MicroPython 上會出錯, 第一個懷疑的就是中文編碼的問題, 如果把程式中傳遞的 "你好" 改成純英文試看看:
再執行一次就會發現可以正確執行:
顯然問題就是出在 urequests.post 對於 json 參數的處理。
在 urequests.post
中會使用 json
模組 (MicroPython 中 json 與 ujson 是同一個模組) 的 dumps
函式將 Python 字典轉成字串格式的 JSON 資料, 可是它的輸出結果會保留以 UTF-16 編碼的中文字, 例如:
其中 \u4f60 是 "你" 的 UTF-16 編碼, 但是 json 規格需要的是 UTF8 編碼, 或是使用 "\u" 跳脫序列標註的 UTF-16 編碼, 好在 bytes 的 encode
方法可以幫我們將字串轉換成 UTF8 編碼的位元組串:
這樣結果就對了。不過因為要自行處理字典轉 json 格式位元組的工作, 所以就不能直接在 urequests.post
中使用 json 參數傳入字典了。
urequests.post
有 data
參數可以直接傳入要送給伺服端的資料, 因此我們就可以將程式改成如下:
要特別留意的是使用 json
參數傳入 Python 字典時, urequests.post
會幫你在表頭加上 'Content-Type: application/json', 自行使用 data
參數傳入 json 資料時就要記得在表頭補上標示遞交內容的格式, 否則無法正確執行。
這樣一來, 就可以正確叫用 API 了: