深入解析Python中的WSGI接口

概述

WSGI接口包含两方面:server/gateway 及 application/framework。
server调用由application提供的可调用对象。
另外在server和application之间还可能有一种称作middleware的中间件。
可调用对象是指:函数、方法、类或者带有callable方法的实例。
关于application

函数、方法、类及带有callable方法的实例等可调用对象都可以作为the application object。
WSGI协议要求:
the application object接受两个参数且可以被多次调用

这两个参数分别为:
1.CGI式的字典;
2.回调函数:application用来向server传递http状态码/消息/http头

另外协议要求可调用对象必须将响应体封装成一个可迭代的strings返回。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
# the application object. 可以使用其他名字,
# 但是在使用mod_wsgi 时必须为 "application"
def application( environ, start_response):
# 函数接受两个参数:
# environ :包含有CGI 式环境变量的字典,由server负责提供内容
# start_response:由server提供的回调函数,其作用是将状态码和响应头返回给server
 
# 构造响应体,以可迭代字符串形式封装
   response_body = 'The request method was %s' % environ[ 'REQUEST_METHOD' ]
 
# HTTP 响应码及消息
   status = '200 OK'
 
# 提供给客户端的响应头.
# 封装成list of tuple pairs 的形式:
# 格式要求:[(Header name, Header value)].
   response_headers = [( 'Content-Type' , 'text/plain' ),
             ( 'Content-Length' , str ( len (response_body)))]
 
# 将响应码/消息及响应头通过传入的start_reponse回调函数返回给server
   start_response(status, response_headers)
 
# 响应体作为返回值返回
# 注意这里被封装到了list中.
   return [response_body]

关于server

从概述中可以知道,WSGI server必须要调用application,同时,从application的协议要求可知:
1. WSGI server必须向application提供环境参数,因此,自身也必须能够获取环境参数。
2. WSGI server接收application的返回值作为响应体。
最简单的WSGI server为Python自带的wsgiref.simple_server
示例如下:

?
1
2
3
from wsgiref.simple_server import make_server
srv = make_server( 'localhost' , 8080 , hello_world)
srv.serve_forever()

关于middleware

middleware的概念没有appllication和server那么容易理解。
假设一个符合application标准的可调用对象,它接受可调用对象作为参数,返回一个可调用对象的对象。
那么对于server来说,它是一个符合标准的可调用对象,因此是application。
而对于application来说,它可以调用application,因此是server。
这样的可调用对象称为middleware。

middleware的概念非常接近decorator。

以一个路由的例子示例:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
import re
 
# 这是一个标准的application object
def index(environ, start_response):
   start_response( '200 OK' , [( 'Content-Type' , 'text/html' )])
   return [ 'index page' ]
 
# 这是一个标准的application object
def hello(environ, start_response):
   start_response( '200 OK' , [( 'Content-Type' , 'text/html' )])
   return [ 'hello page' ]
 
# 这是一个标准的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)
]
# 这是一个middleware
# 根据不同的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 :
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值