# 1.5 部署
即使現在還處在早期階段,我們還是要把(幾乎沒什么內容)的 Rails 應用部署到生產環境。這一步可做可不做,不過在開發過程中盡早且頻繁地部署,可以盡早發現開發中的問題。在開發環境中花費大量精力之后再部署,往往會在發布時遇到嚴重的集成問題。[[13](#fn-13)]
以前,部署 Rails 應用是件痛苦的事。但最近幾年,Rails 開發生態系統不斷成熟,已經出現很多好的解決方案了,包括使用 [Phusion Passenger](http://www.modrails.com/)(Apache 和 Nginx [[14](#fn-14)] Web 服務器的一個模塊)的共享主機和虛擬私有服務器,[Engine Yard](http://engineyard.com/) 和 [Rails Machine](http://railsmachine.com/) 這種提供全方位部署服務的公司,以及 [Engine Yard Cloud](http://cloud.engineyard.com/)、[Ninefold](https://ninefold.com/) 和 [Heroku](http://heroku.com/) 這種云部署服務。
我最喜歡使用 Heroku 部署 Rails 應用。Heroku 專門用于部署 Rails 和其他 Web 應用,部署 Rails 應用的過程異常簡單——只要源碼納入了 Git 版本控制系統就好。(這也是為什么要按照 [1.4 節](#version-control-with-git)介紹的步驟設置 Git。如果你還沒有照著做,現在趕緊做吧。)本節下面的內容專門介紹如何把我們的第一個應用部署到 Heroku 中。其中一些操作相對高級,如果沒有完全理解也不要緊。本節的重點是把應用部署到線上環境中。
## 1.5.1 搭建 Heroku 部署環境
Heroku 使用 [PostgreSQL](http://www.postgresql.org/)[[15](#fn-15)] 數據庫,所以我們要把 `pg` 加入生產組,這樣 Rails 才能和 PostgreSQL 通信:[[16](#fn-16)]
```
group :production do
gem 'pg', '0.17.1'
gem 'rails_12factor', '0.0.2'
end
```
注意,我們還添加了 `rails_12factor`,Heroku 使用這個 gem 伺服靜態資源,例如圖片和樣式表。另外,要加入[代碼清單 1.5](#listing-gemfile-sqlite-version) 所做的改動,避免在生產環境安裝 `sqlite3` gem,這是因為 Heroku 不支持 SQLite。
```
group :development, :test do
gem 'sqlite3', '1.3.9'
gem 'byebug', '3.4.0'
gem 'web-console', '2.0.0.beta3'
gem 'spring', '1.1.3'
end
```
最終得到的 `Gemfile` 如[代碼清單 1.14](#listing-gemfile-pg-gem) 所示。
##### 代碼清單 1.14:增加 gem 后的 `Gemfile`
```
source 'https://rubygems.org'
gem 'rails', '4.2.2'
gem 'sass-rails', '5.0.2'
gem 'uglifier', '2.5.3'
gem 'coffee-rails', '4.1.0'
gem 'jquery-rails', '4.0.3'
gem 'turbolinks', '2.3.0'
gem 'jbuilder', '2.2.3'
gem 'sdoc', '0.4.0', group: :doc
group :development, :test do
gem 'sqlite3', '1.3.9'
gem 'byebug', '3.4.0'
gem 'web-console', '2.0.0.beta3'
gem 'spring', '1.1.3'
end
group :production do
gem 'pg', '0.17.1' gem 'rails_12factor', '0.0.2' end
```
為了準備好部署環境,下面要運行 `bundle install` 命令,并且指定一個特殊的選項,禁止在本地安裝生產環境使用的 gem(即 `pg` 和 `rails_12factor`):
```
$ bundle install --without production
```
因為我們在[代碼清單 1.14](#listing-gemfile-pg-gem) 中只添加了用于生產環境的 gem,所以現在執行這個命令其實不會在本地安裝任何新的 gem,但是又必須執行這個命令,因為我們要把 `pg` 和 `rails_12factor` 添加到 `Gemfile.lock` 中。然后提交這次改動:
```
$ git commit -a -m "Update Gemfile.lock for Heroku"
```
接下來我們要注冊并配置一個 Heroku 新賬戶。第一步是[注冊 Heroku 賬戶](http://api.heroku.com/signup)。然后檢查系統中是否已經安裝 Heroku 命令行客戶端:
```
$ heroku version
```
使用云端 IDE 的讀者應該會看到 Heroku 客戶端的版本號,這表明可以使用命令行工具 `heroku`。在其他系統中,可能需要使用 [Heroku Toolbelt](https://toolbelt.heroku.com/) 安裝。
確認 Heroku 命令行工具已經安裝之后,使用 `heroku` 命令登錄,然后添加 SSH 密鑰:
```
$ heroku login
$ heroku keys:add
```
最后,執行 `heroku create` 命令,在 Heroku 的服務器中創建一個文件夾,用于存放演示應用,如[代碼清單 1.15](#listing-heroku-create) 所示。
##### 代碼清單 1.15:在 Heroku 中創建一個新應用
```
$ heroku create Creating damp-fortress-5769... done, stack is cedar
http://damp-fortress-5769.herokuapp.com/ | git@heroku.com:damp-fortress-5769.git
Git remote heroku added
```
`heroku` 命令會為你的應用分配一個二級域名,立即生效。當然,現在還看不到內容,我們開始部署吧。
## 1.5.2 Heroku 部署第一步
部署應用的第一步是,使用 Git 把主分支推送到 Heroku 中:
```
$ git push heroku master
```
(可能會看到一些提醒消息,現在先不管,[7.5 節](chapter7.html#professional-grade-deployment)會解決。)
## 1.5.3 Heroku 部署第二步
其實沒有第二步了。我們已經完成部署了。現在可以通過 `heroku create` 命令給出的地址(參見[代碼清單 1.15](#listing-heroku-create),如果沒用云端 IDE,在本地可以執行 `heroku open` 命令)查看剛剛部署的應用,如[圖 1.18](#fig-heroku-app) 所示。看到的頁面和[圖 1.12](#fig-hello-world-hello-app) 一樣,但是現在這個應用運行在生產環境中。
圖 1.18:運行在 Heroku 中的第一個應用
## 1.5.4 Heroku 命令
Heroku 提供了[很多命令](http://devcenter.heroku.com/heroku-command),本書只簡單介紹了幾個。下面花幾分鐘再介紹一個命令,其作用是重命名應用:
```
$ heroku rename rails-tutorial-hello
```
你別再使用這個名字了,我已經占用了。或許,現在你無需做這一步,使用 Heroku 提供的默認地址就行。不過,如果你真想重命名應用,基于安全考慮,可以使用一些隨機或難猜到的二級域名,例如:
```
hwpcbmze.herokuapp.com
seyjhflo.herokuapp.com
jhyicevg.herokuapp.com
```
使用這樣隨機的二級域名,只有你將地址告訴別人他們才能訪問你的網站。順便讓你一窺 Ruby 的強大,下面是我用來生成隨機二級域名的代碼,很精妙吧。
```
('a'..'z').to_a.shuffle[0..7].join
```
除了支持二級域名,Heroku 還支持自定義域名。其實[本書的網站](http://railstutorial.org/)[[17](#fn-17)]就放在 Heroku 中。如果你閱讀的是在線版,現在就在瀏覽一個托管于 Heroku 中的網站。在 [Heroku 文檔](http://devcenter.heroku.com/)中可以查看更多關于自定義域名的信息以及 Heroku 相關的其他話題。
- Ruby on Rails 教程
- 致中國讀者
- 序
- 致謝
- 作者譯者簡介
- 版權和代碼授權協議
- 第 1 章 從零開始,完成一次部署
- 1.1 簡介
- 1.2 搭建環境
- 1.3 第一個應用
- 1.4 使用 Git 做版本控制
- 1.5 部署
- 1.6 小結
- 1.7 練習
- 第 2 章 玩具應用
- 2.1 規劃應用
- 2.2 用戶資源
- 2.3 微博資源
- 2.4 小結
- 2.5 練習
- 第 3 章 基本靜態的頁面
- 3.1 創建演示應用
- 3.2 靜態頁面
- 3.3 開始測試
- 3.4 有點動態內容的頁面
- 3.5 小結
- 3.6 練習
- 3.7 高級測試技術
- 第 4 章 Rails 背后的 Ruby
- 4.1 導言
- 4.2 字符串和方法
- 4.3 其他數據類型
- 4.4 Ruby 類
- 4.5 小結
- 4.6 練習
- 第 5 章 完善布局
- 5.1 添加一些結構
- 5.2 Sass 和 Asset Pipeline
- 5.3 布局中的鏈接
- 5.4 用戶注冊:第一步
- 5.5 小結
- 5.6 練習
- 第 6 章 用戶模型
- 6.1 用戶模型
- 6.2 用戶數據驗證
- 6.3 添加安全密碼
- 6.4 小結
- 6.5 練習
- 第 7 章 注冊
- 7.1 顯示用戶的信息
- 7.2 注冊表單
- 7.3 注冊失敗
- 7.4 注冊成功
- 7.5 專業部署方案
- 7.6 小結
- 7.7 練習
- 第 8 章 登錄和退出
- 8.1 會話
- 8.2 登錄
- 8.3 退出
- 8.4 記住我
- 8.5 小結
- 8.6 練習
- 第 9 章 更新,顯示和刪除用戶
- 9.1 更新用戶
- 9.2 權限系統
- 9.3 列出所有用戶
- 9.4 刪除用戶
- 9.5 小結
- 9.6 練習
- 第 10 章 賬戶激活和密碼重設
- 10.1 賬戶激活
- 10.2 密碼重設
- 10.3 在生產環境中發送郵件
- 10.4 小結
- 10.5 練習
- 10.6 證明超時失效的比較算式
- 第 11 章 用戶的微博
- 11.1 微博模型
- 11.2 顯示微博
- 11.3 微博相關的操作
- 11.4 微博中的圖片
- 11.5 小結
- 11.6 練習
- 第 12 章 關注用戶
- 12.1 “關系”模型
- 12.2 關注用戶的網頁界面
- 12.3 動態流
- 12.4 小結
- 12.5 練習