在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的Django框架实现事务交易管理的教程
Apr 20 Python
Google开源的Python格式化工具YAPF的安装和使用教程
May 31 Python
Python 实现淘宝秒杀的示例代码
Jan 02 Python
Python实现简单的文本相似度分析操作详解
Jun 16 Python
对Python闭包与延迟绑定的方法详解
Jan 07 Python
Pandas统计重复的列里面的值方法
Jan 30 Python
python-pyinstaller、打包后获取路径的实例
Jun 10 Python
pyqt5 实现多窗口跳转的方法
Jun 19 Python
Python csv文件的读写操作实例详解
Nov 19 Python
使用Python实现Wake On Lan远程开机功能
Jan 22 Python
Python pexpect模块及shell脚本except原理解析
Aug 03 Python
深入了解Python装饰器的高级用法
Aug 13 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网站提速三大“软”招
2006/10/09 PHP
如何让thinkphp在模型中自动完成session赋值小教程
2014/09/05 PHP
PHP使用new StdClass()创建空对象的方法分析
2017/06/06 PHP
php基于自定义函数记录log日志方法
2017/07/21 PHP
php 中htmlentities导致中文无法查询问题
2018/09/10 PHP
JS判断页面加载状态以及添加遮罩和缓冲动画的代码
2012/10/11 Javascript
JS获得QQ号码的昵称,头像,生日的简单实例
2013/12/04 Javascript
jQuery中removeClass()方法用法实例
2015/01/05 Javascript
JavaScript分页功能的实现方法
2015/04/25 Javascript
JavaScript forEach()遍历函数使用及介绍
2015/07/08 Javascript
使用snowfall.jquery.js实现爱心满屏飞的效果
2017/01/05 Javascript
js 两个日期比较相差多少天的实例
2017/10/19 Javascript
微信小程序class封装http代码实例
2019/08/24 Javascript
关于layui 弹出层一闪而过就消失的解决方法
2019/09/09 Javascript
详解vue中多个有顺序要求的异步操作处理
2019/10/29 Javascript
js实现时钟定时器
2020/03/26 Javascript
详解Vue的组件中data选项为什么必须是函数
2020/08/17 Javascript
[43:57]Liquid vs Mineski 2019国际邀请赛小组赛 BO2 第二场 8.16
2019/08/19 DOTA
Python实现股市信息下载的方法
2015/06/15 Python
实例讲解Python的函数闭包使用中应注意的问题
2016/06/20 Python
Django中ORM表的创建和增删改查方法示例
2017/11/15 Python
python实现括号匹配的思路详解
2018/08/23 Python
Python实现简单的用户交互方法详解
2018/09/25 Python
Python字符串大小写转换拼接删除空白
2019/09/19 Python
Python爬虫 urllib2的使用方法详解
2019/09/23 Python
vue常用指令代码实例总结
2020/03/16 Python
Python Dict找出value大于某值或key大于某值的所有项方式
2020/06/05 Python
Python模块zipfile原理及使用方法详解
2020/08/04 Python
浅谈matplotlib默认字体设置探索
2021/02/03 Python
需要知道的CSS3动画技术
2010/01/01 HTML / CSS
吃透移动端 1px的具体用法
2019/12/16 HTML / CSS
美国购买肉、鸭、家禽、鹅肝和熟食网站:D’Artagnan
2018/11/13 全球购物
祖国在我心中演讲稿600字
2014/05/04 职场文书
停电通知范文
2015/04/16 职场文书
Python文件的操作示例的详细讲解
2021/04/08 Python
Django实现在线无水印抖音视频下载(附源码及地址)
2021/05/06 Python