一、建立一个简单的 hello world 网页

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
import tornado.ioloop
import tornado.web
 
class MainHandler( tornado.web.RequestHandler ):
    def get(self):
        self.write("Hello, world")
        return
 
settings = {
    "cookie_secret": "XmuwPAt8wHdnik4Xvc3GXmbXLifVmPZYhoc9Tx4x1iZ"
}
 
application = tornado.web.Application([
    (r"/", MainHandler)
    ],
    **settings
)
 
if __name__ == "__main__":
    application.listen(8090)
    tornado.ioloop.IOLoop.instance().start()

然后访问 http://localhost:8090/ 就可以访问最简单的网页了

具体的一些简单构建的细节在文档里说的很清楚,我就不累述了

tornado中文入门: http://www.tornadoweb.cn/

tornado中文文档: http://www.tornadoweb.cn/documentation

我主要想记载开发过程中遇到的一些比较技术性的东西

二、用户身份验证

1、一个简单的办法是,根据用户的IP

1
2
# 取得用户的IP地址
self.request.headers.get('X-Real-IP',self.request.remote_ip).decode("utf-8")

2、根据 cookie

上面这个办法当然太简陋了,另外一个办法是通过 cookies

首先要修改 settings

1
2
3
4
settings = {
    "cookie_secret": "XmuwPAt8wHdnik4Xvc3GXmbXLifVmPZYhoc9Tx4x123",
    "login_url": "/login"
}

指定一个加密 cookie 的 secret,再指定一个登陆跳转的页面,当未认证用户尝试登陆需要身份认证的页面的时候就会跳转到这一页面,一般习惯性用 /login

然后给 GET 或 POST 添加一个装饰器

1
2
3
4
5
class MainHandler( BaseHandler ):
    @tornado.web.authenticated
    def get(self):
        self.write("Hello, World")
        return

有了这个装饰器的网页就会要求用户认证身份,认证身份的方法也很简单,注意看 MainHandler 继承于 BaseHandler,这是我自己写的一个类

1
2
3
4
5
6
7
8
9
10
11
12
13
class BaseHandler(tornado.web.RequestHandler):
    def __init__( self, *args, **kwargs ):
        # 继承于 tornado.web.RequestHandler
        tornado.web.RequestHandler.__init__( self, *args, **kwargs )
 
        self.conn = DB()
        self.cursor = self.conn.cursor()
 
    # 这一方法就是 tornado 来判断用户是否已验证的方法
    # 改写这一方法。当返回 True 的时候,tornado就会认为用户已验证
    def get_current_user(self):
        # 我通过返回用户 cookie 的方式来判断是否已验证
        return self.get_secure_cookie("userid")

这时候你就需要有一个登陆页面了,你可以在里面放一个简单的 input 来输入账号密码,然后 POST 给服务器来进行验证,如果通过的话就存入 cookie

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
class Login( BaseHandler ):
def get(self):
    # 显示登陆页面,等待输入
 
def post(self):
  # 获取用户输入的账号密码
  userid = self.get_argument("userid", strip=True)
  password = self.get_argument("password", strip=True)
 
  # 对账号密码进行验证
 
  # 验证成功后存入cookie
  self.set_secure_cookie("userid", str(userid))
 
  # 这样用户就可以通过 BaseHandler 的检查了

三、静态文件夹

有时候需要用到一些静态文件夹,比如  js、css 和 ico 文件

可以通过在 settings 中增加设置来实现

1
2
3
4
5
6
7
8
9
10
11
settings = {
    # 设置静态文件夹,此处设置为了 ./static
    # 就可以直接访问 http://localhost:8090/static/* 的文件了
    "static_path": os.path.join(pwd, "static"),
    # 设置 cookie_secret
    "cookie_secret": "XmuwPAt8wHdnik4Xvc3GXmbXLifVmPZYhoc9Tx4x1iZ",
    # 设置登录页面
    "login_url": "/login",
    # 是否防跨域 POST (具体见文档)
    "xsrf_cookies": True
}

