python之uWSGI和WSGI

Posted zlixing

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了python之uWSGI和WSGI相关的知识,希望对你有一定的参考价值。

WSGI协议

首先弄清下面几个概念:
WSGI:全称是​​​Web Server Gateway Interface​​​,​​WSGI​​​不是服务器,​​python​​​模块,框架,​​API​​​或者任何软件,只是一种规范,描述​​web server​​​如何与​​web application​​​通信的规范。​​server​​​和​​application​​​的规范在​​PEP 3333​​​中有具体描述。要实现WSGI协议,必须同时实现web server和web application,当前运行在​​WSGI​​​协议之上的​​web​​​框架有​​Bottle​​​, ​​Flask​​​, ​​Django​​​。
uwsgi:与​​​WSGI​​​一样是一种通信协议,是​​uWSGI​​​服务器的独占协议,用于定义传输信息的类型(​​type of information​​​),每一个​​uwsgi packet​​​前​​4byte​​​为传输信息类型的描述,与WSGI协议是两种东西,据说该协议是​​fcgi​​​协议的10倍快。
uWSGI:是一个​​​web​​​服务器,实现了​​WSGI​​​协议、​​uwsgi​​​协议、​​http​​协议等。

​WSGI​​​协议主要包括​​server​​​和​​application​​两部分:

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

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

python之uWSGI和WSGI_uWSGIwsgi.png-22.9kB


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

WSGI协议的实现

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

django WSGI application

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

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

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

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

  4. def __call__(self, environ, start_response):
  5. # 加载中间件
  6. if self._request_middleware is None:
  7. with self.initLock:
  8. try:
  9. # Check that middleware is still uninitialized.
  10. if self._request_middleware is None:
  11. self.load_middleware()
  12. except:
  13. # Unload whatever middleware we got
  14. self._request_middleware = None
  15. raise

  16. set_script_prefix(get_script_name(environ))
  17. # 请求处理之前发送信号
  18. signals.request_started.send(sender=self.__class__, environ=environ)
  19. try:
  20. request = self.request_class(environ)
  21. except UnicodeDecodeError:
  22. logger.warning(Bad Request (UnicodeDecodeError),
  23. exc_info=sys.exc_info(),
  24. extra=status_code: 400,)
  25. response = http.HttpResponseBadRequest()
  26. else:
  27. response = self.get_response(request)

  28. response._handler_class = self.__class__

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

 

可以看出​​application​​的流程包括:

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

django WSGI Server

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

  1. def run(addr, port, wsgi_handler, ipv6=False, threading=False):
  2. server_address = (addr, port)
  3. if threading:
  4. httpd_cls = type(str(WSGIServer), (socketserver.ThreadingMixIn, WSGIServer), )
  5. else:
  6. httpd_cls = WSGIServer
  7. # 这里的wsgi_handler就是WSGIApplication
  8. httpd = httpd_cls(server_address, WSGIRequestHandler, ipv6=ipv6)
  9. if threading:
  10. httpd.daemon_threads = True
  11. httpd.set_app(wsgi_handler)
  12. httpd.serve_forever()

 

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

python之uWSGI和WSGI_uWSGI_02xiong (2).png-93.7kB


 

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

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

django simple_server

​django​​​的​​simple_server.py​​​模块实现了一个简单的​​HTTP​​​服务器,并给出了一个简单的​​demo​​​,可以直接运行,运行结果会将请求中涉及到的环境变量在浏览器中展示出来。
其中包括上述描述的整个​​​http​​​请求的所有组件:
​​​ServerHandler​​​, ​​WSGIServer​​​, ​​WSGIRequestHandler​​​,以及​​demo_app​​​表示的简易版的​​WSGIApplication​​​。
可以看一下整个流程:

  1. if __name__ == __main__:
  2. # 通过make_server方法创建WSGIServer实例
  3. # 传入建议application,demo_app
  4. httpd = make_server(, 8000, demo_app)
  5. sa = httpd.socket.getsockname()
  6. print("Serving HTTP on", sa[0], "port", sa[1], "...")
  7. import webbrowser
  8. webbrowser.open(http://localhost:8000/xyz?abc)
  9. # 调用WSGIServer的handle_request方法处理http请求
  10. httpd.handle_request() # serve one request, then exit
  11. httpd.server_close()

  12. def make_server(
  13. host, port, app, server_class=WSGIServer, handler_class=WSGIRequestHandler
  14. ):
  15. """Create a new WSGI server listening on `host` and `port` for `app`"""
  16. server = server_class((host, port), handler_class)
  17. server.set_app(app)
  18. return server

  19. # demo_app可调用对象,接受请求输出结果
  20. def demo_app(environ,start_response):
  21. from io import StringIO
  22. stdout = StringIO()
  23. print("Hello world!", file=stdout)
  24. print(file=stdout)
  25. h = sorted(environ.items())
  26. for k,v in h:
  27. print(k,=,repr(v), file=stdout)
  28. start_response("200 OK", [(Content-Type,text/plain; charset=utf-8)])
  29. 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​​通信。

 

以上是关于python之uWSGI和WSGI的主要内容,如果未能解决你的问题,请参考以下文章

uWSGI ,WSGI和uwsgi的区别

Python Web开发 - WSGI & uWSGI协议

uwsgi + nginx 部署python项目

uwsgi模块

uwsgi和wsgi

nginx,wsgi,uwsgi区别