在Python的Tornado框架中实现简单的在线代理的教程


Posted in Python onMay 02, 2015

实现代理的方式很多种,流行的web服务器也大都有代理的功能,比如http://www.tornadoweb.cn用的就是nginx的代理功能做的tornadoweb官网的镜像。

最近,我在开发一个移动运用(以下简称APP)的后台程序(Server),该运用需要调用到另一平台产品(Platform)的API。对于这个系统来说,可选的一种实现方式方式是APP同时跟Server&Platform两者交互;另一种则在Server端封装掉Platform的API,APP只和Server交互。显然后一种方式的系统架构会清晰些,APP编程时也就相对简单。那么如何在Server端封装Platform的API呢,我首先考虑到的就是用代理的方式来实现。碰巧最近Tornado邮件群组里有人在讨论using Tornado as a proxy,贴主提到的运用场景跟我这碰到的场景非常的相似,我把原帖的代码做了些整理和简化,源代码如下:

# -*- coding: utf-8 -*-
#
# Copyright(c) 2011 Felinx Lee & http://feilong.me/
#
# Licensed under the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License. You may obtain
# a copy of the License at
#
#   http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
 
import logging
 
import tornado.httpserver
import tornado.ioloop
import tornado.options
import tornado.web
import tornado.httpclient
from tornado.web import HTTPError, asynchronous
from tornado.httpclient import HTTPRequest
from tornado.options import define, options
try:
  from tornado.curl_httpclient import CurlAsyncHTTPClient as AsyncHTTPClient
except ImportError:
  from tornado.simple_httpclient import SimpleAsyncHTTPClient as AsyncHTTPClient
 
define("port", default=8888, help="run on the given port", type=int)
define("api_protocol", default="http")
define("api_host", default="feilong.me")
define("api_port", default="80")
define("debug", default=True, type=bool)
 
class ProxyHandler(tornado.web.RequestHandler):
  @asynchronous
  def get(self):
    # enable API GET request when debugging
    if options.debug:
      return self.post()
    else:
      raise HTTPError(405)
 
  @asynchronous
  def post(self):
    protocol = options.api_protocol
    host = options.api_host
    port = options.api_port
 
    # port suffix
    port = "" if port == "80" else ":%s" % port
 
    uri = self.request.uri
    url = "%s://%s%s%s" % (protocol, host, port, uri)
 
    # update host to destination host
    headers = dict(self.request.headers)
    headers["Host"] = host
 
    try:
      AsyncHTTPClient().fetch(
        HTTPRequest(url=url,
              method="POST",
              body=self.request.body,
              headers=headers,
              follow_redirects=False),
        self._on_proxy)
    except tornado.httpclient.HTTPError, x:
      if hasattr(x, "response") and x.response:
        self._on_proxy(x.response)
      else:
        logging.error("Tornado signalled HTTPError %s", x)
 
  def _on_proxy(self, response):
    if response.error and not isinstance(response.error,
                       tornado.httpclient.HTTPError):
      raise HTTPError(500)
    else:
      self.set_status(response.code)
      for header in ("Date", "Cache-Control", "Server", "Content-Type", "Location"):
        v = response.headers.get(header)
        if v:
          self.set_header(header, v)
      if response.body:
        self.write(response.body)
      self.finish()
 
def main():
  tornado.options.parse_command_line()
  application = tornado.web.Application([
    (r"/.*", ProxyHandler),
  ])
  http_server = tornado.httpserver.HTTPServer(application)
  http_server.listen(options.port)
  tornado.ioloop.IOLoop.instance().start()
 
if __name__ == "__main__":
  main()

运行上面的代码后,访问 http://localhost:8888/ 将会完整显示飞龙博客的首页,即代理访问了http://feilong.me/的内容。

我考虑用程序的方式来做代理而不是直接用Nginx来做代理,其中一点是考虑到用程序可以很容易的控制Platform的哪些API是需要代理的,而哪些是要屏蔽掉的,还有哪些可能是要重写的(比如Server的login可能不能直接代理Platform的login,但却要调用到Platform的login API)。

以上这段代码只是做了简单的页面内容代理,并没有对页面进行进一步的解析处理,比如链接替换等,这些就交个有兴趣的朋友去开发了。基于以上这段代码,将其扩展一下,是完全可以实现一个完整的在线代理程序的。

