NoMethodError: undefined method `_conditional_callback_around_*’

rspecを書いていたら、こんなエラーがでました。

NoMethodError: undefined method `_conditional_callback_around_3878'

環境はRails 3.2.8、rspec 2.11.0、rspec-core 2.11.1、rspec-rails 2.11.0です。

itブロックを指定して実行(bundle exec rspec spec/controllers/hoge_spec.rb:20 のような行数指定)すると発生せず、spec全体を実行すると発生します。

発生したのは、このようなコード。action_cacheのテストをするためのspecです。
実際には、ActionController::Caching::Action::ActionCachePathを書き換えています(このままだとrouting errorになります)。

require 'spec_helper'

describe do
  controller(ApplicationController) do
    # caches_action :index *A*
    def index
      render :nothing => true
    end
  end

  before(:all) do
    ActionController::Base.perform_caching = true
  end
  after(:all) do
    ActionController::Base.perform_caching = false
  end
  before do
    controller.class_eval { caches_action :index }
  end

  describe do
    before do
      get :index
    end
    it { response.should be_success }
    it { response.should be_success }
  end
end

controllerでanonymous controllerを定義しています。
この定義はファイルが読み込まれたときに評価されるため、この中でcaches_actionしても(*A*の部分)、config/environments/test.rbのperform_cachingがfalseだと無視されてしまいます。

全specでperform_cachingをtrueにするのは得策ではないので、before(:all)でONにしています。
before(:all)の段階ではcontrollerが手に入らない(nil)ので、before(:each)でclass_evalしています。

この状態で全体を実行すると、冒頭のエラーが発生します。
面白いことに、itを1つだけにすると、エラーは発生しません。

caches_actionは内部で動的にActionCacheFilterを生成して、それをaround_filterしているのですが、それをこのコンテキストで実行すると、filter連番の登録がうまくいかないようです。

結局今回は、キャッシュがテストできればいいやと言うことで、以下の方法に変更しました。

require 'spec_helper'

describe do
  controller(ApplicationController) do
    # ※1
    class << self
      def cache_configured?
        true
      end
    end

    default_caches_action :index

    def index
      render :nothing => true
    end
  end

  before(:all) do
    ActionController::Base.perform_caching = true
  end
  after(:all) do
    ActionController::Base.perform_caching = false
  end
  describe 'CachingTest' do
    before do
      get :index
    end
    it { response.should be_success }
    it { response.should be_success }
  end
end

※1: def self.cache_configured?だと、特異メソッドであるモジュールが優先されてしまうので、このように書きます。

GitHubでも似たエラーが書いてありましたが、ちょっと古い?

デザインも頼めるシステム開発会社をお探しならBPS株式会社までどうぞ 開発エンジニア積極採用中です! Ruby on Rails の開発なら実績豊富なBPS

この記事の著者

baba

ゆとりプログラマー。 高校時代から趣味でプログラミングを初め、そのままコードを書き続けて現在に至る。慶應義塾大学環境情報学部(SFC)卒業。BPS設立初期に在学中から参加している最古参メンバーの一人。Ruby on Rails、PHP、Androidアプリ、Windows/Macアプリ、超縦書の開発などを気まぐれにやる。軽度の資格マニアで、情報処理技術者試験(16区分17回 + 情報処理安全確保支援士試験)、技術士(情報工学部門)、Ruby Programmer Gold、AWSソリューションアーキテクト(アソシエイト)、日商簿記2級、漢検準1級などを保有。

babaの書いた記事

夏のTechRachoフェア2019

週刊Railsウォッチ

インフラ

ActiveSupport探訪シリーズ