python中WSGI是什么,Python应用WSGI详解


Posted in Python onNovember 24, 2017

为了让大家更好的对python中WSGI有更好的理解,我们先从最简单的认识WSGI着手,然后介绍一下WSGI几个经常使用到的接口,了解基本的用法和功能,最后,我们通过实例了解一下WSGI在实际项目中如何使用。

WSGI是什么?

wsgi是一个web组件的接口防范,wsgi将web组件分为三类:web服务器,web中间件,web应用程序

wsgi基本处理模式为:wsgi Server -> wsgi middleware -> wsgi application

WSGI,全称 Web Server Gateway Interface,或者 Python Web Server Gateway Interface ,是为 Python 语言定义的 Web 服务器和 Web 应用程序或框架之间的一种简单而通用的接口。自从 WSGI 被开发出来以后,许多其它语言中也出现了类似接口。

WSGI 的官方定义是,the Python Web Server Gateway Interface。从名字就可以看出来,这东西是一个Gateway,也就是网关。网关的作用就是在协议之间进行转换。

WSGI 是作为 Web 服务器与 Web 应用程序或应用框架之间的一种低级别的接口,以提升可移植 Web 应用开发的共同点。WSGI 是基于现存的 CGI 标准而设计的。

很多框架都自带了 WSGI server ,比如 Flask,webpy,Django、CherryPy等等。当然性能都不好,自带的 web server 更多的是测试用途,发布时则使用生产环境的 WSGI server或者是联合 nginx 做 uwsgi 。

也就是说,WSGI就像是一座桥梁,一边连着web服务器,另一边连着用户的应用。但是呢,这个桥的功能很弱,有时候还需要别的桥来帮忙才能进行处理。

WSGI的作用

WSGI有两方:“服务器”或“网关”一方,以及“应用程序”或“应用框架”一方。服务方调用应用方,提供环境信息,以及一个回调函数(提供给应用程序用来将消息头传递给服务器方),并接收Web内容作为返回值。

所谓的 WSGI中间件同时实现了API的两方,因此可以在WSGI服务和WSGI应用之间起调解作用:从WSGI服务器的角度来说,中间件扮演应用程序,而从应用程序的角度来说,中间件扮演服务器。“中间件”组件可以执行以下功能:

重写环境变量后,根据目标URL,将请求消息路由到不同的应用对象。

允许在一个进程中同时运行多个应用程序或应用框架。

负载均衡和远程处理,通过在网络上转发请求和响应消息。

进行内容后处理,例如应用XSLT样式表。

wsgi server:

理解为一个符合wsgi规范的web server,接收request请求,封装一系列环境变量,按照wsgi规范调用注册的wsgi app,最后将response返回给客户端。

工作流程:

1、服务器创建socket,监听port,等待client 连接

2、当请求过来时,server解析client msg放到环境变量environ中,并调用绑定的handler来处理

3、handler解析这个http请求,将请求消息例如method、path等放到environ中

4、wsgi handler再将一些server端消息也放到environ中,最后server msg,client msg,以及本次请求msg 全部都保存到了环境变量envrion中;

5、wsgi handler调用注册的wsgi app,并将envrion和回调函数传给wsgi app

6、wsgi app将reponse header/status/body回传给wsgi handler

7、handler 通过socket将response msg返回到client

WSGI Application

wsgi application就是一个普通的callable对象,当有请求到来时,wsgi server会调用这个wsgi app。这个对象接收两个参数,通常为environ,start_response。environ就像前面介绍的,可以理解为环境变量,

跟一次请求相关的所有信息都保存在了这个环境变量中,包括服务器信息,客户端信息,请求信息。start_response是一个callback函数,wsgi application通过调用start_response,将response headers/status 返回给wsgi server。此外这个wsgi app会return 一个iterator对象 ,这个iterator就是response body。

Dispatcher Middleware,用来实现URL 路由:(代码说明)

#!/usr/bin/python 
#encoding=utf-8

#利用wsgiref 作为wsgi server
from wsgiref.simple_server import make_server
"""
def simple_app(environ, start_response):
status = '200 ok'
response_headers = [('Content-type', 'text/plain')] #设置http头
start_response(status, response_headers)
return [u"test wsgi app".encode('utf-8')]

class AppClass(object):
def __call__(self, environ, start_response):
status = "200 ok"
response_headers = [('Content-type', 'text/plain')]
start_response(status, response_headers)
return [u"class AppClass".encode('utf-8')]
"""

#wsgi app只要是一个callable对象即可,不一定要是函数
#一个实现了__call__方法示例也ok的

#httpd = make_server('', 8080, simple_app)
"""
app = AppClass()
httpd = make_server('', 8080, app)
httpd.serve_forever()
"""
URL_PATTERNS = (
('AA/', 'AA_app'),
('BB/', 'BB_app'),
)

class Dispatcher(object):
#实现路由功能:
def _match(self, path):
path = path.split('/')[1]
for url, app in URL_PATTERNS:
if path in url:
return app