四:render 的时候将不该显示的 xsrf_form_html() 显示出来了

按照文档设置了 xsrf_cookies: True ,并且在 form 中添加了 {{ xsrf_form_html() }} 后,这条内容却被显示在了网页

这是因为 2.1 之后的 tornado 自动开启了escape,它会自动的将字符符号转义,有两种解决办法,一种是在 HTML 中改写成 {% raw xsrf_form_html() %} 来标明此条语句不转义,另外一种就是在 settings 中将自动转义关掉

1
2
3
4
5
6
7
8
settings = {
    "static_path": os.path.join(pwd, "static"),
    "cookie_secret": "XmuwPAt8wHdnik4Xvc3GXmbXLifVmPZYhoc9Tx4x1iZ",
    "login_url": "/login",
    "xsrf_cookies": True,
    # 关掉自动 escape
    "autoescape":None,
}

五、当 static 文件夹不在本地的时候

static_url 貌似只能解析本地的路径,如果部署在类似 SAE 这样的云平台的时候,代码服务器和存放文件的 Storage 服务器不在一个 domain 上时,static就没法工作了

我的解决办法是自己重写了 tornado.web.RequestHandler 中的 static_url 方法

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
class BaseHandler(tornado.web.RequestHandler):
    def __init__( self, *args, **kwargs ):
        tornado.web.RequestHandler.__init__( self, *args, **kwargs )
 
        self.conn = DB()
        self.cursor = self.conn.cursor()
 
 
    def get_current_user(self):
        return self.get_secure_cookie("userid")
 
    # 覆写 static_url 方法,让其解析到文件服务器的地址
    def static_url(self, path, include_host=None, **kwargs):
        self.require_setting("static_path", "static_url")
 
        base = "http://forecastexam-public.stor.sinaapp.com/static/"
 
        return base + path

六、异步操作

可以看看下面这几篇文章

看完这几篇应该大概明白了吧,tornado 是单线程的,要让单线程的效率达到最高,而且不因此而阻塞,最好的办法就是使用异步,异步说白了就是事件驱动,有事(触发)的时候就叫我(callback),没事的时候就 yield 挂着跑别的程序去,不阻塞主线程

tornado 自己带了一个异步的 httpclient ,所以最简单的办法就是把耗时操作单独拿出来放到另外一个 tornado 服务器里,然后用主服务器的 Asynchttpclient 去请求,等数据服务器处理好了返回数据触发 callback

