设为首页 收藏本站
查看: 842|回复: 0

[经验分享] Python——eventlet.wsgi

[复制链接]

尚未签到

发表于 2015-11-30 09:27:57 | 显示全部楼层 |阅读模式
  eventlet 的 wsgi 模块提供了一种启动事件驱动的WSGI服务器的简洁手段,可以将其作为某个应用的嵌入web服务器,或作为成熟的web服务器,一个这样的web服务器的例子就是 Spawning。
  目录
  一、Eventlet 的 WSGI 服务器
  1. eventlet.wsgi.server()
  2. eventlet.wsgi.format_data_time()
  二、SSL
  三、Post hooks
  四、“100 continue”响应头
  
  一、Eventlet 的 WSGI server
  要启动一个 wsgi 服务器,只需要创建一个套接字,然后用它调用 eventlet.wsgi.server() 就可以。
    例如:



from eventlet import wsgi
import eventlet
def hello_world(env, start_response):
start_response('200 OK', [('Content-Type', 'text/plain')])
return ['Hello, World!\r\n']
wsgi.server(eventlet.listen(('', 8090)), hello_world)
  这个简单的 server 使用 eventlet.listen() 创建了一个套接字,wsgi.server() 监听对应的地址、端口等,将请求传递给 WSGI 应用 hello_world 处理。
  一个稍微形象一些的例子如下:



import eventlet
from eventlet import wsgi
def hello_world(env, start_response):
if env['PATH_INFO'] != '/':
start_response('404 Not Found', [('Content-Type', 'text/plain')])
return ['Not Found\r\n']
start_response('200 OK', [('Content-Type', 'text/plain')])
return ['Hello, World!\r\n']
wsgi.server(eventlet.listen(('', 8090)), hello_world)
  这个例子非常简洁地诠释了 WSGI 的应用接口规范,也涵盖了 eventlet.wsgi 模块中起 server 的用法。
  
  1.
  eventlet.wsgi.server()  
  



eventlet.wsgi.server(
socket,
site,
log=None,
environ=None,
max_size=None,
max_http_version='HTTP/1.1',
protocol=<class eventlet.wsgi.HttpProtocol at 0x7f4f68192f58>, server_event=None,
minimum_chunk_size=None,
log_x_forwarded_for=True,
custom_pool=None,
keepalive=True,
log_output=True,
log_format='%(client_ip)s - - [%(date_time)s] "%(request_line)s" %(status_code)s %(body_length)s %(wall_seconds).6f',
url_length_limit=8192,
debug=True,
socket_timeout=None,
capitalize_response_headers=True)
  这个调用封装了很多功能,创建一个 WSGI server 来处理指定套接字中产生的请求,这个函数会一直循环下去,即时刻监听请求。当 server 退出时,对应的套接字对象 socket 也会被关闭但是底层的文件描述字会被保留,所以这个时候如果用这个 socket 调用 dup() ,将会仍然可以使用。
  
  参数:

socketServer socket, 已经绑定一个端口且监听中
site实现 WSGI 应用的函数
  log

日志输出的位置,应该是一个类文件(file-like)对象,缺省为 sys.stderr
environ添加到每一次请求的 environ 字典中的额外参量
max_size这个 server 时刻允许的最大客户端连接数
max_http_version设为 “HTTP/1.0” 会只支持 HTTP 1.0. 可以支持那些在 HTTP 1.1 下并不能正常工作的应用
protocol弃用,协议类
server_event弃用,用来收集 Server 对象
minimum_chunk_size以字节记的HTTP块的最小尺寸。 可以用来改进哪些产生很多小字串的应用的性能。尽管使用它在技术上违背 WSGI 规范,但可以通过设置environ[‘eventlet.minimum_write_chunk_size’]来在每次请求的程度上覆写
log_x_forwarded_for如果为 True (缺省), 除了在日志的 ‘client_ip’ 段记录实际的客户端 ip 地址外,还记录HTTP头中的 x-forwarded-for 的内容
custom_pool一个定制的 GreenPool 实例,用来孵化客户端的 greenthreads. 如果设置了该项,无视参数 max_size
keepalive设为False 会禁止 server keepalives,所有的连接都会在服务完一个请求后关闭
log_outputBoolean 值,指示 server 是否记录日志
log_formatA python format string that is used as the template to generate log lines. The following values can be formatted into it: client_ip, date_time, request_line, status_code, body_length, wall_seconds. The default is a good example of how to use it
url_length_limit请求URL的最大长度,如果超长,返回414错误
debug如果想要服务器将异常追溯信息子啊500错误中发回给客户端,这里就设置为True。 如果这里设置为 False,server 将会响应空值
socket_timeout客户端连接的套接字操作超时限制,缺省为 None,意味着永久等待
capitalize_response_headers大写响应头的字段名称,缺省为True
  
  2.
  eventlet.wsgi.format_date_time(timestamp)
  将Unix时间戳格式化为一个 HTTP 标准字符串。
  
  二、SSL
  要创建一个安全的 server ,只需要传入一个 SSL 包裹的套接字即可:



