WSGI协议

首先弄清下面几个概念:
WSGI:全称是Web Server Gateway InterfaceWSGI不是服务器python模块,框架,API或者任何软件,只是一种规范,描述web server如何与web application通信的规范。serverapplication的规范在PEP 3333中有具体描述。要实现WSGI协议,必须同时实现web server和web application,当前运行在WSGI协议之上的web框架有BottleFlaskDjango
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.png-22.9kB

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的具体实现部分:

  1.  
    class WSGIHandler(base.BaseHandler):
  2.  
    initLock = Lock()
  3.  
    request_class = WSGIRequest
  4.  
     
  5.  
    def __call__(self, environ, start_response):
  6.  
    # 加载中间件
  7.  
    if self._request_middleware is None:
  8.  
    with self.initLock:
  9.  
    try:
  10.  
    # Check that middleware is still uninitialized.
  11.  
    if self._request_middleware is None:
  12.  
    self.load_middleware()
  13.  
    except:
  14.  
    # Unload whatever middleware we got
  15.  
    self._request_middleware = None
  16.  
    raise
  17.  
     
  18.  
    set_script_prefix(get_script_name(environ))
  19.  
    # 请求处理之前发送信号
  20.  
    signals.request_started.send(sender=self.__class__, environ=environ)
  21.  
    try:
  22.  
    request = self.request_class(environ)
  23.  
    except UnicodeDecodeError:
  24.  
    logger.warning('Bad Request (UnicodeDecodeError)',
  25.  
    exc_info=sys.exc_info(),
  26.  
    extra={'status_code': 400,})
  27.  
    response = http.HttpResponseBadRequest()
  28.  
    else:
  29.  
    response = self.get_response(request)
  30.  
     
  31.  
    response._handler_class = self.__class__
  32.  
     
  33.  
    status = '%s %s' % (response.status_code, response.reason_phrase)
  34.  
    response_headers = [(str(k), str(v)) for k, v in response.items()]
  35.  
    for c in response.cookies.values():
  36.  
    response_headers.append((str('Set-Cookie'), str(c.output(header=''))))
  37.  
    # server提供的回调方法,将响应的header和status返回给server
  38.  
    start_response(force_str(status), response_headers)
  39.  
    if getattr(response, 'file_to_stream', None) is not None and environ.get('wsgi.file_wrapper'):
  40.  
    response = environ['wsgi.file_wrapper'](response.file_to_stream)
  41.  
    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()方法启动服务。

  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服务器处理流程中关键的类和方法。

xiong (2).png-93.7kB
  • 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请求的所有组件:
ServerHandlerWSGIServerWSGIRequestHandler,以及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.  
     
  13.  
    def make_server(
  14.  
    host, port, app, server_class=WSGIServer, handler_class=WSGIRequestHandler
  15.  
    ):
  16.  
    """Create a new WSGI server listening on `host` and `port` for `app`"""
  17.  
    server = server_class((host, port), handler_class)
  18.  
    server.set_app(app)
  19.  
    return server
  20.  
     
  21.  
    # demo_app可调用对象,接受请求输出结果
  22.  
    def demo_app(environ,start_response):
  23.  
    from io import StringIO
  24.  
    stdout = StringIO()
  25.  
    print("Hello world!", file=stdout)
  26.  
    print(file=stdout)
  27.  
    h = sorted(environ.items())
  28.  
    for k,v in h:
  29.  
    print(k,'=',repr(v), file=stdout)
  30.  
    start_response("200 OK", [('Content-Type','text/plain; charset=utf-8')])
  31.  
    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 & uwsgi 
WSGI Tutorial
打造mvc框架之wsgi协议的优缺点及接口实现
Nginx和uWSGI通信机制
理解Python WSGI

