部署 如何在 Ubuntu 云服务器上部署自己的 Rails 应用

xufeisofly · 发布于 2017年04月23日 · 最后由 nouse 回复于 2017年05月04日 · 1018 次阅读
8fce78

前言

自学rails一段时间了,之前只用heroku部署了网站,想尝试把网站以一个更“正经”的方式呈现出来,就买了一个阿里云服务器。参考了网上部分rails部署教程,过程中也遇到了一些问题,所以在完成之后总结了一下,撰写此文,方便其他像我一样的初学者日后能够快速的将一个rails网站部署到云服务器上,不求原理,只讲操作,力求简单易懂。

注:本文大量参考了ruby china中的另一篇文章:《在 Aliyun 上快速部署 Ruby on Rails 》,并根据自己遇到的一些问题做了细微补充

我的本地环境

操作系统:Ubuntu 16.04 32位
Ruby: 2.3.0
Rails: 4.2.6

Step1: 买个云服务器

在此以阿里云服务器为例,我买的是最低配置,Ubuntu 16.04操作系统,成功后会给你一个公网IP,使用ssh连接到这个IP就可以配置这台server ubuntu了,阿里云控制台上可配置云服务器的密码。现在假设我买到的云服务器信息如下:

操作系统:Ubuntu 16.04 32位
公网IP:190.74.8.177
登录密码:******
数据库:MySQL

注:第一次买成了虚拟主机,然后发现虚拟主机是不能通过ssh配置的,不能安装rails环境…还好可以退款…

Step2: 连接云服务器

现在相当于买了台电脑回来,这个电脑没有实体,但我在任何地方都可以远程登录。 登录并输入密码:

$ ssh root@190.74.8.177
root@190.74.8.177's password: ******

然后就会进入服务器账户,但永远以root用户进入不安全,所以登录后我们在server端新建一个用户:

~# adduser sofly

然后我们ctrl+D退出server,在本地重新以sofly的用户登录:

$ ssh sofly@190.74.8.177
sofly@190.74.8.177's password: ******

注:这里可以通过配置ssh-keygen来省去输入密码的过程,此处不介绍,可参考其他文章

Step3: 在云服务器上安装Ruby

以下基本参考《在 Aliyun 上快速部署 Ruby on Rails 》,不予详述。首先安装RVM:

$ \curl -L https://get.rvm.io | bash -s stable
$ echo '[[ -s "$HOME/.rvm/scripts/rvm" ]] && . "$HOME/.rvm/scripts/rvm"' >>~/.bashrc
$ source ~/.bashrc
$ rvm -v

把RVM源替换为taobao源:

$ sed -i -e 's/ftp\.ruby-lang\.org\/pub\/ruby/ruby\.taobao\.org\/mirrors\/ruby/g' ~/.rvm/config/db

安装RVM依赖等

$ rvm requirements
$ rvm pkg install readline
$ rvm pkg install openssl

安装Ruby,版本和我本地ruby一样

$ rvm install 2.3.0
$ rvm use 2.3.0 --default

安装完毕后把Rubygems的源修改成阿里云的源

$ gem source -r https://rubygems.org/
$ gem source -a http://mirrors.aliyun.com/rubygems/

Step4:在云服务器上安装Rails及数据库

继续抄袭该文。 安装Rails,版本和我本地rails一样

$ gem install rails -v 4.2.6

安装MySQL

$ sudo apt-get install mysql-server

或PostgreSQL

$ sudo apt-get install postgresql postgresql-client libpq-dev

Step5:上传本地Rails工程到云服务器

因为之前我的rails工程都提交到github或者bitbucket上了(如何提交至git此处不再赘述,请查阅相关文章),所以直接通过git的方式把代码下载到server上即可

$ git clone https://XXXXXX/project.git (在github或bitbucket上面就可以找到,直接复制)

这样代码就下载到服务器上了,然后安装gem

$ cd project
$ bundle install

创建生产环境数据库并执行迁移

$ RAILS_ENV=production rake db:create
$ RAILS_ENV=production rake db:migrate

