Rails 布署最佳实践

2017-12-28 11:13:31 +08:00
 rina

准备工作

本文介绍的内容为:服务器 ubuntu16.04, Rails 项目,mina 发布

安装 rails 环境

adduser newuser

设置新用户密码

配置 ssh

在本地生成 SSH KEY (如果已经有了,可以忽略这步。)

$ ssh-keygen -t rsa -b 4096 -C "your_email@example.com"

生成完之后,在 ~/.ssh/ 目录下会添加两个文件,默认情况下一个私钥 id_rsa, 一个公钥 id_rsa.pub.

配置无密码登录服务器

这一步是为了给 mina 发布提供方便,因为 mina 发布需要无密登录服务器,执行发布命令。

$ scp ~/.ssh/id_rsa.pub root@your.domain.com:~/.ssh/

$ ssh root@your.domain.com

$ cd ~/.ssh/

$ cat id_rsa.pub >> authorized_keys

$ rm id_rsa.pub

ssh root@your.domain.com

以上是配置 root 用户无密码登录, 同样给 newuser 用户配置成无密码登录。

vi /etc/ssh/sshd_config

更新:

PasswordAuthentication no

为了确保能顺利安装 Rails,先要安装 Node.js 和 Yarn 的系统依赖

以下操作使用 root 账号执行

curl -sL https://deb.nodesource.com/setup_8.x | sudo -E bash -
curl -sS https://dl.yarnpkg.com/debian/pubkey.gpg | sudo apt-key add -
echo "deb https://dl.yarnpkg.com/debian/ stable main" | sudo tee /etc/apt/sources.list.d/yarn.list

sudo apt-get update
sudo apt-get install git-core curl zlib1g-dev build-essential libssl-dev libreadline-dev libyaml-dev libsqlite3-dev sqlite3 libxml2-dev libxslt1-dev libcurl4-openssl-dev python-software-properties libffi-dev nodejs yarn

安装 rbenv

以下操作使用 newuser 账号执行

cd
git clone https://github.com/rbenv/rbenv.git ~/.rbenv
echo 'export PATH="$HOME/.rbenv/bin:$PATH"' >> ~/.bashrc
echo 'eval "$(rbenv init -)"' >> ~/.bashrc
exec $SHELL

git clone https://github.com/rbenv/ruby-build.git ~/.rbenv/plugins/ruby-build
echo 'export PATH="$HOME/.rbenv/plugins/ruby-build/bin:$PATH"' >> ~/.bashrc
exec $SHELL

rbenv install 2.4.3
rbenv global 2.4.3
ruby -v
gem install bundler
rbenv rehash

配置 git 账号

以下操作使用 newuser 账号执行

git config --global color.ui true
git config --global user.name "YOUR NAME"
git config --global user.email "YOUR@EMAIL.com"
ssh-keygen -t rsa -b 4096 -C "YOUR@EMAIL.com"

在服务器上生成了 ssh key 后, 将下面命令输入的内容,复制后粘贴到你 github 账号SSH and GPG keys下:https://github.com/settings/keys

cat ~/.ssh/id_rsa.pub
ssh -T git@github.com

如果成功会输出以下信息:

Hi kfzyutc! You've successfully authenticated, but GitHub does not provide shell access.

安装 Rails

$ gem install rails -v 5.1.4

$ rbenv rehash

$ rails -v
# Rails 5.1.4

安装 MySQL

sudo apt-get install mysql-server mysql-client libmysqlclient-dev

安装 PostgreSQL

sudo sh -c "echo 'deb http://apt.postgresql.org/pub/repos/apt/ xenial-pgdg main' > /etc/apt/sources.list.d/pgdg.list"
wget --quiet -O - http://apt.postgresql.org/pub/repos/apt/ACCC4CF8.asc | sudo apt-key add -
sudo apt-get update
sudo apt-get install postgresql-common
sudo apt-get install postgresql-9.5 libpq-dev

postgres 安装不会为你设置一个用户,所以你需要按照以下步骤创建一个有权创建数据库的用户。 随意用你的用户名替换 chris

sudo -u postgres createuser chris -s

# If you would like to set a password for the user, you can do the following
sudo -u postgres psql
postgres=# \password chris

配置 nginx

upstream RBlog {
  server unix:///home/newuser/xxx/shared/tmp/sockets/puma.sock fail_timeout=0; 
}