这段代码我已放到了我的实验项目里,见https://bitbucket.org/felinx/labs,我将会放更多类似于这样的实验性质的小项目到这个repository里来,有兴趣的朋友可以关注一下。

转载请注明出处:http://feilong.me/2011/09/tornado-as-a-proxy

Python 相关文章推荐
利用Python中SocketServer 实现客户端与服务器间非阻塞通信
Dec 15 Python
numpy.random.seed()的使用实例解析
Feb 03 Python
python实现读取大文件并逐行写入另外一个文件
Apr 19 Python
python操作excel的包(openpyxl、xlsxwriter)
Jun 11 Python
python使用udp实现聊天器功能
Dec 10 Python
Django中如何防范CSRF跨站点请求伪造攻击的实现
Apr 28 Python
Python分支语句与循环语句应用实例分析
May 07 Python
pytorch在fintune时将sequential中的层输出方法,以vgg为例
Aug 20 Python
Python流程控制常用工具详解
Feb 24 Python
django序列化时使用外键的真实值操作
Jul 15 Python
pycharm如何使用anaconda中的各种包(操作步骤)
Jul 31 Python
python实现简单区块链结构
Apr 25 Python
探究Python的Tornado框架对子域名和泛域名的支持
May 02 #Python
Python编程中运用闭包时所需要注意的一些地方
May 02 #Python
按日期打印Python的Tornado框架中的日志的方法
May 02 #Python
详细解读Python的web.py框架下的application.py模块
May 02 #Python
使用Python的web.py框架实现类似Django的ORM查询的教程
May 02 #Python
在ironpython中利用装饰器执行SQL操作的例子
May 02 #Python
用Python编写简单的定时器的方法
May 02 #Python
You might like
php基于单例模式封装mysql类完整实例
2016/10/18 PHP
Laravel框架Request、Response及Session操作示例
2019/05/06 PHP
PHP实现关键字搜索后描红功能示例
2019/07/03 PHP
类之Prototype.js学习
2007/06/13 Javascript
FireFox与IE 下js兼容触发click事件的代码
2008/11/20 Javascript
高效的获取当前元素是父元素的第几个子元素
2013/10/15 Javascript
jQuery使用之标记元素属性用法实例
2015/01/19 Javascript
js实现根据身份证号自动生成出生日期
2015/12/15 Javascript
JavaScript操作class和style样式代码详解
2016/02/13 Javascript
Bootstrap每天必学之级联下拉菜单
2016/03/27 Javascript
AngularJS  双向数据绑定详解简单实例
2016/10/20 Javascript
微信小程序开发之数据存储 参数传递 数据缓存
2017/04/13 Javascript
vue-router之nuxt动态路由设置的两种方法小结
2018/09/26 Javascript
浅谈在vue中使用mint-ui swipe遇到的问题
2018/09/27 Javascript
vue与iframe之间的信息交互的实现
2020/04/08 Javascript
vue通过v-html指令渲染的富文本无法修改样式的解决方案
2020/05/20 Javascript
Vue 数据绑定的原理分析
2020/11/16 Javascript
pycharm debug功能实现跳到循环末尾的方法
2018/11/29 Python
python 两个数据库postgresql对比
2019/10/21 Python
Pycharm操作Git及GitHub的步骤详解
2020/10/27 Python
详解CSS3选择器的使用方法汇总
2015/11/24 HTML / CSS
纯CSS3单页切换导航菜单界面设计的简单实现
2016/08/16 HTML / CSS
html5教程实现Photoshop渐变色效果
2013/12/04 HTML / CSS
HTML5 在canvas中绘制文本附效果图
2014/06/23 HTML / CSS
canvas实现手机的手势解锁的步骤详细
2020/03/16 HTML / CSS
印度婴儿用品在线商店:Firstcry.com
2016/12/05 全球购物
英国最大的专业户外零售商:Mountain Warehouse
2018/06/06 全球购物
写给爸爸的道歉信
2014/01/15 职场文书
中专毕业生自我鉴定
2014/02/02 职场文书
冰淇淋店的创业计划书
2014/02/07 职场文书
教师党员自我评价2015
2015/03/04 职场文书
新员工试用期自我评价
2015/03/10 职场文书
入党积极分子群众意见
2015/06/01 职场文书
导游词之永泰公主墓
2019/12/04 职场文书
Nginx tp3.2.3 404问题解决方案
2021/03/31 Servers
德生TECSUN S-2000使用手册文字版
2022/05/10 无线电