通常服务器程序分为web服务器和应用程序服务器。web服务器是用于处理HTML文件,让客户可以通过浏览器进行访问,主流的web服务器有Apache、IIS、Nginx、lighthttpd等。应用服务器处理业务逻辑,比如使用Python的Django、flask写的程序。通常来自客户端浏览器的请求被web服务器截获,如果是静态请求,则如Nginx会自己做处理,如果是动态请求,则会抛给后端应用服务器来处理。于是如何在web服务器与应用服务器之间进行通信成了主要问题,这就引出了以下三种处理的接口:CGI、FastCGI、WSGI。

CGI

通用网关接口(Comman Gateway Interface)描述了客户端和服务器程序之间传输数据的一种标准,可以让一个客户端,从网页浏览器向执行在网络服务器上的应用程序请求数据。CGI独立于任何语言,CGI程序可以是任何脚本语言或完全独立编程语言实现,只要这个语言可以在这个系统上运行。Unix shell、Python、Ruby、PHP、Perl、C/C++和VB都可以用来编写CGI程序。最初CGI是在1993年由美国国家超级电脑应用中心为NCSA HTTPd web服务器开发的。这个web服务器使用了Unix shell环境变量来保存从web服务器传递出去的参数,然后生成一个运行CGI的独立的进程。CGI的处理流程如下图所示:

  • web服务器收到客户端(浏览器)的请求http request,启动CGI程序,并通过环境变量、标准输入传递数据;
  • CGI进程启动解析器、加载配置(如业务相关配置)、连接其它服务器(如数据库服务器)、逻辑处理等;
  • CGI进程将处理结果通过标准输出、标准错误,传递给web服务器;
  • web服务器收到CGI返回的结果,构建http response返回给客户端,并杀死CGI进程,web服务器与CGI通过环境变量、标准输入、标准输出、标准错误传递数据;

CGI使外部程序与web服务器之间交互成为可能。CGI程序运行在独立的进程中,并对每个web请求建立一个进程,这种方法非常容易实现,但效率很差,难以扩展。面对大量请求,进程的大量建立和消亡使操作系统性能大大下降,此外,由于地址空间无法共享,业限制了资源重用。

FastCGI

快速通用网关接口(Fast Comman Gateway Interface)是通用网关接口(CGI)的改进,描述了客户端和服务器程序之间传输数据的一种标准。FastCGI致力于减少web服务器与CGI程序之间互动的开销,从而使服务器可以同时处理更多的web请求。与为买个请求创建一个新的进程不同,FastCGI使用持续的进程来处理一连串的请求。这些进程由FastCGI进程管理器管理,而不是web服务器。

当进来一个请求时,web服务器把环境变量和这个页面请求通过Unix domain socket(都位于同一物理服务器)或者一个IP socket(FastCGI部署在其他物理服务器)传递给FastCGI进程。

  • web服务器启动时载入初始化FastCGI执行环境,例如IIS ISAPI,Apache mod_fastcgi、nginx_ngx_http_fastcgi_module、lighthttpd mode_fastcgi;
  • FastCGI进程管理器自身初始化,启动多个CGI解释器进程并等待来自web服务器的连接。启动FastCGI进程时,可以配置ip socket或Unix domain socket两种启动方式;
  • 当客户端请求到达web服务器时,web服务器将请求采用socket方式发到FastCGI主进程,FastCGI主进程选择并连接到一个CGI解释器。web服务器将CGI环境变量和标准输入发送到FastCGI子进程;
  • FastCGI子进程完成处理后将标准输出和错误信息从同一socket连接返回web服务器。当FastCGI子进程关闭连接时,请求便处理完成;
  • FastCGI子进程接着等待并处理来自web服务器的下一个连接;

由于FastCGI程序不需要不断地产生新进程,可以大大降低服务器的压力并且产生较高的应用效率。它的速度效率至少要比CGI技术提高5倍以上。它还支持分布式部署,即FastCGI程序可以在web服务器以外的主机上运行。
CGI就是所谓的短生存期应用程序,FastCGI就是所谓的长生存期应用程序。FastCGI像是一个常驻(long-live)型的CGI,它可以一直执行着,不会每次都要花费时间去fork(这是CGI最为人诟病的fork-and-execute模式)。

WSGI