server {
  listen 80;
  server_name example.com;
  root /home/newuser/xxx/current/public;

  location ^~ /assets/ {
    gzip_static on;
    expires max;
    add_header Cache-Control public;
  }

  location /cable {
    proxy_http_version 1.1;
    proxy_set_header Upgrade $http_upgrade;
    proxy_set_header Connection "Upgrade";
    proxy_set_header Host $host;
    proxy_set_header X-Real-IP $remote_addr;
    proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
    proxy_pass http://xxx;
  }

  location ~ ^/(uploads)/  {
    expires max;
    break;
  }


  try_files $uri/index.html $uri @RBlog;
  location @xxx {
    proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
    proxy_set_header Host $http_host;
    proxy_set_header X-Real-IP $remote_addr;
    proxy_redirect off;
    proxy_pass http://xxx;
  }

  error_page 500 502 503 504 /500.html;
  client_max_body_size 20M;
  keepalive_timeout 10;
}

nginx -t
# nginx: the configuration file /etc/nginx/nginx.conf syntax is ok
# nginx: configuration file /etc/nginx/nginx.conf test is successful
nginx -s restart

发布 Rails 项目

如果你是用这个 模板 创建的 Rails 项目, 以下配置信息都已经添加了,只需要更新目录信息就行。

如果没有需要在 Gemfile 里添加这些 gem 包。

gem 'mina', '~>0.3.8', require: false
gem 'mina-puma', '~>0.3.2', require: false
gem 'mina-multistage', '~> 1.0', '>= 1.0.2', require: false
gem 'mina-sidekiq', '~> 0.3.1', require: false
gem 'mina-logs', '>= 0.1.0', require: false
gem 'whenever', :require => false

在你的 Rails 项目下配置 puma,config/puma.rb, 替换 /home/newuser/xxx 目录.

if ENV['RAILS_ENV'] == 'production'
  app_root = "/home/newuser/xxx/shared"
  pidfile "#{app_root}/tmp/pids/puma.pid"
  state_path "#{app_root}/tmp/pids/puma.state"
  bind "unix://#{app_root}/tmp/sockets/puma.sock"
  activate_control_app "unix://#{app_root}/tmp/sockets/pumactl.sock"
  daemonize true
  workers 4
  threads 8, 16
  preload_app!

  on_worker_boot do
    ActiveSupport.on_load(:active_record) do
      ActiveRecord::Base.establish_connection
    end
  end

  before_fork do
    ActiveRecord::Base.connection_pool.disconnect!
  end
else
  plugin :tmp_restart
end

配置 deploy, config/deploy.rb

set :stages, %w(production staging utcprod)
set :default_stage, 'staging'

require 'mina/multistage'
require 'mina/bundler'
require 'mina/rails'
require 'mina/git'
require 'mina/rbenv'
require 'mina/puma'
require "mina_sidekiq/tasks"
require 'mina/logs'
require 'mina/whenever'

set :shared_paths, ['config/database.yml', 'config/newrelic.yml', 'config/application.yml', 'log', 'public/uploads']

task :environment do
  invoke :'rbenv:load'
end

task :setup => :environment do
  queue! %[mkdir -p "#{deploy_to}/shared/tmp/sockets"]
  queue! %[chmod g+rx,u+rwx "#{deploy_to}/shared/tmp/sockets"]

  queue! %[mkdir -p "#{deploy_to}/shared/pids"]
  queue! %[chmod g+rx,u+rwx "#{deploy_to}/shared/pids"]

  queue! %[mkdir -p "#{deploy_to}/shared/tmp/pids"]
  queue! %[chmod g+rx,u+rwx "#{deploy_to}/shared/tmp/pids"]

  queue! %[mkdir -p "#{deploy_to}/#{shared_path}/log"]
  queue! %[chmod g+rx,u+rwx "#{deploy_to}/#{shared_path}/log"]

  queue! %[mkdir -p "#{deploy_to}/#{shared_path}/public/uploads"]
  queue! %[chmod g+rx,u+rwx "#{deploy_to}/#{shared_path}/public/uploads"]

  queue! %[mkdir -p "#{deploy_to}/#{shared_path}/config"]
  queue! %[chmod g+rx,u+rwx "#{deploy_to}/#{shared_path}/config"]

  queue! %[touch "#{deploy_to}/#{shared_path}/config/application.yml"]
  queue  %[echo "-----> Be sure to edit '#{deploy_to}/#{shared_path}/config/application.yml'"]

  queue! %[touch "#{deploy_to}/#{shared_path}/config/database.yml"]
  queue  %[echo "-----> Be sure to edit '#{deploy_to}/#{shared_path}/config/database.yml'"]
