转载

WSGI & uwsgi

WSGI协议

首先弄清下面几个概念:

WSGI:全称是 Web Server Gateway InterfaceWSGI 不是服务器, python 模块,框架, API 或者任何软件,只是一种规范,描述 web server 如何与 web application 通信的规范。 serverapplication 的规范在 PEP 3333 中有具体描述。要实现WSGI协议,必须同时实现 web serverweb application ,当前运行在 WSGI 协议之上的 web 框架有 Bottle , Flask , Django

uwsgi:与 WSGI 一样是一种通信协议,是 uWSGI 服务器的独占协议,用于定义传输信息的类型( type of information ),每一个 uwsgi packet4byte 为传输信息类型的描述,与WSGI协议是两种东西,据说该协议是 fcgi 协议的10倍快。

uWSGI:是一个 web 服务器,实现了 WSGI 协议、 uwsgi 协议、 http 协议等。

WSGI 协议主要包括 serverapplication 两部分:

  • WSGI server 负责从客户端接收请求,将 request 转发给 application ,将 application 返回的 response 返回给客户端;
  • WSGI application 接收由 server 转发的 request ,处理请求,并将处理结果返回给 serverapplication 中可以包括多个栈式的中间件( middlewares ),这些中间件需要同时实现server与application,因此可以在WSGI服务器与WSGI应用之间起调节作用:对服务器来说,中间件扮演应用程序,对应用程序来说,中间件扮演服务器。

WSGI 协议其实是定义了一种 serverapplication 解耦的规范,即可以有多个实现 WSGI server 的服务器,也可以有多个实现 WSGI application 的框架,那么就可以选择任意的 serverapplication 组合实现自己的 web 应用。例如 uWSGIGunicorn 都是实现了 WSGI server 协议的服务器, DjangoFlask 是实现了 WSGI application 协议的 web 框架,可以根据项目实际情况搭配使用。

WSGI & uwsgi

DjangoFlask 框架都有自己实现的简单的 WSGI server ,一般用于服务器调试,生产环境下建议用其他 WSGI server

WSGI协议的实现

Django 为例,分析一下 WSGI 协议的具体实现过程。

django WSGI application

