本文主要給大家介紹的是關于利用python/103186.html">python模擬實現(xiàn)POST請求提交圖片的方法,分享出來供大家參考學習,下面來一看看詳細的介紹:
使用requests來模擬HTTP請求本來是一件非常輕松的事情,比如上傳圖片來說,簡單的幾行代碼即可:
import requestsfiles = {'attachment_file': ('1.png', open('1.png', 'rb'), 'image/png', {})}values = {'next':"http://www.xxxx.com/xxxx"}r = requests.post('http://www.xxxx.com/upload', files=files, data=values) # 成功r = requests.post('http://www.xxxx.com/upload', files=files, data=values) # 失敗r = requests.post('http://www.xxxx.com/upload', files=files, data=values) # 失敗r = requests.post('http://www.xxxx.com/upload', files=files, data=values) # 失敗r = requests.post('http://www.xxxx.com/upload', files=files, data=values) # 失敗...不過我今天在調(diào)試一個django程序的時候卻遇到了大坑————為了偷懶,我直接在ipython中執(zhí)行了上述代碼,第一次提交的時候一切正常,但第二次之后提交就怎么也通過不了django的form驗證。
驗證部分的代碼很簡單:
......form = AttachmentForm(request.POST, request.FILES)if form.is_valid(): form.save(request, obj) messages.success(request,_('Your attachment was uploaded.')) return HttpResponseRedirect(next)......什么鬼!?怎么只有第一次成功提交???后面全失敗??只好一步一步的跟進到django源碼中,發(fā)現(xiàn)問題出在django/forms/fields.py文件中:
def to_python(self, data): if data in validators.EMPTY_VALUES: return None # UploadedFile objects should have name and size attributes. try: file_name = data.name file_size = data.size except AttributeError: raise ValidationError(self.error_messages['invalid']) if self.max_length is not None and len(file_name) > self.max_length: error_values = {'max': self.max_length, 'length': len(file_name)} raise ValidationError(self.error_messages['max_length'] % error_values) if not file_name: raise ValidationError(self.error_messages['invalid']) if not self.allow_empty_file and not file_size: raise ValidationError(self.error_messages['empty']) return data 在第一次執(zhí)行的時候,一切正常,這個data即InMemoryUploadFile文件類型,name、size就是我們上傳的圖片名、大小,而第二次執(zhí)行post請求時候,發(fā)現(xiàn)data.size居然變成了0?!怪不得直接引發(fā)了if not self.allow_empty_file and not file_size這個判斷的異常呢!
由此可知,問題的核心并不出現(xiàn)在django對于表單驗證的部分,而是出自發(fā)送請求的部分。不過發(fā)請求的部分代碼很簡單啊?分別輸出了正常情況和錯誤情況requests發(fā)出的請求包,發(fā)現(xiàn)區(qū)別了:
#正常情況In [28]: r = requests.post('http://www.xxxx.com/upload', files=files, data=values)In [29]: r.request.body#錯誤情況In [33]: r = requests.post('http://www.xxxx.com/upload', files=files, data=values)In [34]: r.request.bodyOut[34]: '--155322d3e780432bb06e58135e041c8f/r/nContent-Disposition: form-data; name="next"/r/n/r/nhttp://www.xxxx.com/upload/r/n--155322d3e780432bb06e58135e041c8f/r/nContent-Disposition: form-data; name="attachment_file"; filename="1.png"/r/nContent-Type: image/png/r/n/r/n/r/n--155322d3e780432bb06e58135e041c8f--/r/n'正常情況沒輸出,錯誤情況反而看著像正常情況下的輸出?這不科學啊?
結(jié)合以上2點,我隱約感覺問題出在數(shù)據(jù)的構(gòu)造上,關鍵在于files = {'attachment_file': ('1.png', open('1.png', 'rb') , 'image/png', {})}這里,首先關于字典、列表這種可變類型作為函數(shù)的參數(shù)傳遞時候就需要特別注意,其次open函數(shù)打開了一個文件,那么哪里關閉文件了呢?
帶著這個懷疑,我把代碼改寫成:
fl = open('1.png','rb')files = {'attachment_file': ('1.png', fl, 'image/png', {})}r1 = requests.post('http://www.xxxx.com/upload', files=files, data=values)fl.close()fl = open('1.png','rb')files = {'attachment_file': ('1.png', fl, 'image/png', {})}r2 = requests.post('http://www.xxxx.com/upload', files=files, data=values)然后再執(zhí)行,果然成功上傳了2張圖片。其實按照正常情況不會出現(xiàn)測試時候這種打開一張圖片不停上傳的情形,不過也正因為這樣才會遇到如此有意思的問題。關于requests中files對象的處理代碼在models.py文件中,有興趣的讀者可以自行調(diào)試。
另外,requests調(diào)用時上傳文件名中不能包含中文,否則也不能通過django表單驗證,這里也不深究原因了。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作能帶來一定的幫助,如果有疑問大家可以留言交流,謝謝大家對VEVB武林網(wǎng)的支持。
新聞熱點
疑難解答
圖片精選