探究Python的Tornado框架对子域名和泛域名的支持


Posted in Python onMay 02, 2015

其实Tornado对子域名和泛域名(除了特别说明外,以下子域名和泛域名均简称为泛域名)的支持并不是什么新鲜事,两年多前我用Tornado写的开源网站 http://poweredsites.org 就有了对泛域名的支持,但是Tornado的官方文档里并没有明确对此功能进行说明,虽然源代码里是有注释的,终是有点隐晦,这不,近日mywaiting同学就遇到了这个问题,我应邀特撰此博文,分享下我对此的一点点经验。

通常,用Tornado添加url映射路由表是直接传handlers给Application这种方式的,比如官方的chatdemo:

class Application(tornado.web.Application):
  def __init__(self):
    handlers = [
      (r"/", MainHandler),
      (r"/auth/login", AuthLoginHandler),
      (r"/auth/logout", AuthLogoutHandler),
      (r"/a/message/new", MessageNewHandler),
      (r"/a/message/updates", MessageUpdatesHandler),
    ]
    settings = dict(
      cookie_secret="43oETzKXQAGaYdkL5gEmGeJJFuYh7EQnp2XdTP1o/Vo=",
      login_url="/auth/login",
      template_path=os.path.join(os.path.dirname(__file__), "templates"),
      static_path=os.path.join(os.path.dirname(__file__), "static"),
      xsrf_cookies=True,
      autoescape="xhtml_escape",
    )
    tornado.web.Application.__init__(self, handlers, **settings)

这种方式其实添加的是一个域名通配的url映射表,即域名&子域名不限,只要访问能够解析到这个chatdemo上,“/auth/login” “/auth/login”这些url就都能够正常运行。假设www.feilong.me、abc.feilong.me、feilong2.me这个三个(子)域名均配置为可由这个chatdemo程序来host,那么访问这三个(子)域名均可以正常使用这个chatdemo,总之域名是无关的。

实际上,这种方式它的内部是通过Application里的这个add_handlers来实现的(原码注释如下):

def add_handlers(self, host_pattern, host_handlers):
    """Appends the given handlers to our handler list.
 
    Note that host patterns are processed sequentially in the
    order they were added, and only the first matching pattern is
    used. This means that all handlers for a given host must be
    added in a single add_handlers call.
    """

只不过它是隐式的调用这个add_handlers而已,其关键点就在于第一个参数host_pattern(匹配域名的)上,上面那种方式,默认添加的host_pattern是”.*$”,即域名通配,若要支持泛域名,只需要显式的调用add_handlers来添加相应的host_pattern和handlers即可。

接下来就以poweredsites的源码来介绍Tornado对泛域名的支持,app.py里的Application里面有这么几句:

   

super(Application, self).__init__(handlers, **settings)
 
  # add handlers for sub domains
  for sub_handler in sub_handlers:
    # host pattern and handlers
    self.add_handlers(sub_handler[0], sub_handler[1])

常见的方式super(Application, self).__init__(handlers, **settings)添加的是根域名poweredsites的handlers,接着用for循环显式添加的是子域名和泛域名的handlers。这里的sub_handlers依次放有各子域名的handlers,其最后一个是泛域名的handlers:

sub_handlers.append(site.sub_handlers)
sub_handlers.append(blog.sub_handlers)
sub_handlers.append(admin.sub_handlers)
# wildcard subdomain handler for project should be the last one.
sub_handlers.append(project.sub_handlers)

指定的子域名的sub_handlers(site.sub_handlers)是这个样子的,这里的第一个元素就是host_pattern:

sub_handlers = ["^sites.poweredsites.org$",
        [
         (r"/", _WebsiteIndexHandler),
         (r"/feeds", _WebsitesFeedsHandler),
         (r"/([a-z0-9]{32})", _WebsiteHandler),
         (r"/([^/]+)", WebsiteHandler),
         ]
        ]

泛域名(project.sub_handlers)的区别也就在于这第一个元素,即用来做host_pattern的是通配一些子域名的:

sub_handlers = ["^[a-zA-Z_\-0-9]*\.poweredsites.org$",
        [(r"/", ProjectIndexHandler),
         (r"/top", ProjectTopHandler),
         (r"/opensource", ProjectOpensourceHandler),
         ]
        ]

在用到了泛域名的ProjectIndexHandler里,运行时具体的子域名就可以通过下面这样的方式获得:

class ProjectIndexHandler(ProjectBaseHandler):
  def get(self):
    subdomain = self.request.host.split(".")[0]