wsgi.server(eventlet.wrap_ssl(
eventlet.listen(('', 8090)),
certfile='cert.crt',
keyfile='private.key',
server_side=True),
hello_world)
  应用可以通过环境变量  env['wsgi.url_scheme']  判断自己是不是在一个SSL server中。
  
  三、支持 Post Hooks的非标准扩展
  Eventlet 的 WSGI server 支持对于 WSGI 规范的非标准扩展——用 env['eventlet.posthooks'] 包含一个有若干post hooks 的数组,这些 post hooks 会在完全发送完响应后被调用。每一个 post hook 是一个格式为 (func, args, kwargs) 的元组,以 WSGI environment 字典加  args 和 kwargs 为参数调用 func。
  例如:



from eventlet import wsgi
import eventlet
def hook(env, arg1, arg2, kwarg3=None, kwarg4=None):
print('Hook called: %s %s %s %s %s' % (env, arg1, arg2, kwarg3, kwarg4))
def hello_world(env, start_response):
env['eventlet.posthooks'].append(
(hook, ('arg1', 'arg2'), {'kwarg3': 3, 'kwarg4': 4}))
start_response('200 OK', [('Content-Type', 'text/plain')])
return ['Hello, World!\r\n']
wsgi.server(eventlet.listen(('', 8090)), hello_world)
  上面的代码会为每一个处理的请求打印 WSGI 环境和其他的函数参数。
  当需要在完全响应一个客户端请求后(或客户端过早地断开连接后)执行一段代码时 Post hook 非常有用。 一个例子就是精确记录带宽的使用情况,因为用户断开连接时消耗的带宽小于 Content-Length 中标出的值。
  
  四、“100 Continue” 响应头
  Eventlet 的 WSGI server 支持发送(可选的)headers 和 HTTP “100 Continue” 临时响应. This is useful in such cases where a WSGI server expects to complete a PUT request as a single HTTP request/response pair, and also wants to communicate back to client as part of the same HTTP transaction. An example is where the HTTP server wants to pass hints back to the client about characteristics of data payload it can accept. As an example, an HTTP server may pass a hint in a header the accompanying “100 Continue” response to the client indicating it can or cannot accept encrypted data payloads, and thus client can make the encrypted vs unencrypted decision before starting to send the data).
  This works well for WSGI servers as the WSGI specification mandates HTTP expect/continue mechanism (PEP333).
  要定义 “100 Continue” 响应头, 需要调用 env['wsgi.input'] 里的 set_hundred_continue_response_header() 如下:



from eventlet import wsgi
import eventlet
def wsgi_app(env, start_response):
# Define "100 Continue" response headers
env['wsgi.input'].set_hundred_continue_response_headers(
[('Hundred-Continue-Header-1', 'H1'),
('Hundred-Continue-Header-k', 'Hk')])
# The following read() causes "100 Continue" response to
# the client.  Headers 'Hundred-Continue-Header-1' and
# 'Hundred-Continue-Header-K' are sent with the response
# following the "HTTP/1.1 100 Continue\r\n" status line
text = env['wsgi.input'].read()
start_response('200 OK', [('Content-Length', str(len(text)))])
return [text]

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-145167-1-1.html 上篇帖子: Python 闭包 下篇帖子: python解析Yahoo的XML格式的天气预报,获取当天和近期几天的天气:
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表