深入解析Python中的WSGI接口
概述
WSGI接口包含兩方面:server/gateway 及 application/framework。
server調(diào)用由application提供的可調(diào)用對(duì)象。
另外在server和application之間還可能有一種稱作middleware的中間件。
可調(diào)用對(duì)象是指:函數(shù)、方法、類或者帶有callable方法的實(shí)例。
關(guān)于application
函數(shù)、方法、類及帶有callable方法的實(shí)例等可調(diào)用對(duì)象都可以作為the application object。
WSGI協(xié)議要求:
the application object接受兩個(gè)參數(shù)且可以被多次調(diào)用
這兩個(gè)參數(shù)分別為:
1.CGI式的字典;
2.回調(diào)函數(shù):application用來向server傳遞http狀態(tài)碼/消息/http頭
另外協(xié)議要求可調(diào)用對(duì)象必須將響應(yīng)體封裝成一個(gè)可迭代的strings返回。
# the application object. 可以使用其他名字, # 但是在使用mod_wsgi 時(shí)必須為 "application" def application( environ, start_response): # 函數(shù)接受兩個(gè)參數(shù): # environ :包含有CGI 式環(huán)境變量的字典,由server負(fù)責(zé)提供內(nèi)容 # start_response:由server提供的回調(diào)函數(shù),其作用是將狀態(tài)碼和響應(yīng)頭返回給server # 構(gòu)造響應(yīng)體,以可迭代字符串形式封裝 response_body = 'The request method was %s' % environ['REQUEST_METHOD'] # HTTP 響應(yīng)碼及消息 status = '200 OK' # 提供給客戶端的響應(yīng)頭. # 封裝成list of tuple pairs 的形式: # 格式要求:[(Header name, Header value)]. response_headers = [('Content-Type', 'text/plain'), ('Content-Length', str(len(response_body)))] # 將響應(yīng)碼/消息及響應(yīng)頭通過傳入的start_reponse回調(diào)函數(shù)返回給server start_response(status, response_headers) # 響應(yīng)體作為返回值返回 # 注意這里被封裝到了list中. return [response_body]
關(guān)于server
從概述中可以知道,WSGI server必須要調(diào)用application,同時(shí),從application的協(xié)議要求可知:
1. WSGI server必須向application提供環(huán)境參數(shù),因此,自身也必須能夠獲取環(huán)境參數(shù)。
2. WSGI server接收application的返回值作為響應(yīng)體。
最簡單的WSGI server為Python自帶的wsgiref.simple_server
示例如下:
from wsgiref.simple_server import make_server srv = make_server('localhost', 8080, hello_world) srv.serve_forever()
關(guān)于middleware
middleware的概念沒有appllication和server那么容易理解。
假設(shè)一個(gè)符合application標(biāo)準(zhǔn)的可調(diào)用對(duì)象,它接受可調(diào)用對(duì)象作為參數(shù),返回一個(gè)可調(diào)用對(duì)象的對(duì)象。
那么對(duì)于server來說,它是一個(gè)符合標(biāo)準(zhǔn)的可調(diào)用對(duì)象,因此是application。
而對(duì)于application來說,它可以調(diào)用application,因此是server。
這樣的可調(diào)用對(duì)象稱為middleware。
middleware的概念非常接近decorator。
以一個(gè)路由的例子示例:
import re # 這是一個(gè)標(biāo)準(zhǔn)的application object def index(environ, start_response): start_response('200 OK', [('Content-Type', 'text/html')]) return ['index page'] # 這是一個(gè)標(biāo)準(zhǔn)的application object def hello(environ, start_response): start_response('200 OK', [('Content-Type', 'text/html')]) return ['hello page'] # 這是一個(gè)標(biāo)準(zhǔn)的application object def not_found(environ, start_response): start_response('404 NOT FOUND', [('Content-Type', 'text/plain')]) return ['Not Found Page'] # map urls to functions urls = [ (r'^$', index), (r'hello/?$', hello) ] # 這是一個(gè)middleware # 根據(jù)不同的route返回不同的application object def application(environ, start_response): path = environ.get('PATH_INFO', '').lstrip('/') for regex, callback in urls: match = re.search(regex, path) if match is not None:
相關(guān)文章
關(guān)于Python 3中print函數(shù)的換行詳解
最近在學(xué)習(xí)python3,發(fā)現(xiàn)了一個(gè)問題想著總結(jié)出來,所以下面這篇文章主要給大家介紹了關(guān)于Python 3中print函數(shù)換行的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)需要的朋友們具有一定的參考學(xué)習(xí)價(jià)值,感興趣的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。2017-08-08python基于xml parse實(shí)現(xiàn)解析cdatasection數(shù)據(jù)
這篇文章主要介紹了python基于xml parse實(shí)現(xiàn)解析cdatasection數(shù)據(jù)的方法,是非常實(shí)用技巧,需要的朋友可以參考下2014-09-09在PyCharm中控制臺(tái)輸出日志分層級(jí)分顏色顯示的方法
今天小編就為大家分享一篇在PyCharm中控制臺(tái)輸出日志分層級(jí)分顏色顯示的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2019-07-07python正則表達(dá)式去除兩個(gè)特殊字符間的內(nèi)容方法
今天小編就為大家分享一篇python正則表達(dá)式去除兩個(gè)特殊字符間的內(nèi)容方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-12-12Python Http請(qǐng)求json解析庫用法解析
這篇文章主要介紹了Python Http請(qǐng)求json解析庫用法解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-11-11