探究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生成随机MAC地址
Mar 10 Python
python基于socket实现网络广播的方法
Apr 29 Python
python简单实现旋转图片的方法
May 30 Python
pygame 精灵的行走及二段跳的实现方法(必看篇)
Jul 10 Python
python获取外网IP并发邮件的实现方法
Oct 01 Python
Python matplotlib 画图窗口显示到gui或者控制台的实例
May 24 Python
python 定时器,实现每天凌晨3点执行的方法
Feb 20 Python
在Python中使用turtle绘制多个同心圆示例
Nov 23 Python
Python 实现顺序高斯消元法示例
Dec 09 Python
Django ORM 查询表中某列字段值的方法
Apr 30 Python
使用Python Tkinter实现剪刀石头布小游戏功能
Oct 23 Python
Python读取ini配置文件传参的简单示例
Jan 05 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中计算中文字符串长度、截取中文字符串的函数代码
2011/08/09 PHP
PHP SESSION的增加、删除、修改、查看操作
2015/03/20 PHP
php base64 编码与解码实例代码
2017/03/21 PHP
兼容Mozilla必须知道的知识。
2007/01/09 Javascript
js 控制下拉菜单刷新的方法
2013/03/03 Javascript
仿新浪微博登陆邮箱提示效果的js代码
2013/08/02 Javascript
给文字加上着重号的JS代码
2013/11/12 Javascript
原生js编写设为首页兼容ie、火狐和谷歌
2014/06/05 Javascript
网站基于flash实现的Banner图切换效果代码
2014/10/14 Javascript
一款基于jQuery的图片场景标注提示弹窗特效
2015/01/05 Javascript
使用jQuery操作HTML的table表格的实例解析
2016/03/13 Javascript
浅谈js中调用函数时加不加括号的问题
2016/07/28 Javascript
利用forever和pm2部署node.js项目过程
2017/05/10 Javascript
vue中eslintrc.js配置最详细介绍
2018/12/21 Javascript
JS实现的合并两个有序链表算法示例
2019/02/25 Javascript
微信小程序 轮播图实现原理及优化详解
2019/09/29 Javascript
VUE注册全局组件和局部组件过程解析
2019/10/10 Javascript
vue.js实现三级菜单效果
2019/10/19 Javascript
Vue登录拦截 登录后继续跳转指定页面的操作
2020/08/04 Javascript
Python计算回文数的方法
2015/03/11 Python
Python引用类型和值类型的区别与使用解析
2017/10/17 Python
对Python subprocess.Popen子进程管道阻塞详解
2018/10/29 Python
python 获取键盘输入,同时有超时的功能示例
2018/11/13 Python
python使用python-pptx删除ppt某页实例
2020/02/14 Python
Python pip install之SSL异常处理操作
2020/09/03 Python
美国男女折扣服饰百货连锁店:Stein Mart
2017/05/02 全球购物
全球独特生活方式产品和礼品购物网站:AHAlife
2018/09/18 全球购物
一套比较完整的软件测试人员面试题
2012/05/13 面试题
GWT都有什么特性
2016/12/02 面试题
新闻专业个人自我评价
2013/09/21 职场文书
大学生安全责任书
2014/07/25 职场文书
业务员工作态度散漫检讨书
2014/11/02 职场文书
财务人员个人工作总结
2015/02/27 职场文书
演讲开场白和结束语
2015/05/29 职场文书
公司酒会致辞
2015/07/30 职场文书
创业计划书之韩国烧烤店
2019/09/19 职场文书