質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

90.12%

Railsでコントローラのテスト(Rspec)を実行した時にテストが通らないと思っていたら、なぜか通っているのはなぜ!?

解決済

回答 1

投稿

  • 評価
  • クリップ 1
  • VIEW 1,656

shyamahira

score 17

everyday Rails RspecによるRailsテスト入門という参考書のコントローラのテストについて、下記のような例題がありました。


  • Pathリクエストをテストする

spec/controllers/contacts_controller_spec.rb

describe ContactsController do
~略~
  describe 'PATCH #update' do
    before :each do
      @contact = create(:contact,
        firstname: 'Lawrence',
        lastname: 'Smith')
    end

    context "valid attributes" do
      example '要求された@contactを取得すること' do
        patch :update, id: @contact, contact: attributes_for(:contact)
        expect(assigns(:contact)).to eq(@contact)
      end
~略~

テストを実行すると、テストは通るのですが、なぜ上記のテストでテストがパスするのかがわかりませんでした。(assigns(:contact)の値と、@contactの値が一致していない気がします。)

ちなみにFactoryとControllerの内容は下記のとおりです。

spec/factories/contacts.rb

FactoryGirl.define do
  factory :contact do
    firstname { Faker::Name.first_name }
    lastname { Faker::Name.last_name }
    email { Faker::Internet.email }
  end
end

app/controllers/contacts_controller.rb

class ContactsController < ApplicationController
  #before_action :authenticate, except: [:index, :show]
  before_action :set_contact, only: [:show, :edit, :update, :destroy]
~略~
  # PATCH/PUT /contacts/1
  # PATCH/PUT /contacts/1.json
  def update
    respond_to do |format|
      if @contact.update(contact_params)
        format.html { redirect_to @contact, notice: 'Contact was successfully updated.' }
        format.json { render :show, status: :ok, location: @contact }
      else
        format.html { render :edit }
        format.json { render json: @contact.errors, status: :unprocessable_entity }
      end
    end
  end
~略~

テスト内の各変数のプリントデバッグを試みると、下記のようになります。

spec/controllers/contacts_controller_spec.rb

describe ContactsController do
~略~
  describe 'PATCH #update' do
    before :each do
      @contact = create(:contact,
        firstname: 'Lawrence',
        lastname: 'Smith')
    end

    context '有効な属性の場合' do
      example '要求された@contactを取得すること' do
        debug_attributes_for = attributes_for(:contact)
        patch :update, id: @contact, contact: debug_attributes_for
        expect(assigns(:contact)).to eq(@contact)
        ###
        puts("debug_attributes_forデバッグ")
        p debug_attributes_for
        puts("@contactデバッグ")
        p @contact
        puts("assigns(:contact)デバッグ")
        p assigns(:contact)
        ###
      end
~略~

テスト実行結果

~略~
  PATCH#update
    有効な属性の場合
debug_attributes_forデバッグ
{:firstname=>"Ardith", :lastname=>"Effertz", :email=>"lesly.carter@mcdermott.org"}
@contactデバッグ
#<Contact id: 1, firstname: "Lawrence", lastname: "Smith", email: "westley@dickinson.org", created_at: "2015-12-14 04:06:43", updated_at: "2015-12-14 04:06:43">
assigns(:contact)デバッグ
#<Contact id: 1, firstname: "Ardith", lastname: "Effertz", email: "lesly.carter@mcdermott.org", created_at: "2015-12-14 04:06:43", updated_at: "2015-12-14 04:06:43">
~略~
Finished in 0.4773 seconds (files took 1.86 seconds to load)
20 examples, 0 failures, 4 pending

プリントデバッグした結果を見てもassigns(:contact)@contactが一致していませんが、テストは通っていて、なぜテストが通るのかが理解できないでいます。

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 1

checkベストアンサー

+1

ActiveRecordのモデルの比較は idしか比較していないからです。

この例だとどちらのContactもidが1なので、等しいとみなされています。

https://github.com/rails/rails/blob/master/activerecord/lib/active_record/core.rb#L409-L414

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2015/12/21 22:15

    お礼が遅くなり失礼しました。

    おかげさまですっきりしました。

    これからは疑問に感じたらソースを見るように心がけたいと思います。

    ありがとうございました。

    キャンセル

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 90.12%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる
  • トップ
  • RSpecに関する質問
  • Railsでコントローラのテスト(Rspec)を実行した時にテストが通らないと思っていたら、なぜか通っているのはなぜ!?