否则最终网站页面会显示(之前heroku部署时也经常遇到)

We're sorry, but something went wrong

重新compile assets,这样所有的图片,CSS,scripts才会加载

$ RAILS_ENV=production rake assets:precompile

Step6: 安装Passenger for Nginx

Nginx是HTTP服务器,运行nginx类似于本地开启rails server,才能实现网站的访问,首先安装passenger:

$ gem install passenger

然后通过source编译的方式安装Nginx

$ rvmsudo passenger-install-nginx-module

一路回车即可,在这里选择1回车:

Automatically download and install Nginx?Nginx doesn't support loadable modules such as some other web servers do, so in order to install Nginx with Passenger support, it must be recompiled.Do you want this installer to download, compile and install Nginx for you?
1. Yes: download, compile and install Nginx for me. (recommended) The easiest way to get started. A stock Nginx 1.4.4 with Passenger support, but with no other additional third party modules, will be installed for you to a directory of your choice.
2. No: I want to customize my Nginx installation. (for advanced users) Choose this if you want to compile Nginx with more third party modules besides Passenger, or if you need to pass additional options to Nginx's 'configure' script. This installer will 1) ask you for the location of the Nginx source code, 2) run the 'configure' script according to your instructions, and 3) run 'make install'.
Whichever you choose, if you already have an existing Nginx configuration file, then it will be preserved.Enter your choice (1 or 2) or press Ctrl-C to abort:1[ENTER]

最后看到这句话即安装成功

Nginx with Passenger support was successfully installed.

Step7:说两句Nginx的坑点

很多教程上是通过apt的方式安装nginx的,而我们是通过recompile Nginx from source的方式安装的。前者的nginx目录是/usr/sbin/nginx,配置文件目录是/etc/nginx/nginx.conf,后者的nginx目录是/opt/nginx/sbin/nginx,配置文件目录是/opt/nginx/conf/nginx.conf,是不一样的,最好先确认一下这些文件的位置再进行后续操作。 另一个区别是nginx启动和停止方式不同,apt-get安装的nginx可以通过初始化脚本/etc/init.d/nginx或命令service nginx restart启动,但目前source安装方式是不能通过这两种方法启动的。 启动方式如下:

$ sudo /opt/nginx/sbin/nginx

停止方式如下:

$ ps auxw | grep nginx
root     29743  0.0  0.0  10192   564 ?        Ss   13:39   0:00 nginx: master process /opt/nginx/sbin/nginx
sofly    29744  0.0  0.4  10428  4500 ?        S    13:39   0:00 nginx: worker process
sofly    30080  0.0  0.0   5108   792 pts/1    S+   13:42   0:00 grep --color=auto nginx
$ sudo kill 29743

所以,如果想重启nginx就要先kill在启动。虽然有点麻烦,但亲测可用。

