最近項目中需要與管易雲erp做對接,看了他的接口文檔,php的示例代碼,於是用python仿寫。
其中傳的參數data中前面幾個json數據是固定的,最后需要加一個簽名,該簽名是對前面的json數據字符串化后,首尾拼接上screct字符串,再做md5處理(32位大寫),再將該簽名添加到之前的json中作為post參數傳遞過去。
問題就出在組裝json字符串和簽名中,因為python內置的字典是無序的,導致我組裝好的json數據作為參數傳遞給自己編寫的簽名函數時,字典內部的順序是變化的,所以簽名前后的md5值不一樣,導致頻頻報錯,自己起初沒在意數據順序的事,因為對接金蝶erp時沒有簽名這一項,數據傳遞的很簡單,導致排錯時方向出現錯誤。后來想到這了顧開始解決。
解決方法是使用collections庫中的OrderedDict(有序字典)模塊,組裝好的數據就不會亂序,做的md5簽名也就前后一致了
但,在對json數據使用json.dumps()方法時,字符串化后的數據在逗號后會有一個空格,導致md5值出錯
原理:
解決方法是
json.dumps(data,separators=(',',':'))
最后,將自己寫的函數留下來做個例子
def getShops(): data = OrderedDict() data["appkey"] = appkey data["sessionkey"] = sessionkey data["method"] = method data["page_no"] = "1" data["page_size"] = "10" data["sign"] = sign(data,secret) response = requests.post(url=url, data=json.dumps(data)) print(response.text) def sign(data,secret): str = json.dumps(data,separators=(',',':')) fullStr = secret+str+secret signCode = hashlib.md5(fullStr.encode("utf-8")).hexdigest().upper() #print(signCode) return signCode
再訪問新增訂單接口時,又報簽名出錯,按着之前的邏輯去查,發現json.dumps()含中文數據時,會將中文默認轉為ascii碼,從而導致MD5不一致,需轉成中文才行
原理:
def dumps(obj, skipkeys=False, ensure_ascii=True, check_circular=True, allow_nan=True, cls=None, indent=None, separators=None, default=None, sort_keys=False, **kw):
解決:將上述代碼紅色參數改為False即可。