wsgi&nginx-理解的更多相关文章

  1. 说说我对 WSGI 的理解

    先说下 WSGI 的表面意思,Web Server Gateway Interface 的缩写,即 Web 服务器网关接口. 之前不知道 WSGI 意思的伙伴,看了上面的解释后,我估计也还是不清楚,所 ...

  2. Flask + WSGI + Nginx 云部署

    这几天学着用flask写一些rest api,然后部署到云上.这个过程虽然网上有很多的教程,但还是遇到不少的问题! 采用flask的原因是因为它比较容易上手吧.用flask有专门restful api ...

  3. Linux 集群概念 , wsgi , Nginx负载均衡实验 , 部署CRM(Django+uwsgi+nginx), 部署学城项目(vue+uwsgi+nginx)

    Linux 集群概念 , wsgi , Nginx负载均衡实验 , 部署CRM(Django+uwsgi+nginx), 部署学城项目(vue+uwsgi+nginx) 一丶集群和Nginx反向代理 ...

  4. WSGI的理解 perfect

    https://blog.csdn.net/hzrandd/article/details/10099871 https://blog.csdn.net/cloudxli/article/detail ...

  5. django wsgi nginx 配置

    """ WSGI config for HelloWorld project. It exposes the WSGI callable as a module-leve ...

  6. 对于python WSGI的理解

    首先看看WSGI的目的是什么? 是用来定义一个统一的接口. 这个接口是针对Web服务器和python Web应用之间的. 以增加Python web应用在不同Web 服务器之间的可移植性. 也就是说如 ...

  7. WSGI的理解

    Python web开发中,服务端程序可分为2个部分: 服务器程序(用来接收.整理客户端发送的请求) 应用程序(处理服务器程序传递过来的请求) 在开发应用程序的时候,我们会把常用的功能封装起来,成为各 ...

  8. 阿里云部署 Flask + WSGI + Nginx 详解

    抵不住朋友的诱惑,今天终于入手了一台阿里云服务器,是Ubuntu 1.4 32位版本,最初考虑是用来尝尝鲜只是买了个最低配的,价格算起来与在国外买个空间的价格相当吧(可能一年才贵100多),但用起来感 ...

  9. 关于Nginx理解

    由于微信小程序要使用Https,但是又不能修改已有线上的配置.所以最简单的方法就是使用nginx转发,在nginx上使用https,然后再转发到内部服务器.Nginx由于其优良的性能.一台4核16GB ...

  10. nginx理解与配置

    准备: http服务器:①tomcat②apache③nginx(c语言开发) 文件系统:①mgfs ②mgbd:存储小文件 ③fastDFS:存储大文件.小文件,分布式文件系统 nginx是一种ht ...

随机推荐

  1. mysql存储引擎的对比

  2. ERROR: java.lang.NullPointerException的一种情况

    java.lang.NullPointerException错误,错误原因就是以下六条没配置完: 1.JAVA环境配置正确.2.源码里面的包没有与tomcat的包冲突.3.把数据库文件给导入到了SQL ...

  3. 移动端自动化测试-Mac-IOS-Appium环境搭建

    第一步 安装JDK,本机如果带有1.7及以上版本的,则可忽略此安装步骤. 百度下载JDK,并配置环境变量 vim ~/.bash_profile 检查是否安装成功 java -version 第二步 ...

  4. arp欺骗图解

    ARP协议:地址转换协议,工作在OSI模型的数据链路层,在以太网中,网络设备之间互相通信是用MAC地址而不是IP地址,ARP协议就是用来把IP地址转换为MAC地址的. 防止ARP攻击的方法: 1.使用 ...

  5. centos7-jdk快速安装

    安装之前先检查一下系统有没有自带open-jdk 命令: rpm -qa |grep java rpm -qa |grep jdk rpm -qa |grep gcj 如果没有输入信息表示没有安装. ...

  6. 【MySQL】【1】表中存在重复记录,删除保留其中一条

    --删除题库(TABLE_Q )中,标题(TITLE )和类型(TYPE )都相同的数据,仅保留ID最小的一条 DELETE TABLE_Q FROM TABLE_Q, ( ) T2 WHERE TA ...

  7. ALV打印模板(存代码)

    *&---------------------------------------------------------------------* *& Report ZMMF013 * ...

  8. Ubuntu 14.04(64位)+GTX970+CUDA8.0+Tensorflow配置 (双显卡NVIDIA+Intel集成显卡) ------本内容是长时间的积累,有时间再详细整理

    (后面内容是本人初次玩GPU时,遇到很多坑的问题总结及尝试解决办法.由于买独立的GPU安装会涉及到设备的兼容问题,这里建议还是购买GPU一体机(比如https://item.jd.com/396477 ...

  9. 第一讲(3)osgearth编译

    前题条件完成osg 3.0的编译Step 1 下载osgEarth 2.1.1https://github.com/gwaldron/osgearth/downloads------------> ...

  10. VisualSVN+TortoiseSVN搭建版本控制系统教程

    Tortoise VisualSVN用作SVN的服务端,TortoiseSVN用作SVN的客户端. 一.安装和配置VisualSVN 1.1安装VisualSVN 下载链接:https://www.v ...