Python中WebService客戶(hù)端接口調(diào)用及身份驗(yàn)證的問(wèn)題
WebService客戶(hù)端接口調(diào)用及身份驗(yàn)證問(wèn)題
最近由于業(yè)務(wù)需求,需要實(shí)現(xiàn)python Webservice的服務(wù)以及接口調(diào)用。
服務(wù)端代碼可自行百度,這里主要描述客戶(hù)端以及我遇到的HTTP身份驗(yàn)證的問(wèn)題,不多說(shuō)直接上代碼。
from suds.client import Client from suds.transport.http import HttpAuthenticated import base64 import urllib2 # 一:無(wú)需身份驗(yàn)證的簡(jiǎn)單調(diào)用 url = "http://localhost:8899/?wsdl" client = Client(url) # 可以print client進(jìn)行相關(guān)信息查看 client.service.methodName(*args) # 方法調(diào)用 req = str(client.last_sent()) # 保存請(qǐng)求報(bào)文,因?yàn)榉祷氐氖且粋€(gè)實(shí)例,所以要轉(zhuǎn)換成str response = str(client.last_received()) # 保存返回報(bào)文,返回的也是一個(gè)實(shí)例 # 二:需要身份驗(yàn)證的調(diào)用 # 1. client = Client(url=wsdl_url, username=username, password=password) # 2. t = HttpAuthenticated(username=username, password=password) client = Client(url=url, transport=t) # 3. t = HttpAuthenticated(username=username, password=password) t.handler = urllib2.HTTPBasicAuthHandler(t.pm) # 這種我的報(bào)錯(cuò)了,t.pm告知我沒(méi)這個(gè)對(duì)象 t.urlopener = urllib2.build_opener(t.handler) client = Client(url=wsdl_url, transport=t) # 4. base64string = base64.encodestring('%s:%s' % (username, password)).replace('\n', '') authenticationHeader = { "SOAPAction" : "ActionName", "Authorization" : "Basic %s" % base64string } client = Client(url=wsdl_url, headers=authenticationHeader)
這是我針對(duì)身份驗(yàn)證做的一些資料搜集、匯整。
Python調(diào)用WebService接口踩坑記錄
應(yīng)用場(chǎng)景:
?需要與某運(yùn)營(yíng)商的某部門(mén)進(jìn)行某些數(shù)據(jù)的對(duì)接,對(duì)方扔了一個(gè)接口文檔過(guò)來(lái),需要根據(jù)文檔中的WebService接口規(guī)范進(jìn)行數(shù)據(jù)的上報(bào)。但是在調(diào)用對(duì)面接口的時(shí)候,一直返回500。雖然狀態(tài)碼是500,但是根據(jù)報(bào)錯(cuò)信息以及給錢(qián)就是大爺?shù)脑瓌t,最后還是需要調(diào)整自身的腳本文件來(lái)完成對(duì)接。
1.報(bào)錯(cuò)信息
一開(kāi)始跟大部分帖子使用的第三方庫(kù)一樣,我也是使用了suds庫(kù)。
#步驟很簡(jiǎn)單,導(dǎo)入相關(guān)庫(kù),定義url獲取返回即可 import suds from suds.client import Client url='xxx' client=Clinet(url) result = client.service.xxx('xxx') #第一個(gè)xxx是webservice接口中你要調(diào)用的函數(shù)名,第二個(gè)xxx是你要輸入的參數(shù)。如果不需要就空著。假如是復(fù)雜參數(shù),可以使用client.factory方式來(lái)構(gòu)建。
上述三個(gè)步驟就能夠調(diào)用最簡(jiǎn)單的webservice接口了。
但是。。。。。
按理說(shuō)是很簡(jiǎn)單的一個(gè)步驟,沒(méi)想到卡我好幾天。
1.1報(bào)錯(cuò)信息一
suds.WebFault: Server raised fault: 'Fault occurred while processing
百度了以后,大部分帖子解釋的錯(cuò)誤原因是由于上傳的數(shù)據(jù)中存在空值,需要將空值替換成NULL等等。
所以我就想著是否是數(shù)據(jù)出了問(wèn)題,然后檢查了一下自己傳入的數(shù)據(jù)有沒(méi)有少了某個(gè)字段,或者某些字段是空著的
然后對(duì)著接口文檔一陣猛看,最后發(fā)現(xiàn)自己的數(shù)據(jù)很標(biāo)準(zhǔn),甚至找到了接口文檔中好些錯(cuò)別字。。。
然后就懷疑是否跟xml數(shù)據(jù)的格式有關(guān)系,縮進(jìn)換行啥的。。然后引發(fā)了第二個(gè)報(bào)錯(cuò)。
1.2報(bào)錯(cuò)信息二
Error reading XMLStreamReader: Illegal processing instruction target ("xml"); xml (case insensitive) is reserved by the specs.
at [row,col {unknown-source}]: [2,5]
這個(gè)報(bào)錯(cuò)信息說(shuō)的是我這邊的xml開(kāi)頭不規(guī)范,但是我一看我的數(shù)據(jù):
<?xml version="1.0" encoding="utf-8"?>
這很標(biāo)準(zhǔn)啊,也絲毫沒(méi)有問(wèn)題。。。
然后百度了說(shuō)是得在數(shù)據(jù)前后加
<![CDATA[ ?]]>
然后接著一通瞎操作也沒(méi)有解決上述兩個(gè)報(bào)錯(cuò)的任何一個(gè)。。。
但是。。。我最后還是繞過(guò)了500,成功調(diào)用!
2.解決方法
最后的解決方法就是使用requests庫(kù),不得不說(shuō)這個(gè)庫(kù)是真的頂,yyds!!!
webservice接口本質(zhì)上就是使用HTTP的POST請(qǐng)求,只不過(guò)他post過(guò)去的是xml格式的數(shù)據(jù)。之所以很多人使用suds庫(kù)是因?yàn)樵搸?kù)能自動(dòng)組裝好相應(yīng)xml開(kāi)頭,比如
<?xml version="1.0" encoding="utf-8"?> <soap:Envelope ? ? xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" ? ? xmlns:sam="http://service.springboot.huaxun.com/" ? ? xmlns:xsd="http://www.w3.org/2001/XMLSchema" ? ? xmlns:soap="http://schemas.xmlsoap.org/soap/envelope/"> ? ? <soap:Header/> ? ? <soap:Body> ? ? ? ? xxxxxxxxxxxxxxx ? ? </soap:Body> </soap:Envelope>
下方的代碼實(shí)際上是輸入Body中的數(shù)據(jù)。
result = client.service.xxx('xxx')
然后suds會(huì)自動(dòng)組裝好數(shù)據(jù)并post出去。
所以理論上,suds庫(kù)能完成的requests庫(kù)也可以完成,只不過(guò)xml得我們自己來(lái)組裝了。
造成報(bào)錯(cuò)一的主要原因是命名空間未正確指定,由于網(wǎng)上該庫(kù)的教程很少,我也沒(méi)能成功從源碼中找到對(duì)命名空間的修改方式,所以使用了requests。
url='xxx' str3='xxx' header={ ?? ?'Content-Type':'text/xml; charset=utf-8', } r = requests.post(url,headers=header,data=str3.encode('utf-8')) print(r) print(r.text)
打完收工!
有些遺憾的是,仍未成功使用suds調(diào)用成功!不清楚如何指定調(diào)用函數(shù)的namespace。
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
python使用response.read()接收json數(shù)據(jù)的實(shí)例
今天小編就為大家分享一篇python使用response.read()接收json數(shù)據(jù)的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-12-12Python實(shí)現(xiàn)以時(shí)間換空間的緩存替換算法
緩存是指可以進(jìn)行高速數(shù)據(jù)交換的存儲(chǔ)器,它先于內(nèi)存與CPU交換數(shù)據(jù),因此速度很快。緩存就是把一些數(shù)據(jù)暫時(shí)存放于某些地方,可能是內(nèi)存,也有可能硬盤(pán)。下面給大家介紹Python實(shí)現(xiàn)以時(shí)間換空間的緩存替換算法,需要的朋友參考下2016-02-02Django利用elasticsearch(搜索引擎)實(shí)現(xiàn)搜索功能
這篇文章主要介紹了Django利用elasticsearch(搜索引擎)實(shí)現(xiàn)搜索功能,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-11-11使用Python實(shí)現(xiàn)Mysql數(shù)據(jù)庫(kù)相關(guān)操作詳解
這篇文章主要介紹了使用Python實(shí)現(xiàn)Mysql數(shù)據(jù)庫(kù)相關(guān)操作詳解,pymysql是Python中操作數(shù)據(jù)庫(kù)的第三方模塊,通過(guò)這個(gè)模塊的相關(guān)方法,我們可以連接并且去操作mysql數(shù)據(jù)庫(kù),需要的朋友可以參考下2023-08-08在VSCode中添加Python解釋器并安裝Python庫(kù)的方法
這篇文章主要介紹了在VSCode中添加Python解釋器并安裝Python庫(kù)的方法,本文分步驟通過(guò)圖文并茂的形式給大家介紹的非常詳細(xì),需要的朋友可以參考下2023-02-02