上面的文章里提到了 celery,我没用过 celery,但是 Python 里最常用的异步就是 select 了吧,在 Windows 上只支持 socket,道理同上,把耗时操作放到另外的模块里,然后用 socket 通信传递数据。如果是 epoll 的话,还需要考虑是 level(状态触发) 还是 edge(边缘触发),这个是后话…(我也还在学习中

代码就不贴了,上面的例子里代码挺多了

tornado 杂记的更多相关文章

  1. Python(九)Tornado web 框架

    一.简介 Tornado 是 FriendFeed 使用的可扩展的非阻塞式 web 服务器及其相关工具的开源版本.这个 Web 框架看起来有些像web.py 或者 Google 的 webapp,不过 ...

  2. 使用tornado,我们可以做什么?

    以下介绍都是建立在python2.x的基础上面,tornado使用任意版本皆可. 如果我们需要对外提供一个http server(web api)/websocket server时,我们都可以使用t ...

  3. tornado session

    [转]tornado入门 - session cookie 和session 的区别: 1.cookie数据存放在客户的浏览器上,session数据放在服务器上. 2.cookie不是很安全,别人可以 ...

  4. tornado template

    若果使用Tornado进行web开发可能会用到模板功能,页面继承,嵌套... 多页应用模板的处理多半依赖后端(SPA就可以动态加载局部视图),就算是RESTfull的API设计,也不妨碍同时提供部分模 ...

  5. tornado上手

    http://www.tornadoweb.org/en/stable/ http://www.cnblogs.com/fanweibin/p/5418697.html import tornado. ...

  6. tornado+sqlalchemy+celery,数据库连接消耗在哪里

    随着公司业务的发展,网站的日活数也逐渐增多,以前只需要考虑将所需要的功能实现就行了,当日活越来越大的时候,就需要考虑对服务器的资源使用消耗情况有一个清楚的认知.     最近老是发现数据库的连接数如果 ...

  7. [Erlang 0118] Erlang 杂记 V

       我在知乎回答问题不多,这个问题: "对你职业生涯帮助最大的习惯是什么?它是如何帮助你的?",我还是主动回答了一下.    做笔记 一开始笔记软件做的不好的时候就发邮件给自己, ...

  8. centos 6.7 搭建tornado + nginx + supervisor的方法(已经实践)

    首先,本来不想写这篇博客了,但是我测试了很多网上的例子包括简书的,全不行,我总结原因是自己太笨,搞了俩个晚上,后来决定,自己还是写一篇记录下来,保证自己以后使用 环境: centos6.7 64 py ...

  9. tornado中将cookie值设置为json字符串

    不熟悉,找了很久,能FQ的话, https://groups.google.com/forum/#!topic/python-tornado/9Y--NgwjP_w 2楼有解释. tornado.es ...

随机推荐

  1. 启动tomcat时 一闪而过解决方法

    1 首先确定JAVA 已经配好了环境变量,具体配置方法,找一下度娘. 测试方法:进入cmd -> javac -version 能看到JAVA的版本信息,证明配置成功了. 2 分析一下问题出现的 ...

  2. sqlserver:同一数据库内负责表结构。

    select * into 新表 from 旧表   where 1=2; ## where 表示要携带的数据,随便写个恒不等式就可以只复制表结构不携带数据.

  3. cocos2dx 兼容性

    报错: Caused by: java.lang.UnsatisfiedLinkError: Couldn't load cocos2dcpp: findLibrary returned null 因 ...

  4. Error loading: \Java\jdk1.6.0_35\jre\bin\server\jvm.dll

    先看看错误:complie: [exec] Error loading: D:\Program Files\Java\jdk1.6.0_35\jre\bin\server\jvm.dll [exec] ...

  5. Codeforces Round #180 (Div. 2) B. Sail 贪心

    B. Sail 题目连接: http://www.codeforces.com/contest/298/problem/B Description The polar bears are going ...

  6. codeforces Round #258(div2) D解题报告

    D. Count Good Substrings time limit per test 2 seconds memory limit per test 256 megabytes input sta ...

  7. 【android开发】Android防止内存溢出浅析

    近期项目做得差点儿相同了,測试出现了一些问题,当中一个就是内存溢出问题,在三星手机上測试最easy出现内存溢出,在其它手机上,比方华为就没有发生,也是比較郁闷.这个问题在之前的公司,做项目时也遇到过, ...

  8. 大礼包!ANDROID内存优化(大汇总)

    写在最前: 本文的思路主要借鉴了2014年AnDevCon开发者大会的一个演讲PPT,加上把网上搜集的各种内存零散知识点进行汇总.挑选.简化后整理而成. 所以我将本文定义为一个工具类的文章,如果你在A ...

  9. 深入了解android平台的jni---本地多线程调用java代码

    一.jni调用java对象     JNI提供的功能之一是在本地代码中使用Java对象.包括:创建一个java类对象和通过函数传递一个java对象.创建一个java类对象,首先需要得到得到使用Find ...

  10. poj 3249 Test for Job (DAG最长路 记忆化搜索解决)

    Test for Job Time Limit: 5000MS   Memory Limit: 65536K Total Submissions: 8990   Accepted: 2004 Desc ...