web服务器网关接口(Python Web Server Gateway Interface)是为Python语言定制的web服务器和web应用程序或框架之间的一种简单通用的接口。自从WSGI被开发出来之后,许多其他语言中也出现了类似接口。WSGI是作为web服务器与web应用程序或应用框架之间的一种低级别的接口,以提升可移植web应用开发的共同点。WSGI是基于现存的CGI标准而设计的。
WSGI分为两个部分:一为"服务器"或"网关",另一为"应用程序"或"应用框架"。在处理一个WSGI请求时,服务器会为应用程序提供环境资讯及一个回调函数(callback function)。当应用程序完成处理请求后,透过前述的回调函数,将结果回传给服务器。所谓的WSGI中间件同时实现了API的两方,因此可以在WSGI服务和WSGI应用之间起调解作用:从WSGI服务器的角度来说,中间件扮演应用程序,而从应用程序角度来说,中间件扮演服务器。"中间件"组件可以执行以下功能:

  • 重写环境变量,根据目标URL,将请求消息路由到不同的应用对象;
  • 允许在一个进程中同时运行多个应用程序或应用框架;
  • 负载均衡和远程处理,通过在网络上转发请求和响应消息;
  • 进行内容后处理,例如应用XSLT样式表;

以前,如何选择合适的web应用程序框架成为困扰Python初学者的一个问题,这是因为,一般而言,web应用框架的选择将限制可用的web服务器的选择,反之亦然。那时的Python应用程序通常为CGI、FastCGI、mod_python中的一个设计,甚至是为特定web服务器的自定义的API接口而设计的。WSGI没有官方的实现,因为WSGI更像一个协议。只要遵循这些协议,WSGI应用(application)都可以在任何服务器(server)上运行,反之亦然。WSGI就是Python的CGI包装,相对于FastCGI是PHP的CGI包装。
WSGI将web组件分为三类:web服务器、web中间件、web应用程序,WSGI基本处理模式为:

1、wsgi server/gateway
wsgi server可以理解为一个符合WSGI规范的web server,接收request请求,封装一系列环境变量,按照wsgi规范调用注册的wsgi app,最后将response返回给客户端。文字很难理解清楚wsgi server到底是什么东西,以及做些什么事情,最直观的方式还是看wsgi server的实现代码。以Python自带的wsgiref为例,wsgiref是按照wsgi规范实现的一个简单wsgi server。

  • 服务器创建socket,监听端口,等待客户端连接;
  • 当有请求来时,服务器解析客户端信息放到环境变量environ中,并调用绑定的handler来处理请求;
  • handler解析这个http请求,将请求信息如method、path等放到environ中;
  • wsgi handler再将一些服务器信息也放到environ中,最后服务器信息,客户端信息,本次请求信息全部都保存到了环境变量environ中;
  • wsgi handler调用注册的wsgi app,并将environ和回调函数传给wsgi app;
  • wsgi app将response header/status/body回传给wsgi handler;
  • 最终handler还是通过socket将response信息返回给客户端;

2、wsgi application
wsgi application就是一个普通的callback对象,当有请求到来时,wsgi server会调用这个wsgi app。这个对象接收两个参数,通常为environ,start_response。environ就是前面介绍的,可以理解为环境变量,跟一次请求相关的所有信息都保存在了这个环境变量中,包括服务器信息,客户端信息,请求信息。start_response是一个callback函数,wsgi application通过start_response,将response header/status返回给wsgi server。此外这个wsgi app会返回一个迭代器对象,这个迭代器对象就是response body。

3、wsgi middleware
有些功能可能介于服务器程序和应用程序之间,例如,服务器拿到了客户端请求的URL,不同的URL需要交由不同的函数处理,这个功能叫URL路由,这个功能就可以放在二者中间实现,这个中间层就是middleware。middleware对服务器程序和应用都是透明的,也就是说,服务器程序以为它就是应用程序,而应用程序以为它就是服务器。这就告诉我们,middleware需要把自己伪装成一个服务器,接收应用程序,调用它,同时middleware还需要把自己伪装成一个应用程序,传给服务器程序。其实无论是服务器程序,middleware还是应用程序,都在服务端,为客户提供服务,之所以把它们抽象成不同层,就是为了控制复杂度,使得每一次都不太复杂,各司其职。

uwsgi

uwsgi旨在为部署分布式集群的网络应用开发一套完整的解决方案。uwsgi主要面向web及其标准服务,已经成功的应用于多种不同的语言。由于uwsgi的可扩展架构,它能够被无限制的扩展用来支持更多的平台和语言。目前可以使用C,C++和Object-C来编写插件。项目名称中的wsgi是为了向同名的Python web标准表示感谢,因为wsgi为该项目开发了第一个插件。uwsgi是一个web服务器,它实现了wsgi协议、uwsgi协议、http等协议。uwsgi既不用wsgi协议也不用FastCGI协议,而是自创了一个uwsgi协议,uwsgi协议是一个uWSGI服务器自由的协议,它用于定义传输信息的类型,每一个uwsgi packet前4字节为传输信息类型描述,它与WSGI相比是两样东西,据说该协议大约是fcgi协议的10倍快。uWSGI主要特点如下:

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

