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获取文件版本信息、公司名和产品名的方法
Oct 05 Python
浅谈django开发者模式中的autoreload是如何实现的
Aug 18 Python
python实现微信发送邮件关闭电脑功能
Feb 22 Python
python爬取淘宝商品详情页数据
Feb 23 Python
使用python对文件中的单词进行提取的方法示例
Dec 21 Python
用Python实现大文本文件切割的方法
Jan 12 Python
python的pytest框架之命令行参数详解(上)
Jun 27 Python
python关闭占用端口方式
Dec 17 Python
Python字典fromkeys()方法使用代码实例
Jul 20 Python
Python过滤序列元素的方法
Jul 31 Python
如何实现一个python函数装饰器(Decorator)
Oct 12 Python
Python进行特征提取的示例代码
Oct 15 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
介绍一些PHP判断变量的函数
2012/04/24 PHP
分享一下贝贝成长进度的php代码
2012/09/14 PHP
php截取字符串函数分享
2015/02/02 PHP
Symfony2学习笔记之模板用法详解
2016/03/17 PHP
PHP实现将几张照片拼接到一起的合成图片功能【便于整体打印输出】
2017/11/14 PHP
探索Emberjs制作一个简单的Todo应用
2012/11/07 Javascript
原生JS实现表单checkbook获取已选择的值
2013/07/21 Javascript
jquery cookie的用法总结
2013/11/18 Javascript
深入分析node.js的异步API和其局限性
2016/09/05 Javascript
AngularJS 面试题集锦
2016/09/06 Javascript
javaScript+turn.js实现图书翻页效果实例代码
2017/02/16 Javascript
Vuejs 页面的区域化与组件封装的实现
2017/09/11 Javascript
node.js学习之断言assert的使用示例
2017/09/28 Javascript
JavaScript实现星级评价效果
2019/05/17 Javascript
原生js实现无缝轮播图效果
2021/01/28 Javascript
[01:09]模型精美,特效酷炫!TI9不朽宝藏Ⅰ鉴赏
2019/05/10 DOTA
Python对象的深拷贝和浅拷贝详解
2014/08/25 Python
解决pycharm无法调用pip安装的包问题
2018/05/18 Python
Python实现的远程登录windows系统功能示例
2018/06/21 Python
解决python打不开文件(文件不存在)的问题
2019/02/18 Python
详解利用python+opencv识别图片中的圆形(霍夫变换)
2019/07/01 Python
python关于变量名的基础知识点
2020/03/03 Python
python实现猜拳游戏
2020/03/04 Python
详解Python中string模块除去Str还剩下什么
2020/11/30 Python
马来西亚太阳镜、眼镜和隐形眼镜网上商店:Focus Point
2018/12/13 全球购物
德国受欢迎的旅游和休闲网站:lastminute.de
2019/09/23 全球购物
开办加工厂创业计划书
2014/01/03 职场文书
仓库组长岗位职责
2014/01/29 职场文书
教师产假请假条范文
2014/04/10 职场文书
弘扬雷锋精神演讲稿
2014/05/10 职场文书
湖南省党的群众路线教育实践活动总结会议新闻稿
2014/10/21 职场文书
服务整改报告
2014/11/06 职场文书
诚信承诺书
2015/01/19 职场文书
go原生库的中bytes.Buffer用法
2021/04/25 Golang
比较node.js和Deno
2021/04/27 Javascript
ORM模型框架操作mysql数据库的方法
2021/07/25 MySQL