end

desc "Deploys the current version to the server."
task :deploy => :environment do
  queue  %[echo "-----> Server: #{domain}"]
  queue  %[echo "-----> Path: #{deploy_to}"]
  queue  %[echo "-----> Branch: #{branch}"]

  deploy do
    invoke :'sidekiq:quiet'
    invoke :'git:clone'
    invoke :'deploy:link_shared_paths'
    invoke :'bundle:install'
    invoke :'rails:db_migrate'
    invoke :'rails:assets_precompile'
    invoke :'deploy:cleanup'

    to :launch do
      invoke :'puma:hard_restart'
      invoke :'sidekiq:restart'
      invoke :'whenever:update'
    end
  end
end

desc "Deploys the current version to the server."
task :first_deploy => :environment do
  queue  %[echo "-----> Server: #{domain}"]
  queue  %[echo "-----> Path: #{deploy_to}"]
  queue  %[echo "-----> Branch: #{branch}"]

  deploy do
    invoke :'git:clone'
    invoke :'deploy:link_shared_paths'
    invoke :'bundle:install'
    invoke :'deploy:cleanup'

    to :launch do
      invoke :'rails:db_create'
    end
  end
end

config/deploy/production.rb,将 domain, deploy_to, repository, branch, user 替换成你自己的信息。

set :domain, 'your.domain.com'
set :deploy_to, '/home/newuser/xxx'
set :repository,  'git@github.com:xxx/xxx.git'
set :branch, 'master'
set :user, 'newuser'
set :puma_config, ->{ "#{deploy_to}/#{current_path}/config/puma/production.rb" }

这些配置信息更新好之后,提交到 github

首次发布

mina setup
mina first_deploy

后续发布

mina deploy

本地查看生产环境日志

mina log

本地登录生产环境数据库控制台

mina console

本地登录服务器

mina ssh

参考文档

Ubuntu 安装 Rails 环境: https://gorails.com/setup/ubuntu/16.04

Mina 配置: https://github.com/80percent/rails-template

原文: http://liuzhen.me/articles/17

3696 次点击
所在节点    Ruby on Rails
9 条回复
linuxchild
2017-12-28 14:17:00 +08:00
lz 是把这个当博客了么。。
rina
2017-12-28 14:26:19 +08:00
@linuxchild

首先我并不认为我这篇文章没有价值,在发之前我也考虑了最近已经发过几篇会不会发太多了,但是从我博客的这篇文章的访问量看远远超过其他文章,所以我觉得应该有很多人需要,而且目前为止也有人收藏,另一篇文章收藏数也有 10 几个人,从中也证明确实对一些人是有帮助的,如果你不喜欢可以不看。
msg7086
2017-12-28 19:38:08 +08:00
这是哪门子最佳实践……
要我说最佳实践的话,一个,服务器用 puppet 或者 chef 管理组件,一个,用 CI 保证程序能在几个主要 ruby 版本上运行,然后让操作系统去管理 ruby 的版本升级。
再或者索性上 docker,让 docker 的 up 主来管理 ruby 的升级。
sparkle2015
2017-12-28 22:49:58 +08:00
对我来说还挺有价值的,前段时间刚在 EC2 上折腾过一次 Rails 的布署,把这些流程都走过一遍。不过用的是 capistrano,好奇 lz 为什么没用 capistrano (这个应该是主流了吧)。
msg7086
2017-12-29 00:18:54 +08:00
@sparkle2015 mina 比 capistrano 不知道快到哪里去了。
bydmm
2017-12-29 10:07:00 +08:00
docker push
docker compose up
zealinux
2017-12-29 12:53:31 +08:00
lz 用 mina 的新版了吗?
问题一大堆,mina-puma
r00tt
2017-12-29 13:14:26 +08:00
还在用着 capistrano,感觉还行
yuanfnadi
2017-12-31 18:35:46 +08:00
docker run

一句话就够了

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/418224

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX