日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

python wsgiref源碼解析

瀏覽:26日期:2022-06-28 10:32:38

python web開發中http請求的處理流程通常是: web-browser , web-server , wsgi 和 web-application四個環節, 我們學習過基于bottle實現的web-application,也學習了http.server。再完成python3源碼中自帶的wsgiref的庫,就可以拼接最后一個環節wsgi。本文會分下面幾個部分:

wsgi相關概念 cgi示例 wsgiref源碼 wsgi小結 小技巧 wsgi 相關概念CGI

CGI(Common Gateway Interface)通用網關接口。1993年由美國NCSA(National Center for Supercomputing Applications)發明。它具有簡單易用、語言無關的特點。雖然今天已經少有人直接使用CGI進行編程,但它仍被主流的Web服務器,如Apache、IIS、Nginx等廣泛支持。

python wsgiref源碼解析

CGI提供了一種接口規范,可以讓應用程序, 一般是各種腳本語言,比如perl, php, python等來擴展web服務,讓服務動態起來。

WSGI

WSGI(Web Server Gateway Interface)web服務網關接口。是web服務和web應用程序之間的接口規范,在PEP3333中提出。

python wsgiref源碼解析

wsgi讓應用程序和web服務之間解耦,應用程序只需要遵守規范,就可以在各種不同的web服務部署運行。比如上圖中,基于flask/django實現的應用程序可以使用gunicorn部署,也可以使用nginx+uwsgi部署。

ASGI

ASGI(Asynchronous Server Gateway Interface) 異步服務器網關接口。ASGI繼承自wsgi,旨在在具有異步功能的Python Web服務器,框架和應用程序之間提供標準接口。ASGI具有WSGI向后兼容性實現以及多個服務器和應用程序框架。

python wsgiref源碼解析

wsgi中使用請求響應模型,每個請求可以同步獲得一個響應。在ASGI中,請求的響應變成異步實現,一般用于websocket協議。(asgi的內容,涉及異步實現,本文就不多介紹)

cgi 示例

單純的概念理解比較難。下面我們配合示例一起來學習,先從CGI開始。

http 模塊提供了一個簡單的文件目錄服務:

