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でも似たエラーが書いてありましたが、ちょっと古い?