此处参考:[https://www.phusionpassenger.com/library/install/nginx/install/oss/rubygems_rvm/]

启动Nginx后,在浏览器中输入你的公网IP,即:190.74.8.177,看到

Welcome to nginx page

说明nginx server已成功启动,但还没有连接rails

Step8:配置Nginx启动rails

编辑nginx配置文件

$ sudo vim /opt/nginx/conf/nginx.conf

我的配置文件是这样的,加注释的几处注意一下就行了

user  sofly; #此处设置为部署时的用户名
worker_processes  1; #此处为云服务器核数
error_log  logs/error.log;
error_log  logs/error.log  notice;
error_log  logs/error.log  info;
pid        /run/nginx.pid; #此处为nginx.pid的目录,位置应该是在这里
events {
    worker_connections  1024;
}
http {
    passenger_root /home/sofly/.rvm/gems/ruby-2.3.0/gems/passenger-5.1.2; #去掉这两处前面的注释符号#
    passenger_ruby /home/sofly/.rvm/gems/ruby-2.3.0/wrappers/ruby; #去掉这两处前面的注释符号#

    include       mime.types;
    default_type  application/octet-stream;
    sendfile        on;
    keepalive_timeout  65;

    gzip  on;
    gzip_disable "msie6";

    server {
        listen       80;
        server_name  localhost;
        passenger_enabled on;
        root         /home/sofly/project/public; #此处设着为rails工程public文件夹位置
    }
}

配置完毕后重启nginx

$ ps auxw | grep nginx
root     29743  0.0  0.0  10192   564 ?        Ss   13:39   0:00 nginx: master process /opt/nginx/sbin/nginx
sofly    29744  0.0  0.4  10428  4500 ?        S    13:39   0:00 nginx: worker process
sofly    30080  0.0  0.0   5108   792 pts/1    S+   13:42   0:00 grep --color=auto nginx
$ sudo kill 29743
$ sudo /opt/nginx/sbin/nginx

在浏览器中输入服务器IP,发现并没有显示网页,而是出现如下的话

incomplete response received from application

What happened...

Step9: 配置rails工程的production secret_key_base

出现上面问题的原因是rails生产环境没有配置secret_key_base变量,解决方法:

$ cd project
$ bundle exec rake secret # rails 4.2.6还需要bundle exec,请根据rails版本自行匹配

将输出的一大串字码粘贴到rails工程中/config/secrets.yml去,替换掉该文件中的<%= ENV["SECRET_KEY_BASE"] %>,如下:

production:
  secret_key_base: <%= ENV["SECRET_KEY_BASE"] %>

然后再重启passenger,(一定要有否则不生效)

$ touch project/tmp/restart.txt

现在再刷新浏览器,就可以发现网站成功显示了。

此处参考:[http://stackoverflow.com/questions/29241053/incomplete-response-received-from-application-from-nginx-passenger]

注:若还是失败,请查看/opt/nginx/logs/error.log中的错误日志

PS. 从硬件转行后端之后的第一篇贴,有问题处希望大家指正。以上。

=====================================================

持续补充:部署过程中遇到的坑

  1. Rails production环境上传并resize图片报错:

"We're sorry, but something went wrong."

原因:没有安装imagemagick

$ sudo apt-get install imagemagick libmagickcore-dev libmagickwand-dev
共收到 9 条回复
1107

楼主再研究一下为网站挂上 SSL 吧,现在 HTTPS 属于标配了,当应用作为 iOS 应用的后端服务的时候也强制 HTTPS 方式连接。这个可以通过购买商业证书,或者免费的 Let's encrypt 来解决。另外,上 HTTPS 之后,类似 HSTS 之类的应用层的安全技术,还有浏览器里的混合内容之类的知识点也会浮现出来,如何提高网站 SSL 的强度也是一个话题。

部署的话,如果不使用 Docker 方案,Rails 世界最常用的是 Mina 或 Capistrano,话说虽然后者也是 Ruby 写成,但并不是 Rails 专利,很多 PHP、Python 甚至 .Net 应用,都由它部署。论坛里相关的文章很多,可以搜索一下看看。

然后还有大坑,运维。

这样整套网站部署运维的体系就算全了

1107

啊 另外建议升级到 Rails 5 来做后续的学习,即将发布的 Rails 5.1 的变化还是挺革命性的,上一次革命性变化还是五年前的 Rails 3.1。

22720

用passenger的话,建议参考官网提供的文档安装,这样就可以使用包提供的services来启动和停止了

8fce78
1107jasl 回复

受教了,最近就遇到了SSL的问题,正在试图搞明白~

1107
8fce78xufeisofly 回复

论坛里应该有相关文章可以搜下,期待搞定之后更新咯~

8fce78
22720ACzero 回复

谢谢~我也觉得上面kill线程的方法太繁琐,后面仔细研究下

15139

rvm,个人觉得ruby还是用系统包比较方便点……

nginx 1.4.4这是什么千年老妖……

rails,为什么不用bundle安装要单独提前安装?

8744

用 mina 自动部署哇

775

又看到了熟悉的RMagick,这个和database driver还有nokogiri一样,每个Rails项目都要用,但安装和部署都很头疼。

需要 登录 后方可回复, 如果你还没有账号请点击这里 注册