python3 -m http.serverServing HTTP on :: port 8000 (http://[::]:8000/) ...

這個服務只有靜態的展示功能,我們可以利用cgi擴展一個動態功能。

cgi腳本

創建cgi-bin目錄,這是CGI中約定的目錄名稱。然后編寫 hello.py, 代碼如下:

#!/usr/bin/env pythonimport timeimport sqlite3import osDB_FILE = 'guests.db'def init_db():pass # 詳情請見附件def update_total(ts):pass # 詳情請見附件print(’<html>’)print(’<head>’)print(’<meta charset='utf-8'>’)print(’<title>Hello Word!</title>’)print(’</head>’)print(’<body>’)print(’<h2>Hello Python!</h2>’)if not os.path.exists(DB_FILE):init_db()total = update_total(time.time())print(f’total guest: {total}!’)print(’</body>’)print(’</html>’)

為了代碼簡潔,省略了db操作部分的具體實現。還需要給腳本可執行權限:

源碼在這里

chmod 755 hello.py

./hello.py<html><head><meta charset='utf-8'><title>Hello Word!</title></head><body><h2>Hello Python!</h2>total guest: 4!</body></html>

啟動http.server中的cgi服務:

python -m http.server --cgi

注意后面的 --cgi 參數,讓服務使用cgi-handler。啟動后使用 curl 訪問:

curl -v http://127.0.0.1:8000/cgi-bin/hello.py* Trying 127.0.0.1...* TCP_NODELAY set* Connected to 127.0.0.1 (127.0.0.1) port 8000 (#0)> GET /cgi-bin/hello.py HTTP/1.1> Host: 127.0.0.1:8000> User-Agent: curl/7.64.1> Accept: */*>* HTTP 1.0, assume close after body< HTTP/1.0 200 Script output follows< Server: SimpleHTTP/0.6 Python/3.8.5< Date: Sun, 31 Jan 2021 13:09:29 GMT< <html>< <head>< <meta charset='utf-8'>< <title>Hello Word!</title>< </head>< <body>< <h2>Hello Python!</h2>< total guest: 5! # 訪客數< </body>< </html>* Closing connection 0

可以看到 hello.py 正確執行,訪客數+1。因為數據存儲在db中,重啟服務仍然有效。

cgi服務實現

cgi的實現,主要就是下面的代碼:

# http.serverclass CGIHTTPRequestHandler(SimpleHTTPRequestHandler): def run_cgi(self): import subprocess cmdline = [scriptfile] if self.is_python(scriptfile): interp = sys.executable cmdline = [interp, ’-u’] + cmdline if ’=’ not in query: cmdline.append(query) try: nbytes = int(length) except (TypeError, ValueError): nbytes = 0 p = subprocess.Popen(cmdline,stdin=subprocess.PIPE,stdout=subprocess.PIPE,stderr=subprocess.PIPE,env = env) if self.command.lower() == 'post' and nbytes > 0: data = self.rfile.read(nbytes) # throw away additional data [see bug #427345] while select.select([self.rfile._sock], [], [], 0)[0]: if not self.rfile._sock.recv(1): break stdout, stderr = p.communicate(data) self.wfile.write(stdout) p.stderr.close() p.stdout.close() status = p.returncode

可見cgi的實現就是:

使用subprocess.Popen新開了一個進程去執行腳本 重定向腳本的輸出到當前socket的wfile,也就是http請求的返回上

代碼也驗證了為什么需要授予 hello.py 的可執行權限。

從例子可以了解到http.server專注于提供http服務,app.py專注于業務功能,兩者通過cgi進行銜接。

wsgiref

wsgiref是python自帶的wsgi的實現參考(reference), 主要代碼結構:

文件 描述 handlers.py wsgi實現 headers.py 管理http-header simple_server.py 支持wsgi的http服務 util.py&&validator.py 工具和驗證器

WSGIServer的代碼:

class WSGIServer(HTTPServer): '''BaseHTTPServer that implements the Python WSGI protocol''' application = None def server_bind(self): '''Override server_bind to store the server name.''' HTTPServer.server_bind(self) self.setup_environ() def setup_environ(self): # 初始化環境變量 # Set up base environment env = self.base_environ = {} env[’SERVER_NAME’] = self.server_name env[’GATEWAY_INTERFACE’] = ’CGI/1.1’ env[’SERVER_PORT’] = str(self.server_port) env[’REMOTE_HOST’]=’’ env[’CONTENT_LENGTH’]=’’ env[’SCRIPT_NAME’] = ’’ def get_app(self): return self.application def set_app(self,application): # 注入application的class,注意是class self.application = application

WSGIServer并不復雜,繼承自http-server,接受application注入,就把web-server和we-application銜接起來。銜接后的動作,則是老規矩,交給HTTPRequestHandler去實現。同時wsgi服務多了一個準備env的動作,約定了一些wsgi的環境變量。

class WSGIRequestHandler(BaseHTTPRequestHandler): server_version = 'WSGIServer/' + __version__ def get_environ(self): pass def handle(self): '''Handle a single HTTP request''' self.raw_requestline = self.rfile.readline(65537) if len(self.raw_requestline) > 65536: ... self.send_error(414) return if not self.parse_request(): # An error code has been sent, just exit return handler = ServerHandler( self.rfile, self.wfile, self.get_stderr(), self.get_environ(), multithread=False, ) # 創建新的業務handler handler.request_handler = self handler.run(self.server.get_app()) # 創建application對象

WSGIRequestHandler覆蓋了handler,處理完成http協議(parse_request)后, 又做了四個動作:

創建environ 創建ServerHandler對象 創建app對象 運行app

environ處理主要是把http請求的header信息附帶在wsgi-server的環境變量上:

def get_environ(self): env = self.server.base_environ.copy() # wsgi-server的環境變量 env[’SERVER_PROTOCOL’] = self.request_version env[’SERVER_SOFTWARE’] = self.server_version env[’REQUEST_METHOD’] = self.command ... host = self.address_string() if host != self.client_address[0]: env[’REMOTE_HOST’] = host env[’REMOTE_ADDR’] = self.client_address[0] if self.headers.get(’content-type’) is None: env[’CONTENT_TYPE’] = self.headers.get_content_type() else: env[’CONTENT_TYPE’] = self.headers[’content-type’] length = self.headers.get(’content-length’) if length: env[’CONTENT_LENGTH’] = length for k, v in self.headers.items(): k=k.replace(’-’,’_’).upper(); v=v.strip() if k in env: continue # skip content length, type,etc. if ’HTTP_’+k in env: env[’HTTP_’+k] += ’,’+v # comma-separate multiple headers else: env[’HTTP_’+k] = v return env

ServerHandler對象的創建,接受輸入/輸出/錯誤,以及環境變量信息:

class ServerHandler(BaseHandler): def __init__(self,stdin,stdout,stderr,environ, multithread=True, multiprocess=False ): self.stdin = stdin self.stdout = stdout self.stderr = stderr self.base_env = environ self.wsgi_multithread = multithread self.wsgi_multiprocess = multiprocess ...

重點在ServerHandler的run函數:

class BaseHandler: def run(self, application): '''Invoke the application''' # Note to self: don’t move the close()! Asynchronous servers shouldn’t # call close() from finish_response(), so if you close() anywhere but # the double-error branch here, you’ll break asynchronous servers by # prematurely closing. Async servers must return from ’run()’ without # closing if there might still be output to iterate over. ... self.setup_environ() self.result = application(self.environ, self.start_response) self.finish_response() ...

關鍵的3個步驟:

setup_environ 繼續構建環境變量 接受application處理http請求的返回 完成http響應

setup_environ對env進行了進一步的包裝,附帶了請求的in/error,這樣讓使用env就可以對http請求進行讀寫。

def setup_environ(self): '''Set up the environment for one request''' env = self.environ = self.os_environ.copy() self.add_cgi_vars() # 子類實現 self.environ.update(self.base_env) env[’wsgi.input’] = self.get_stdin() # 注意沒有stdout env[’wsgi.errors’] = self.get_stderr() env[’wsgi.version’] = self.wsgi_version env[’wsgi.run_once’] = self.wsgi_run_once env[’wsgi.url_scheme’] = self.get_scheme() env[’wsgi.multithread’] = self.wsgi_multithread env[’wsgi.multiprocess’] = self.wsgi_multiprocess if self.wsgi_file_wrapper is not None: env[’wsgi.file_wrapper’] = self.wsgi_file_wrapper if self.origin_server and self.server_software: env.setdefault(’SERVER_SOFTWARE’,self.server_software)

env的處理過程,可以理解成3步:1)附加server的運行信息 2)附加請求的http頭(協議信息) 3)附加請求的流信息。env,可以換個說法就是http請求的所有上下文環境。

