基于soap協(xié)議使用python模擬短信網(wǎng)關(guān)
問題1:soap協(xié)議是有什么作用? 答:soap提供了一種標(biāo)準(zhǔn)的方法,使得運行在不同的操作系統(tǒng)并使用不同的技術(shù)和編程語言的應(yīng)用程序可以互相進(jìn)行通信。 用一個簡單的例子來說明soap使用過程,一個soap消息可以發(fā)送到一個具有Web Service功能的Web站點。 例如:一個含有房價信息的數(shù)據(jù)庫,消息的參數(shù)中標(biāo)明這是一個查詢消息,此站點將返回一個XML格式的信息,其中包含了查詢 結(jié)果(價格,位置,特點,或者其它信息)。由于數(shù)據(jù)是用一種標(biāo)準(zhǔn)化的可分析的結(jié)構(gòu)來傳遞的,所以可以直接被第三方站點所 利用。
問題2:如何實現(xiàn)soap? 答: 簡單理解,開放協(xié)議soap=rpc+http+xml,采用http作為底層通訊協(xié)議;rpc作為一致性的調(diào)用途徑;xml作為數(shù)據(jù)傳送的格式。
問題3:如何使用python實現(xiàn)基于soap協(xié)議的短信網(wǎng)關(guān)Web Service? 答: Service 是使用soaplib搭建。 sudo apt-get install libxml2-dev sudo apt-get install libxslt-dev pip install soaplib
soaplib2.0.0.beta1 官網(wǎng):http://soaplib.github.com/soaplib/2_0/
python實現(xiàn)soap協(xié)議的server.py
import soaplib from soaplib.core.service import rpc, DefinitionBase, soap from soaplib.core.model.primitive import String, Integer from soaplib.core.server import wsgi from soaplib.core.model.clazz import Array useridDict = {"Jack":"123456"} def recordSM(tos, msg, time): _recordFile = open("../resource/sm", 'a') _recordString = 'to:%s msg:%s time:%s\n' % (tos, msg, time) _recordFile.write(_recordString) _recordFile.close() class IsmgService(DefinitionBase): @soap(String, String, String, String, String, _returns=String) def sendSms(self, userid, password, mobiles, msg, time): if useridDict.has_key(userid): if useridDict[userid] == password: recordSM(mobiles, msg, time) return 'success' return 'lose' if __name__=='__main__': try: from wsgiref.simple_server import make_server soap_application = soaplib.core.Application([IsmgService], 'tns') wsgi_application = wsgi.Application(soap_application) server = make_server('localhost', 7789, wsgi_application) server.serve_forever() except ImportError: print "Error: example server code requires Python >= 2.5"
Client 官網(wǎng):https:///suds/ svn co http://svn./svn/suds/trunk pip install . python實現(xiàn)的client
from suds.client import Client client = Client('http://localhost:7789/?wsdl') userid = 'Jack' password = '123456' mobiles = '13812345678' msg = 'hello world!' time = '2011-03-01' result = client.service.sendSms(userid, password, mobiles, msg, time) print result
備注: 出現(xiàn)錯誤 URLError:<urlopen error[Error -2]> name or service not know 產(chǎn)生的原因 在不能與外網(wǎng)通信的情況下,soaplib和suds.client要去請求www.進(jìn)行校驗 解決方案: 把soaplib自動生成的wsdl保存到本地,去掉<xs:import namespace="http://www./2001/XMLSchema" />,用python -m SimpleHTTPServer起一個Http server 例如http://127.0.0.1:7788/wsdl,Client去引用這個URL,如Client('http://localhost:7788/wsdl')
問題4:什么是wsdl? 答:是基于XML的用于描述Web Services以及如何訪問Web Services的語言。 這種文檔可描述某個Web Services。它可規(guī)定服務(wù)的位置,以及此服務(wù)提供的操作(或方法)。
<portType> web service 執(zhí)行的操作 它可描述一個web service, 可被執(zhí)行的操作,以及相關(guān)的消息 可以把<portType>元素比作一個函數(shù)庫
<message> web service 使用的消息 每個消息均由一個或多個部件組成??梢园堰@些部件比作一個函數(shù)調(diào)用的參數(shù)
<types> web service 使用的數(shù)據(jù)類型 為了最大程度的平臺中立性,wsdl使用xml Schema語法來定義數(shù)據(jù)類型
<binding> web service 使用的通信協(xié)議 為每個端口定義消息格式和協(xié)議細(xì)節(jié) binding元素有兩個屬性-name屬性和type屬性 name屬性定義binding的名稱,而type屬性指向用于binding的端口。
具體作用: 1.服務(wù)做些什么--服務(wù)所提供的操作(方法) 2.如何訪問服務(wù)--和服務(wù)交互的數(shù)據(jù)格式以及必要協(xié)議 3.服務(wù)位于何處--協(xié)議相關(guān)的地址,如URL
portType(與message和type元素的細(xì)節(jié)相結(jié)合)描述了Web服務(wù)是什么,binding元素描述了如何使用Web服務(wù),port及service元素描述了Web服務(wù)的位置。
使用方式: 可以根據(jù)wsdl反向生成客戶端代碼,如python的suds.client庫 是可以通過瀏覽器去訪問的,但是需要wsdl文檔支持。例如:URL是http://xxx/method?arg1=xx&arg2=xx
備注:
webService協(xié)議主要包括兩個方面:傳輸協(xié)議和數(shù)據(jù)表示,關(guān)于傳輸協(xié)議可以是http或其它,數(shù)據(jù)表示也可以是鍵值對,xml或者其它。
問題5:什么是wsgi? 答: wsgi的全寫是Web Server Gateway interface。定義一個標(biāo)準(zhǔn)的溝通方式,讓你寫的程序可以 和服務(wù)器溝通,但是wsgi不是設(shè)計用來給任何語言使用的,它是設(shè)計給python用的。wsgi透過環(huán) 境變量來取得信息,例如request_method,server_name,http_xxx。
實例展示:
from wsgiref.simple_server import make_server def my_app(environ, start_response): status = '200 OK' response_headers = [('Content-type', 'text/plain')] start_response(status, response_headers) return [u"你好! 歡迎來到Victor的第一個WSGI程式".encode('utf8')] httpd = make_server('', 8000, my_app) print "Serving on port 8000..." httpd.serve_forever()
wsgiref是python內(nèi)建模塊,通過瀏覽器訪問http://127.0.0.1:8000
利用wsgi你可以寫一個Middleware
from wsgiref.simple_server import make_server def my_app(environ, start_response): status = '200 OK' response_headers = [('Content-type', 'text/plain')] start_response(status, response_headers) return ['hello world'] class Middleware(object): def __init__(self, app, encoding='utf8'): self.app = app self.encoding = encoding def __call__(self, environ, start_response): content = self.app(environ, start_response) for item in content: if item == 'hello world': return 'Hi!' return "hi!" app = Middleware(my_app) httpd = make_server('', 8000, app) print "Serving on port 8000..." httpd.serve_forever()
wsgi帶來的好處: 最簡單的網(wǎng)頁程序做的事情就使用者發(fā)送一個request到你的服務(wù)器,接著你的程序產(chǎn)生內(nèi)容,然后送出response回去給Browser。 復(fù)雜的事情如:session,cookie,認(rèn)證,錯誤處理,而wsgi的Middleware將這些該做的事情限定于Middleware中,它們跟上下 游溝通都是按標(biāo)準(zhǔn)來實現(xiàn),這些Middleware都能重復(fù)被利用。
詳細(xì)描述整個請求過程: 首先,經(jīng)過Registry Manager是用來管理request-local的相關(guān)物件。 接著,Status Code Redirect在Request上不做任何事情。 往下層跑,ErrorHandler也是不對Request做任何事。 接著,Cache Middle,視情況記錄Request相關(guān)內(nèi)容。 然后,是Session Middleware取出cookie還有在環(huán)境中準(zhǔn)備session的資料。 接著,Routes依網(wǎng)址決定要給那個Controller來處理。 最后,Request經(jīng)過層層關(guān)卡來到了真正的Pylons App,產(chǎn)生網(wǎng)頁結(jié)果,Response會一層一層被回傳回去。 到Cache有需要可以暫存下來,其中有例外拋出ErrorHandler會處理,Status Code Redirect可以重導(dǎo)頁面到特定的錯誤頁面 |
|