需要说明的是,Tornado里面的url映射表和Django一样是有顺序的,即url依次序由上到下匹配,只要匹配到就立即结束,不再往下匹配,而带子域名和泛域名的url路由其匹配优先级是要高于通配域名”.*$”的(这个不用你操心,add_handlers会自动为你做到这一点)。同样的,对于泛域名,因为其子域名是通配的,因此指定子域名的handlers需要放到泛域名前添加,如admin、blog这类子域名的handlers要放在泛域名之前,这就是poweredsites里sub_handlers.append(project.sub_handlers)放到最后一条的原因,project这条是对应泛域名的,http://tornado.poweredsites.org 就是靠这一条来实现的。

备注:需要支持泛域名,首先要你的域名解析支持泛域名。

转载请注明出处:http://feilong.me/2012/08/wildcard-subdomain-support-in-tornado

Python 相关文章推荐
python多重继承新算法C3介绍
Sep 28 Python
python中__call__方法示例分析
Oct 11 Python
numpy.random.seed()的使用实例解析
Feb 03 Python
Python爬虫实战:分析《战狼2》豆瓣影评
Mar 26 Python
Python 单元测试(unittest)的使用小结
Nov 14 Python
在Qt中正确的设置窗体的背景图片的几种方法总结
Jun 19 Python
Python面向对象之Web静态服务器
Sep 03 Python
django在保存图像的同时压缩图像示例代码详解
Feb 11 Python
python查找特定名称文件并按序号、文件名分行打印输出的方法
Apr 24 Python
基于PyTorch的permute和reshape/view的区别介绍
Jun 18 Python
Python classmethod装饰器原理及用法解析
Oct 17 Python
Python爬虫之爬取二手房信息
Apr 27 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
用Python程序抓取网页的HTML信息的一个小实例
May 02 #Python
You might like
PHP新手上路(五)
2006/10/09 PHP
PHP操作MySQL的mysql_fetch_* 函数的常见用法教程
2015/12/25 PHP
PHP const定义常量及global定义全局常量实例解析
2020/05/28 PHP
Javascript 陷阱 window全局对象
2008/11/26 Javascript
使用Jquery搭建最佳用户体验的登录页面之记住密码自动登录功能(含后台代码)
2011/07/10 Javascript
jQuery EasyUI API 中文文档 - NumberBox数字框
2011/10/13 Javascript
动态加载script文件的两种方法
2013/08/15 Javascript
浅析jquery的作用与优势
2013/12/02 Javascript
JavaScript实现简单的二级导航菜单实例
2015/04/15 Javascript
jQuery form插件之formDdata参数校验表单及验证后提交
2016/01/23 Javascript
返回函数的JavaScript函数
2016/06/14 Javascript
Angular.js实现动态加载组件详解
2017/05/28 Javascript
小程序实现列表多个批量倒计时
2021/01/29 Javascript
ES6 Object方法扩展的应用实例分析
2019/06/25 Javascript
python写的一个文本编辑器
2014/01/23 Python
python实现获取序列中最小的几个元素
2014/09/25 Python
python 反向输出字符串的方法
2018/07/16 Python
Python拼接微信好友头像大图的实现方法
2018/08/01 Python
对pandas读取中文unicode的csv和添加行标题的方法详解
2018/12/12 Python
基于python实现微信好友数据分析(简单)
2020/02/16 Python
Python 排序最长英文单词链(列表中前一个单词末字母是下一个单词的首字母)
2020/12/14 Python
CSS3属性 line-clamp控制文本行数的使用
2020/03/19 HTML / CSS
html5 touch事件实现触屏页面上下滑动(一)
2016/03/10 HTML / CSS
古驰英国官网:GUCCI英国
2020/03/07 全球购物
旅游专业职业生涯规划范文
2014/01/13 职场文书
学生党员的自我评价范文
2014/03/01 职场文书
行政助理工作职责范本
2014/03/04 职场文书
小学六年级学生评语
2014/04/22 职场文书
绿色家庭事迹材料
2014/05/01 职场文书
工会主席事迹材料
2014/06/03 职场文书
力学专业求职信
2014/07/23 职场文书
2014年个人技术工作总结
2014/12/08 职场文书
2015年简历自我评价范文
2015/03/11 职场文书
学生会工作感言
2015/08/07 职场文书
Java实现房屋出租系统详解
2021/10/05 Java/Android
聊聊mysql都有哪几种分区方式
2022/04/13 MySQL