application還接收一個回調函數start_response,主要是按照http協議的規范,生成響應狀態和response_header:

def start_response(self, status, headers,exc_info=None): '''’start_response()’ callable as specified by PEP 3333''' self.status = status self.headers = self.headers_class(headers) status = self._convert_string_type(status, 'Status') assert len(status)>=4,'Status must be at least 4 characters' assert status[:3].isdigit(), 'Status message must begin w/3-digit code' assert status[3]==' ', 'Status message must have a space after code' return self.write

application對請求的處理:

def demo_app(environ,start_response): from io import StringIO stdout = StringIO() print('Hello world!', file=stdout) print(file=stdout) # http請求及環境 h = sorted(environ.items()) for k,v in h: print(k,’=’,repr(v), file=stdout) # 回調寫入http_status, response_headers start_response('200 OK', [(’Content-Type’,’text/plain; charset=utf-8’)]) # 返回處理結果response_body return [stdout.getvalue().encode('utf-8')]

響應仍然由ServerHandler寫入:

def finish_response(self): if not self.result_is_file() or not self.sendfile(): for data in self.result: self.write(data) self.finish_content()

可以使用下面命令測試這個流程:

python -m wsgiref.simple_serverServing HTTP on 0.0.0.0 port 8000 ...127.0.0.1 - - [31/Jan/2021 21:43:05] 'GET /xyz?abc HTTP/1.1' 200 3338wsgi 小結

簡單小結wsgi的實現。在http請求的處理流程web-browser <-> web-server <-> wsgi <-> web-application中,體現了分層的思想,每層做不同的事情:

