最近使用 Tornado 重写了博客,于是查看了很多关于部署基于 Tornado 开发的网站的资料,比较成熟的方案就是使用 Nginx 来做反向代理,使用 Supervisord 来作为进程管理工具。至于什么叫反向代理,为什么 Tornado 需要使用 Supervisord 来进行进程管理,可以自己 Google 一下。现在主要介绍一下配置和其中遇到的一些问题。
Nginx 的配置文件可以参考 Tornado 的文档最后面的例子,基本上就那样了,需要说明的是,Tornado 使用 Nginx 来处理静态文件,所以在 nginx.conf
中有这样一段:
location ^~ /static/ {
root /home/blog/app; # static 文件夹所在的目录
if ($query_string) {
expires max;
}
}
和后面 Supervisord 配置有关的主要还是负载均衡的这段配置:
upstream http {
server 127.0.0.1:8000;
server 127.0.0.1:8001;
}
这个博客我开了两个 Tornado 进程,端口号分别是8000和8001,现在开始来看 supervisord.conf
的配置,其它配置可以自行 Google,这里主要讲和这两个端口有关的故事,话说需要开两个 Tornado 进程,需要使用 Supervisord 来进行管理,于是和这个有关的配置如下:
[program:blog-8000]
command=python /home/blog/app/app.py --port=8000
autostart=true ; supervisord守护程序启动时自动启动tornado
autorestart=true ; supervisord守护程序重启时自动重启tornado
redirect_stderr=true ; 将stderr重定向到stdout
stdout_logfile = /home/blog/blog-8000.log
[program:blog-8001]
command=python /home/blog/app/app.py --port=8001
autostart=true ; supervisord守护程序启动时自动启动tornado
autorestart=true ; supervisord守护程序重启时自动重启tornado
redirect_stderr=true ; 将stderr重定向到stdout
stdout_logfile = /home/blog/blog-8001.log
使用 Tornado 都知道,在 Tornado 中 main
函数都是这样的写法:
http_server = tornado.httpserver.HTTPServer(Application())
http_server.listen(port)
tornado.ioloop.IOLoop.instance().start()
[以下方法太丑陋,请看下面更新]其中必须指定 port
,这样怎么同一个文件指定两个端口呢?当然就像上面的 supervisord.conf
的配置那样,使用 --port=xxxx
来指定端口了,于是app.py
中可以写:
if __name__ == "__main__":
import sys
port = int(sys.argv[1].split('=')[1])
http_server = tornado.httpserver.HTTPServer(Application())
http_server.listen(port)
tornado.ioloop.IOLoop.instance().start()
这样就解决了。现在就可以通过 Supervisord 来启动 Tornado 进程了,即使 Tornado 进程不小心退出,Supervisord 也可以让它再次运行。不过 Supervisord 自己也完蛋了呢?比如服务器重启了,怎么办?当然是让它开机自启动了,Linux 中的开机自启动有很多方法,这里介绍一个比较传统的方法,就是把 Supervisord 加入 /etc/init.d
中, 把这个脚本保存为 supervisord
,然后加入到 /etc/init.d
文件夹中,再以 root 权限执行如下命令:
chmod +x /etc/init.d/supervisord
update-rc.d supervisord defaults
这样即使重启服务器后,Supervisord 也会自动运行,当然它也会运行它管理的进程,最后在安装 Nginx 时,如果自己不需要加入一些第三方模块,建议直接使用官方源进行安装,这样升级等非常方便,而且也是开机自启动,这样一下来,即使重启服务器,什么都不用做,网站也会原地满血复活。
最后注意:启动脚本 supervisord
中的 PIDFILE=/tmp/$NAME.pid
这个设置一定要和supervisord.conf
中的配置相同。(顺便吐槽一下这个程序的名字,太长了,不过写完这篇文章,我居然就背着了)
2012.06.27:在启动指定端口时,可以使用 Tornado 自带的命令行工具,如下:
from tornado.options import define, options
define("port", default=8888, help="run on the given port", type=int)
这样就默认为8888端口,这样__main__
就可以如下:
import tornado.options
tornado.options.parse_command_line()
http_server = tornado.httpserver.HTTPServer(Application())
http_server.listen(options.port)
tornado.ioloop.IOLoop.instance().start()
启动时就可以指定端口了。