def __call__(self, environ, start_response):
path = environ.get('PATH_INFO', '/')
app = self._match(path)
if app:
app = globals()[app]
return app(environ, start_response)
else:
start_response("404 NOT FOUND",[('Content-type', 'text/plain')])
return ["page dose not exists"]

def AA_app(environ, start_response):
start_response("200 OK",[('Content-type', 'text/html')])
return ["AA page"]

def BB_app(environ, start_response):
start_response("200 OK",[('Content-type', 'text/html')])
 return ["BB page"]

app = Dispatcher()
httpd = make_server('', 8090, app)
httpd.serve_forever()

测试结果:
server端:
root@u163:~/cp163/python# python wsgi_app.py 
192.168.2.162 - - [04/Nov/2015 18:44:06] "GET /AA HTTP/1.1" 200 7
192.168.2.162 - - [04/Nov/2015 18:44:22] "GET /BB HTTP/1.1" 200 7

client端:
root@u162:~# curl http://192.168.2.163:8090/AA
AA page
root@u162:~# curl http://192.168.2.163:8090/BB
BB page
root@u162:~#

下面在给大家推荐一篇关机接口的详细介绍文章:深入解析Python中的WSGI接口

Python 相关文章推荐
Python continue语句用法实例
Mar 11 Python
Python入门篇之数字
Oct 20 Python
Python list操作用法总结
Nov 10 Python
用Python将IP地址在整型和字符串之间轻松转换
Mar 22 Python
深入理解Python单元测试unittest的使用示例
Nov 18 Python
python实现12306抢票及自动邮件发送提醒付款功能
Mar 08 Python
python实现文件的分割与合并
Aug 29 Python
python 调试冷知识(小结)
Nov 11 Python
Django后台管理系统的图文使用教学
Jan 20 Python
在keras中model.fit_generator()和model.fit()的区别说明
Jun 17 Python
详解在OpenCV中如何使用图像像素
Mar 03 Python
Python采集爬取京东商品信息和评论并存入MySQL
Apr 12 Python
python中print()函数的“,”与java中System.out.print()函数中的“+”功能详解
Nov 24 #Python
Python内置函数——__import__ 的使用方法
Nov 24 #Python
Django中login_required装饰器的深入介绍
Nov 24 #Python
Python多进程库multiprocessing中进程池Pool类的使用详解
Nov 24 #Python
pip安装Python库时遇到的问题及解决方法
Nov 23 #Python
python清理子进程机制剖析
Nov 23 #Python
Python3 加密(hashlib和hmac)模块的实现
Nov 23 #Python
You might like
神族 PROTOSS 概述
2020/03/14 星际争霸
php5.5中类级别的常量使用介绍
2013/10/02 PHP
浅谈PHP发送HTTP请求的几种方式
2017/07/25 PHP
Laravel 5.4因特殊字段太长导致migrations报错的解决
2017/10/22 PHP
让whoops帮我们告别ThinkPHP6的异常页面
2020/03/02 PHP
JS 动态加载脚本的4种方法
2009/05/05 Javascript
nodejs之请求路由概述
2014/07/05 NodeJs
js使用split函数按照多个字符对字符串进行分割的方法
2015/03/20 Javascript
JS实现带圆弧背景渐变效果的导航菜单代码
2015/10/13 Javascript
javascript中字体浮动效果的简单实例演示
2015/11/18 Javascript
js遍历json对象所有key及根据动态key获取值的方法(必看)
2017/03/09 Javascript
input框中自动展示当前日期yyyy/mm/dd的实现方法
2017/07/06 Javascript
详解利用 Express 托管静态文件的方法
2017/09/18 Javascript
js和jQuery以及easyui实现对下拉框的指定赋值方法
2018/01/23 jQuery
JavaScript实现的弹出遮罩层特效经典示例【基于jQuery】
2019/07/10 jQuery
在vue中使用防抖函数组件操作
2020/07/26 Javascript
vue axios请求成功却进入catch的原因分析
2020/09/08 Javascript
微信小程序实现转盘抽奖
2020/09/21 Javascript
Python查询Mysql时返回字典结构的代码
2012/06/18 Python
python列表与元组详解实例
2013/11/01 Python
Python中基本的日期时间处理的学习教程
2015/10/16 Python
Django学习教程之静态文件的调用详解
2018/05/08 Python
Django的ListView超详细用法(含分页paginate)
2020/05/21 Python
python实现快速文件格式批量转换的方法
2020/10/16 Python
奥地利网上书店:Weltbild
2017/07/14 全球购物
联想加拿大官方网站:Lenovo Canada
2018/04/05 全球购物
中海讯通笔试题
2015/09/15 面试题
linux面试题参考答案(10)
2013/11/04 面试题
教师专业理论水平的自我评价分享
2013/11/09 职场文书
物业管理计划书
2014/01/10 职场文书
高一生物教学反思
2014/01/17 职场文书
店长职务说明书
2014/02/04 职场文书
纪检干部对照检查材料
2014/08/22 职场文书
课外小组活动总结
2014/08/27 职场文书
违章停车检讨书
2014/10/21 职场文书
MySQL into_Mysql中replace与replace into用法案例详解
2021/09/14 MySQL