web-server處理http/tcp協議,線程/進程的調度等底層實現 wsgi承上啟下,接受http請求,調用applicaiton處理請求,完成響應 application處理上層業務邏輯小技巧

在wsgiref代碼中一樣有各種小的技巧, 學習后可以讓我們的代碼更pythonic。

環境變量都這樣設置:

def setup_environ(self): # Set up base environment env = self.base_environ = {} env[’SERVER_NAME’] = self.server_name env[’GATEWAY_INTERFACE’] = ’CGI/1.1’ ...

我之前大概都是這樣寫:

def setup_environ(self): self.base_environ = {} self.base_environ[’SERVER_NAME’] = self.server_name self.base_environ[’GATEWAY_INTERFACE’] = ’CGI/1.1’

對比后,可以發現前面的寫法更簡潔一些。

比如流的持續寫入:

def _write(self,data): result = self.stdout.write(data) if result is None or result == len(data): return from warnings import warn warn('SimpleHandler.stdout.write() should not do partial writes', DeprecationWarning) while True: data = data[result:] # 持續的寫入,直到完成 if not data: break result = self.stdout.write(data)

比如header的處理,實際上是把數組當作字典使用:

class Headers: '''Manage a collection of HTTP response headers''' def __init__(self, headers=None): headers = headers if headers is not None else [] self._headers = headers # 內部存儲使用數組 def __setitem__(self, name, val): '''Set the value of a header.''' del self[name] self._headers.append( (self._convert_string_type(name), self._convert_string_type(val))) .... def __getitem__(self,name): '''Get the first header value for ’name’ Return None if the header is missing instead of raising an exception. Note that if the header appeared multiple times, the first exactly which occurrence gets returned is undefined. Use getall() to get all the values matching a header field name. ''' return self.get(name) def get(self,name,default=None): '''Get the first header value for ’name’, or return ’default’''' name = self._convert_string_type(name.lower()) for k,v in self._headers: if k.lower()==name: return v return default

這樣對 Content-Type: application/javascript; charset=utf-8 這樣的值,可以使用下面方式使用:

if self.headers.get(’content-type’) is None: env[’CONTENT_TYPE’] = self.headers.get_content_type()else: env[’CONTENT_TYPE’] = self.headers[’content-type’]

為什么用數組,而不是用字典呢?我猜測是因為header的特性是數據多為讀操作。

以上就是python wsgiref源碼解析的詳細內容,更多關于python wsgiref源碼的資料請關注好吧啦網其它相關文章!