WSGI application 应该实现为一个可调用对象,例如函数、方法、类(包含 __call__ 方法)。需要接收两个参数:

  • 一个字典,该字典可以包含了客户端请求的信息以及其他信息,可以认为是请求上下文,一般叫做 environment (编码中多简写为 environenv
  • 一个用于发送 HTTP 响应状态( HTTP status )、响应头( HTTP headers )的回调函数

通过回调函数将响应状态和响应头返回给 server ,同时返回响应正文( response body ),响应正文是可迭代的、并包含了多个字符串。下面是 Djangoapplication 的具体实现部分:

class WSGIHandler(base.BaseHandler):
    initLock = Lock()
    request_class = WSGIRequest

    def __call__(self, environ, start_response):
        # 加载中间件
        if self._request_middleware is None:
            with self.initLock:
                try:
                    # Check that middleware is still uninitialized.
                    if self._request_middleware is None:
                        self.load_middleware()
                except:
                    # Unload whatever middleware we got
                    self._request_middleware = None
                    raise

        set_script_prefix(get_script_name(environ))
        # 请求处理之前发送信号
        signals.request_started.send(sender=self.__class__, environ=environ)
        try:
            request = self.request_class(environ)
        except UnicodeDecodeError:
            logger.warning('Bad Request (UnicodeDecodeError)',
                exc_info=sys.exc_info(),
                extra={'status_code': 400,})
            response = http.HttpResponseBadRequest()
        else:
            response = self.get_response(request)

        response._handler_class = self.__class__

        status = '%s %s' % (response.status_code, response.reason_phrase)
        response_headers = [(str(k), str(v)) for k, v in response.items()]
        for c in response.cookies.values():
            response_headers.append((str('Set-Cookie'), str(c.output(header=''))))
        # server提供的回调方法,将响应的header和status返回给server
        start_response(force_str(status), response_headers)
        if getattr(response, 'file_to_stream', None) is not None and environ.get('wsgi.file_wrapper'):
            response = environ['wsgi.file_wrapper'](response.file_to_stream)
        return response

可以看出 application 的流程包括:

  • 加载所有中间件,以及执行框架相关的操作,设置当前线程脚本前缀,发送请求开始信号;
  • 处理请求,调用 get_response() 方法处理当前请求,该方法的的主要逻辑是通过 urlconf 找到对应的 viewcallback ,按顺序执行各种 middlewarecallback
  • 调用由 server 传入的 start_response() 方法将响应 headerstatus 返回给 server
  • 返回响应正文

django WSGI Server

负责获取 http 请求,将请求传递给 WSGI application ,由 application 处理请求后返回 response 。以 Django 内建 server 为例看一下具体实现。 通过 runserver 运行 django 项目,在启动时都会调用下面的 run 方法,创建一个 WSGIServer 的实例,之后再调用其 serve_forever() 方法启动服务。

def run(addr, port, wsgi_handler, ipv6=False, threading=False):
    server_address = (addr, port)
    if threading:
        httpd_cls = type(str('WSGIServer'), (socketserver.ThreadingMixIn, WSGIServer), {})
    else:
        httpd_cls = WSGIServer
    # 这里的wsgi_handler就是WSGIApplication
    httpd = httpd_cls(server_address, WSGIRequestHandler, ipv6=ipv6)
    if threading:
        httpd.daemon_threads = True
    httpd.set_app(wsgi_handler)
    httpd.serve_forever()

下面表示 WSGI server 服务器处理流程中关键的类和方法。

WSGI & uwsgi
  • WSGIServer
    run() 方法会创建 WSGIServer 实例,主要作用是接收客户端请求,将请求传递给 application ,然后将 application 返回的 response 返回给客户端。
    • 创建实例时会指定 HTTP 请求的 handlerWSGIRequestHandler
    • 通过 set_appget_app 方法设置和获取 WSGIApplication 实例 wsgi_handler
    • 处理http请求时,调用 handler_request 方法,会创建 WSGIRequestHandler 实例处理http请求。
    • WSGIServerget_request 方法通过 socket 接受请求数据
  • WSGIRequestHandler
    • WSGIServer 在调用 handle_request 时创建实例,传入 requestcient_addressWSGIServer 三个参数, __init__ 方法在实例化同时还会调用自身的 handle 方法
    • handle 方法会创建 ServerHandler 实例,然后调用其 run 方法处理请求
  • ServerHandler
    • WSGIRequestHandler 在其 handle 方法中调用 run 方法,传入 self.server.get_app() 参数,获取 WSGIApplication ,然后调用实例( __call__ ),获取 response ,其中会传入 start_response 回调,用来处理返回的 headerstatus
    • 通过 application 获取 response 以后,通过 finish_response 返回 response
  • WSGIHandler
    • WSGI 协议中的 application ,接收两个参数, environ 字典包含了客户端请求的信息以及其他信息,可以认为是请求上下文, start_response 用于发送返回 statusheader 的回调函数

虽然上面一个 WSGI server 涉及到多个类实现以及相互引用,但其实原理还是调用 WSGIHandler ,传入请求参数以及回调方法 start_response() ,并将响应返回给客户端。

django simple_server

djangosimple_server.py 模块实现了一个简单的 HTTP 服务器,并给出了一个简单的 demo ,可以直接运行,运行结果会将请求中涉及到的环境变量在浏览器中展示出来。其中包括上述描述的整个 http 请求的所有组件:

ServerHandler , WSGIServer , WSGIRequestHandler ,以及 demo_app 表示的简易版的 WSGIApplication

可以看一下整个流程:

if __name__ == '__main__':
    # 通过make_server方法创建WSGIServer实例
    # 传入建议application,demo_app
    httpd = make_server('', 8000, demo_app)
    sa = httpd.socket.getsockname()
    print("Serving HTTP on", sa[0], "port", sa[1], "...")
    import webbrowser
    webbrowser.open('http://localhost:8000/xyz?abc')
    # 调用WSGIServer的handle_request方法处理http请求
    httpd.handle_request()  # serve one request, then exit
    httpd.server_close()
    
def make_server(
    host, port, app, server_class=WSGIServer, handler_class=WSGIRequestHandler
):
    """Create a new WSGI server listening on `host` and `port` for `app`"""
    server = server_class((host, port), handler_class)
    server.set_app(app)
    return server

# demo_app可调用对象,接受请求输出结果
def demo_app(environ,start_response):
    from io import StringIO
    stdout = StringIO()
    print("Hello world!", file=stdout)
    print(file=stdout)
    h = sorted(environ.items())
    for k,v in h:
        print(k,'=',repr(v), file=stdout)
    start_response("200 OK", [('Content-Type','text/plain; charset=utf-8')])
    return [stdout.getvalue().encode("utf-8")]

demo_app() 表示一个简单的WSGI application实现,通过 make_server() 方法创建一个 WSGIServer 实例,调用其 handle_request() 方法,该方法会调用 demo_app() 处理请求,并最终返回响应。

uWSGI

uWSGI 旨在为部署分布式集群的网络应用开发一套完整的解决方案。主要面向 web 及其标准服务。由于其可扩展性,能够被无限制的扩展用来支持更多平台和语言。 uWSGI 是一个 web 服务器,实现了 WSGI 协议, uwsgi 协议, http 协议等。 uWSGI 的主要特点是:

  • 超快的性能
  • 低内存占用
  • app 管理
  • 详尽的日志功能(可以用来分析 app 的性能和瓶颈)
  • 高度可定制(内存大小限制,服务一定次数后重启等)

uWSGI 服务器自己实现了基于 uwsgi 协议的 server 部分,我们只需要在 uwsgi 的配置文件中指定 application 的地址, uWSGI 就能直接和应用框架中的 WSGI application 通信。

参考阅读:

WSGI Tutorial

打造mvc框架之wsgi协议的优缺点及接口实现

Nginx和uWSGI通信机制

理解Python WSGI
原文  http://rainybowe.com/blog/2017/01/04/WSGI&uwsgi/index.html
正文到此结束
Loading...