Python——eventlet.wsgi

  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() ,将会仍然可以使用。

 

  参数:

socket Server 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_output Boolean 值,指示 server 是否记录日志
log_format A 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]
posted @ 2014-12-17 12:27  王智愚  阅读(4310)  评论(0编辑  收藏  举报