標簽: Python 編程
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
日本 国产 欧美色综合| 在线观看免费一区二区| 亚洲专区在线| 精品网站999| 蜜桃视频欧美| 人人精品久久| 黄色在线网站噜噜噜| 日韩精品一区二区三区免费观看| 久久都是精品| 国产精品亚洲一区二区在线观看 | 欧美日韩激情| 欧美一区成人| 伊人久久大香线蕉av超碰演员| 国产日韩在线观看视频| 成人污污视频| 美女91精品| 欧美1区2区3| 蜜臀国产一区二区三区在线播放| 麻豆精品新av中文字幕| 久久亚洲国产| 超碰99在线| 六月婷婷一区| 性欧美videohd高精| 日韩在线观看中文字幕| 91精品一区国产高清在线gif| 四虎成人精品一区二区免费网站| 亚洲午夜天堂| 国产精品毛片aⅴ一区二区三区| 在线国产一区| 久久在线91| 黑丝一区二区三区| 国产成人精品福利| 国产精品巨作av| 中文字幕一区二区精品区| 伊人精品一区| 麻豆精品在线视频| 一区二区三区国产盗摄| 成人精品天堂一区二区三区| 国产不卡人人| 久久99精品久久久久久园产越南| 欧美中文高清| 综合激情五月婷婷| 蜜桃久久精品一区二区| 国产美女一区| 欧美+亚洲+精品+三区| 精品网站aaa| 日韩精品亚洲专区在线观看| 亚洲精品高潮| 尤物精品在线| 国产午夜久久| 亚洲性视频h| 红桃视频欧美| 欧美不卡高清| 最新日韩av| 亚洲精品一二三区区别| 国产综合婷婷| 国内亚洲精品| 在线一区av| 成人va天堂| 麻豆精品一区二区综合av| 久久精品99国产精品| 国产精品亚洲一区二区在线观看| 啪啪亚洲精品| 日韩超碰人人爽人人做人人添| 色综合视频一区二区三区日韩| 日本不卡一区二区三区| 日韩精品一区二区三区中文在线| 性色av一区二区怡红| 中文无码久久精品| 国产视频网站一区二区三区| 国产精品2023| 国产麻豆一区| 91一区二区三区四区| 人人草在线视频| 精品一区亚洲| 妖精视频成人观看www| 日本在线观看不卡视频| 国产日韩精品视频一区二区三区| 欧美日韩一区自拍| 欧美gv在线| 免费人成在线不卡| 国产精品3区| 免费av一区二区三区四区| 中文无码日韩欧| 国产成人免费av一区二区午夜| 国产精品精品国产一区二区| 亚洲大全视频| 天堂av在线一区| 日本va欧美va欧美va精品| 精品久久一区| 久久精品国内一区二区三区| 亚洲二区免费| 日本精品久久| а√在线中文在线新版| 日韩影院免费视频| 美女在线视频一区| 激情六月综合| 国产精品最新| 在线综合亚洲| 麻豆国产一区| 日韩精品成人在线观看| 国产韩日影视精品| 婷婷五月色综合香五月| 精品久久电影| 欧美天堂一区| 好吊日精品视频| 国产极品一区| 四虎国产精品免费久久| 激情欧美国产欧美| 色偷偷色偷偷色偷偷在线视频| 日韩中文欧美在线| 久久精品三级| 亚洲精品高潮| 香蕉久久夜色精品国产| 久久久久久自在自线| 国产精品麻豆成人av电影艾秋| 中文字幕亚洲影视| 美女少妇全过程你懂的久久| 日韩在线短视频| 精品三级久久久| 在线亚洲自拍| 麻豆精品久久| 久久av网站| 樱桃成人精品视频在线播放| 国产精品毛片一区二区在线看| 日本伊人久久| 综合亚洲色图| 日本亚洲三级在线| 99精品在线观看| 精品视频国内| 麻豆极品一区二区三区| 国产亚洲精aa在线看 | 五月天久久777| 少妇精品久久久一区二区三区| 麻豆精品蜜桃| sm久久捆绑调教精品一区| 国产欧美欧美| 国产精品xxx在线观看| 亚洲久久视频| 噜噜噜久久亚洲精品国产品小说| 亚洲激情另类| 亚洲制服少妇| 亚洲精选av| 日韩精彩视频在线观看| 亚洲精选成人| 91久久午夜| 好看的av在线不卡观看| 中文在线一区| 日韩欧美久久| 国产视频一区在线观看一区免费| 国产日韩专区| 视频精品一区| 国产精品**亚洲精品| 麻豆精品在线| 欧美福利一区| 美女亚洲一区| 日韩制服丝袜av| 精品国产91| 999国产精品永久免费视频app| 女主播福利一区| 日本欧美一区二区在线观看| 久久一区国产| 国产欧美一区二区精品久久久 | 亚洲欧美日韩综合国产aⅴ| 国产一区二区高清| 中文字幕一区二区三区在线视频| 国产亚洲一区二区三区啪| 国产美女精品视频免费播放软件| 美国三级日本三级久久99| 日韩激情一区二区| 国产精品久久亚洲不卡| 中文字幕成在线观看| 国产二区精品| 91成人在线| 成人影视亚洲图片在线| 制服诱惑一区二区| 国产亚洲欧美日韩精品一区二区三区 | 国产精品一国产精品| 激情综合网站| 日本免费在线视频不卡一不卡二| 国产成人免费| 国产欧美日本| 久久福利一区| 理论片午夜视频在线观看| 日韩高清三区| 蜜桃精品在线| 亚洲精品在线国产| 狠狠色狠狠色综合日日tαg| 青青伊人久久| 一区二区小说| 精品国产欧美日韩一区二区三区| 久久黄色影视| 亚洲激情av| 国产成人精品三级高清久久91| 亚洲性视频在线| 四虎4545www国产精品| 亚洲开心激情| 伊人成人在线视频| 亚洲欧美伊人| 成人影视亚洲图片在线|