网关协议:CGI和WSGI的更多相关文章

  1. openstack项目【day23】:keystone组件网关协议

    本节内容 一 静态页面和动态页面 二 什么是web server 三 什么是网关协议 3.1 引子 3.2 网关协议 四 网关协议CGI.FastCGI.WSGI.UWSGI 五 网关协议与keyst ...

  2. keystone系列三:网关协议

    一 静态页面和动态页面 在了解了http协议后,我们知晓,一个web server的本质就是 浏览器发送一个HTTP请求: 服务器收到请求,生成一个HTML文档: 服务器把HTML文档作为HTTP响应 ...

  3. openstack 之~keystone之网关协议

    第一:静态页面和动态页面 上一篇博客介绍了HTTP后,我们知道一个web server的本质就是 浏览器发送一个HTTP请求: 服务器收到请求,生成一个HTML文档: 服务器把HTML文档作为HTTP ...

  4. 网关协议学习:CGI、FastCGI、WSGI、uWSGI

    一直对这四者的概念和区别很模糊,现在就特意梳理一下它们的关系与区别. CGI CGI即通用网关接口(Common Gateway Interface),是外部应用程序(CGI程序)与Web服务器之间的 ...

  5. 网关协议学习:CGI、FastCGI、WSGI

    网关协议学习:CGI.FastCGI.WSGI https://www.biaodianfu.com/cgi-fastcgi-wsgi.html

  6. 网关协议:CGI、FastCGI、WSGI

    CGI就像是一座桥,把网页和WEB服务器中的执行程序连接起来,它把HTML接收的指令传递给服务器的执行程序,再把服务器执行程序的结果返还给HTML页. CGI CGI即通用网关接口(Common Ga ...

  7. 【转】网关协议学习:CGI、FastCGI、WSGI、uWSGI

    一直对这四者的概念和区别很模糊,现在就特意梳理一下它们的关系与区别. CGI CGI即通用网关接口(Common Gateway Interface),是外部应用程序(CGI程序)与Web服务器之间的 ...

  8. OpenStack☞网关协议

    一 动态页面与静态页面区别 静态页面:每一个网页都有一个固定的URL,且网页的URL以.html..htm..shtml等常见的形式为后缀. 网页内容已经发布到网站服务器上,无论是否有用户访问,每个静 ...

  9. 漫谈CGI FastCGI WSGI

    作者:auxten链接:https://zhuanlan.zhihu.com/p/20054757来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注明出处. CGI(Common ...

随机推荐

  1. 重磅发布!阿里云推PostgreSQL 10 高可用版

    摘要: 近日,阿里云重磅发布PostgreSQL 10 高可用本地SSD盘版,相比原 9.4 版本又新增了JSONB.BRIN索引.GROUPING SETS/CUBE/ROLLUP.UPSERT等多 ...

  2. PHP chroot() 函数

    改变根目录: <?php// Change root directorychroot("/path/to/chroot/"); // Get current director ...

  3. GPIO软件模拟IIC时序

    一.MPU6050中的IIC时序 1.1 START和STOP SDA和SCL在高电平时,SDA拉低表示START.SCL拉低,表示可以传输数据. SDA和SCL在低电平时,SDA拉高表示STOP. ...

  4. paper 144:人生苦短,快用Python

    1.Python 语言特点 Python是一种面向对象.直译式计算机程序设计语言,这种语言的语法简捷而清晰,具有丰富和强大的类库,基本上能胜任你平时需要的编程工作. Python的优点: (1)编写的 ...

  5. getstu

    #coding: utf- #title..href... import urllib.request import time url=[ page= j= : url[j]='http://www. ...

  6. Sublime text 3 3103 注册码(2016.2.9更新)

    Sublime text 3 (Build 3103) license key,these all tested available on 2016/02/10 .Feel free to enjoy ...

  7. C++——运行时类型识别RTTI

    1.实现方式 typeid运算符,返回表达式的类型 dynamic_cast运算符,基类的指针或引用安全地转换成派生类的指针或引用 2.适用于:使用基类的指针或引用执行派生类的操作,且该操作不是虚函数 ...

  8. Rust <5>:测试

    测试运行顺序:单元测试(同处于源文件中,以 #[cfg(tests)] 标记 mod,以 #[test] 标记 function).集成测试(位于项目根路径下的 tests 目录下,不需要 #[cfg ...

  9. Scrapy框架: 第一个程序

    首先创建项目: scrappy start project maitian 第二步: 明确要抓取的字段items.py import scrapy class MaitianItem(scrapy.I ...

  10. mysql中BLACKHOOL的作用

    MySQL在5.x系列提供了Blackhole引擎–"黑洞". 其作用正如其名字一样:任何写入到此引擎的数据均会被丢弃掉, 不做实际存储:Select语句的内容永远是空. 和Lin ...