VPS 运行 Node.js 的一些经验

VPS 系统选择

各系统资源占用对比

Debian、CentOS 占用的内存较小,Ubuntu 占用内存较大

经测试,在跑了 4个 Node.js 站点的低配(512MB 内存) DigitalOcean 中,Ubuntu 动辄收到内存到达 90% 的警报

各系统安装难易对比

Ubuntu、Debian 较为简单,CentOS 稍麻烦,32位系统比64位更节省内存,但是考虑到一些流行的开源程序只支持 Ubuntu 14.04+ 的 64位系统,所以首选 Ubuntu 系统。如果你同时用作科学上网使用,建议 Ubuntu 14.04 64位,因为这个版本在一些 VPS 上可以安装锐速来加速(比如 DigitalOcean/Linode

DigitalOcean 甚至推出了 Ubuntu + Node.js 一键部署,一分钟内(官方号称 55秒)就能把环境搭好,适合菜鸟(注:通过此链接注册,账户内会得到 10刀,而最低套餐 5刀/月,即可免费使用 2个月)

注:强烈建议 Create Droplet 的时候勾选上 Enable Backups(允许自动备份),虽然这将多出 20%的套餐费用,数据无价!

Debian/Ubuntu 系统下安装 Node.js

apt-get install curl  
curl -sL https://deb.nodesource.com/setup | bash -  
apt-get install -y nodejs  
apt-get install -y build-essential  

via: https://github.com/joyent/node/wiki/installing-node.js-via-package-manager#debian-and-ubuntu-based-linux-distributions

如何降低 Node.js 站点的内存占用

减少站点数量

比如我跑了 4个站点,2个面向公众,2个私用,这 2个私有项目按功用分为了两个站点,看似组织很合理,但为了性能起见,将这 2个合为一个之后资源占用减少了四分之一左右

优化 MySQL 在小内存 VPS 上的内存占用

MySQL 默认配置会占好几十 M 内存,有时候会经常由于内存不足挂掉,需要对其优化一下

1. 使用 SQLite 代替 MySQL

SQLite 的缺点是不支持并发写入操作,对代码的逻辑影响比较大

2. 修改 MySQL 配置

其实 MySQL 安装后自带了几个配置示例,位于 /usr/share/doc/mysql-server-5.5/examples/

如果文件是 gz 后缀,需要先解压才能看到源文件,之后可以得到

my-small.cnf  
my-medium.cnf  
...

等文件,文件开头有注释可以查看一下,比如 my-small.cnf

# This is for a system with little memory (<= 64M) where MySQL is only used
# from time to time and it's important that the mysqld daemon
# doesn't use much resources.

你可以选择合适的文件替换 /etc/mysql/my.cnf,随后重启 MySQL 即可

/etc/init.d/mysql restart

目前,我自己的小项目使用的是 my-medium.cnf,在 512M 的 VPS 上占用了不到 10M 的内存,够省吧

创建 swap 分区

类似与 Windows 下的虚拟内存,当内存不足的时候,把一部分硬盘空间虚拟成内存使用,从而解决内存容量不足的情况

尤其是运行 MySQL 数据库时,在小内存 VPS 中经常会遇到因为内存不足数据库被断开连接的情形,那么可以通过创建 swap 解决

通常情况下, VPS 安装系统时会自动创建,而 DigitalOcean 需要我们手动创建,一般设为物理内存的 1.5倍,以 512MB 内存为例,执行如下命令即可

cd /var  
touch swap.img  
chmod 600 swap.img  
dd if=/dev/zero of=/var/swap.img bs=1024k count=1000  
mkswap /var/swap.img  
swapon /var/swap.img  
echo "/var/swap.img    none    swap    sw    0    0" >> /etc/fstab  
echo "vm.swappiness=10" >> /etc/sysctl.conf  
echo "vm.vfs_cache_pressure=50" >> /etc/sysctl.conf  
echo "Swap created and added to /etc/fstab for boot up."  
cd ~  

via https://www.digitalocean.com/community/tutorials/how-to-configure-virtual-memory-swap-file-on-a-vps
https://laracasts.com/discuss/channels/forge/does-forge-run-ok-on-a-digitalocean-512mb-instance

其他的可根据内存大小更改 count=1000 的数值,内存 * 1.5 即可

定时重启网站

重启是释放软件资源的一个行之有效的方式:P

Node.js 进程守护

运行 Node.js 站点要考虑项目异常退出,服务器重启等情况,需要一个进程守护程序,比如 PM2

安装 PM2
npm install pm2 -g  
创建 PM2 自启动脚本
pm2 startup debian  

这样 VPS 系统重启后 PM2 就能自动运行了,当然前提是 Node.js 站点都是 PM2 来启动的

使用 PM2 启动 Node.js 站点

进入站点根目录,原先的启动命令是

node app.js  

现在改用

pm2 start app.js --name myappname  

myappname 改为你想要的名称即可,加上 --name 参数的原因是如果有多个 Node.js 站点,并且它们各自的启动文件都是 app.js,那么在进程中很难辨别各个站点

还有更多用法,比如以生产模式启动 Ghost

NODE_ENV=production pm2 start index.js --name ghost  

随后保存

pm2 save  

按同样的方式把所有站点都添加进来,这样 PM2 启动时就能自动启动所有站点了

控制 PM2 的日志大小

曾经有一次,PM2 撑爆了 VPS 的硬盘!我们还需要一个东东来限制一下

pm2 install pm2-logrotate;pm2 set pm2-logrotate:retain 30  

最多保存 30 条

查看所有 PM2 进程
pm2 list  

这样就能看到所有站点的运行状态了,比如内存占用、重启次数等等

┌──────────┬────┬──────┬──────┬────────┬───────────┬────────┬──────────────┬──────────┐
│ App name │ id │ mode │ PID  │ status │ restarted │ uptime │       memory │ watching │
├──────────┼────┼──────┼──────┼────────┼───────────┼────────┼──────────────┼──────────┤
│ app      │ 1  │ fork │ 5853 │ online │        99 │ 6h     │ 103.625 MB   │ disabled │
│ rss      │ 3  │ fork │ 5857 │ online │        26 │ 6h     │  95.340 MB   │ disabled │
│ ghost    │ 4  │ fork │ 5861 │ online │        23 │ 6h     │  92.555 MB   │ disabled │
└──────────┴────┴──────┴──────┴────────┴───────────┴────────┴──────────────┴──────────┘

定时重启 PM2

系统内置定时执行任务的功能,编辑 etc/crontab 文件,在文件末尾添加

00 11 * * * root /usr/bin/pm2 restart all

注:最后一定要加个空行,否则最后一行设置可能不执行

之后重启 corn 以便设置生效

/etc/init.d/cron restart

这样就设置好了每天凌晨三点自动重启 PM2,来达到重启所有站点的目的

可使用每分钟执行来测试命令是否能正常执行

*/1 * * * * root /usr/bin/pm2 restart all
时间格式介绍

00 11 * * * 这种时间格式详见 http://zh.wikipedia.org/wiki/Cron#.E6.97.B6.E9.97.B4.E8.AE.BE.E7.BD.AE

你可能会问了,凌晨三点不是应写为 00 3 * * * 吗?为什么写为 11 呢?

我们的服务器 DigitalOcean 身处美国 San Francisco 节点,输入命令

date  

可以看到输出的是 UTC 时间,详见 http://zh.wikipedia.org/wiki/%E5%8D%8F%E8%B0%83%E4%B8%96%E7%95%8C%E6%97%B6

Thu Jan  1 10:42:06 UTC 2015  

我们所在的北京时区是 UTC+8,而美国时区位于 UTC-8 ~ UTC-5,当 UTC 0点时,北京时间为 0 + 8 = 8点,而 UTC-8 时区为 -(24 - 8) = -16(前一天16点)

其中 UTC-8 又称作太平洋时间,如果你是 Apple 开发者,苹果发给你的放假邮件里就是使用的太平洋时间,你消化不了的话只要记住咱们比苹果快 16小时就够了

我的站点主要面向美国用户,这里设置的是 UTC-8 的凌晨三点,所以是 11,考虑 UTC-5 是凌晨六点,满足需求了,如果你是面向国内用户的凌晨三点,则是 19

更改时区

如果上面看不懂的话,最简单的方法是更改服务器时区啦

dpkg-reconfigure tzdata