# 8.6 練習
電子書中有練習的答案,如果想閱讀參考答案,請[購買電子書](http://railstutorial-china.org/#purchase)。
避免練習和正文沖突的方法參見[3.6 節](chapter3.html#mostly-static-pages-exercises)中的說明。
1. 在[代碼清單 8.32](#listing-user-model-remember) 中,我們定義了生成令牌和摘要的類方法,前面都加上了 `User`。這么定義沒問題,而且因為我們會使用 `User.new_token` 和 `User.digest` 調用,或許這樣定義意思更明確。不過,定義類方法有兩種更常用的方式,一種有點讓人困惑,一種極其讓人困惑。運行測試組件,確認[代碼清單 8.59](#listing-token-digest-self)(有點讓人困惑)和[代碼清單 8.60](#listing-token-digest-class-self)(極其讓人困惑)中的實現方式是正確的。(注意,在[代碼清單 8.59](#listing-token-digest-self) 和[代碼清單 8.60](#listing-token-digest-class-self) 中,`self` 是 `User` 類,而用戶模型中的其他 `self` 都是用戶對象實例。這就是讓人困惑的根源所在。)
2. [8.4.5 節](#remember-me-checkbox)說過,由于應用現在的設計方式,在[代碼清單 8.51](#listing-remember-me-test) 的集成測試中無法獲取 `remember_token` 虛擬屬性。不過,在測試中使用一個特殊的方法可以獲取,這個方法是 `assigns`。在測試中,可以訪問控制器中定義的實例變量,方法是把實例變量的符號形式傳給 `assigns` 方法。例如,如果 `create` 動作中定義了 `@user` 變量,在測試中可以使用 `assigns(:user)` 獲取這個變量。現在,會話控制器中的 `create` 動作定義了一個普通的變量(不是實例變量),名為 `user`,如果我們把它改成實例變量,就可以測試 `cookies` 中是否包含用戶的記憶令牌。填寫[代碼清單 8.61](#listing-login-create-user-instance) 和[代碼清單 8.62](#listing-improved-remember-me-test) 中缺少的內容(`?` 和 `FILL_IN`),完成改進后的“記住我”復選框測試。
##### 代碼清單 8.59:使用 `self` 定義生成令牌和摘要的方法 GREEN
app/models/user.rb
```
class User < ActiveRecord::Base
.
.
.
# 返回指定字符串的哈希摘要
def self.digest(string) cost = ActiveModel::SecurePassword.min_cost ? BCrypt::Engine::MIN_COST :
BCrypt::Engine.cost
BCrypt::Password.create(string, cost: cost)
end
# 返回一個隨機令牌
def self.new_token SecureRandom.urlsafe_base64
end
.
.
.
end
```
##### 代碼清單 8.60:使用 `class << self` 定義生成令牌和摘要的方法 GREEN
app/models/user.rb
```
class User < ActiveRecord::Base
.
.
.
class << self # 返回指定字符串的哈希摘要
def digest(string) cost = ActiveModel::SecurePassword.min_cost ? BCrypt::Engine::MIN_COST :
BCrypt::Engine.cost
BCrypt::Password.create(string, cost: cost)
end
# 返回一個隨機令牌
def new_token SecureRandom.urlsafe_base64
end
end
.
.
.
```
##### 代碼清單 8.61:在 `create` 動作中使用實例變量的模板
app/controllers/sessions_controller.rb
```
class SessionsController < ApplicationController
def new
end
def create
?user = User.find_by(email: params[:session][:email].downcase) if ?user && ?user.authenticate(params[:session][:password]) log_in ?user params[:session][:remember_me] == '1' ? remember(?user) : forget(?user) redirect_to ?user else
flash.now[:danger] = 'Invalid email/password combination'
render 'new'
end
end
def destroy
log_out if logged_in?
redirect_to root_url
end
end
```
##### 代碼清單 8.62:改進后的“記住我”復選框測試模板 GREEN
test/integration/users_login_test.rb
```
require 'test_helper'
class UsersLoginTest < ActionDispatch::IntegrationTest
def setup
@user = users(:michael)
end
.
.
.
test "login with remembering" do
log_in_as(@user, remember_me: '1')
assert_equal assigns(:user).FILL_IN, FILL_IN end
test "login without remembering" do
log_in_as(@user, remember_me: '0')
assert_nil cookies['remember_token']
end
.
.
.
end
```
- 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 練習