2012-05-18 13 views
15

Transactional Befestigungen in rspec verhindern after_commit aus aufgerufen wird, aber selbst wenn ich sie deaktivieren mitWarum läuft after_commit nicht einmal mit use_transactional_fixtures = false

RSpec.configure do |config| 
    config.use_transactional_fixtures = false 
end 

Die after_commit callback nicht läuft.

Hier ist eine Rails-Anwendung mit den neuesten rspec/Schienen, die ich das Problem auf produziert haben: git://github.com/sheabarton/after_commit_demo.git

Antwort

25

Eine Möglichkeit, um die commit Callbacks manuell auszulösen. Beispiel:

Ein wenig spät, aber hoffe das hilft anderen.

+0

Großartiger Vorschlag. Vielen Dank. –

+0

Vielen Dank. – baash05

2

This Gist hat mir geholfen.

Es Monkey-Patches ActiveRecord zu feuern nach_commit Rückrufe, auch wenn transaktionale Fixtures verwenden.

module ActiveRecord 
    module ConnectionAdapters 
    module DatabaseStatements 
     # 
     # Run the normal transaction method; when it's done, check to see if there 
     # is exactly one open transaction. If so, that's the transactional 
     # fixtures transaction; from the model's standpoint, the completed 
     # transaction is the real deal. Send commit callbacks to models. 
     # 
     # If the transaction block raises a Rollback, we need to know, so we don't 
     # call the commit hooks. Other exceptions don't need to be explicitly 
     # accounted for since they will raise uncaught through this method and 
     # prevent the code after the hook from running. 
     # 
     def transaction_with_transactional_fixtures(options = {}, &block) 
     rolled_back = false 

     transaction_without_transactional_fixtures do 
      begin 
      yield 
      rescue ActiveRecord::Rollback => e 
      rolled_back = true 
      raise e 
      end 
     end 

     if !rolled_back && open_transactions == 1 
      commit_transaction_records(false) 
     end 
     end 
     alias_method_chain :transaction, :transactional_fixtures 

     # 
     # The @_current_transaction_records is an stack of arrays, each one 
     # containing the records associated with the corresponding transaction 
     # in the transaction stack. This is used by the 
     # `rollback_transaction_records` method (to only send a rollback hook to 
     # models attached to the transaction being rolled back) but is usually 
     # ignored by the `commit_transaction_records` method. Here we 
     # monkey-patch it to temporarily replace the array with only the records 
     # for the top-of-stack transaction, so the real 
     # `commit_transaction_records` method only sends callbacks to those. 
     # 
     def commit_transaction_records_with_transactional_fixtures(commit = true) 
     unless commit 
      real_current_transaction_records = @_current_transaction_records 
      @_current_transaction_records = @_current_transaction_records.pop 
     end 

     begin 
      commit_transaction_records_without_transactional_fixtures 
     rescue # works better with that :) 
     ensure 
      unless commit 
      @_current_transaction_records = real_current_transaction_records 
     end 
     end 
     end 
     alias_method_chain :commit_transaction_records, :transactional_fixtures 
    end 
    end 
end 

Legen Sie eine neue Datei in Ihrem Rails.root/spec/support-Verzeichnis, z. .

Rails 3 wird automatisch in der Testumgebung geladen.

+0

Das ist für mich gearbeitet, aber machte meine specs unusably langsam. Ich werde stattdessen stattdessen nach after_save wechseln, obwohl ich befürchte, dass es nicht immer zu 100% mit meiner Geschäftslogik übereinstimmt. –

8

In meinem Fall habe ich ein solches Problem mit database_cleaner Einstellungen platziert unten aufgelöst:

config.use_transactional_fixtures = false 

config.before(:suite) do 
    DatabaseCleaner.strategy = :deletion 
    DatabaseCleaner.clean_with(:truncation) 
end 

config.before(:each) do 
    DatabaseCleaner.start 
end 

config.after(:each) do 
    DatabaseCleaner.clean 
end 

Dank Testing after_commit/after_transaction with Rspec

+0

Dies ist eigentlich die richtige Antwort in den meisten Fällen, obwohl die Lösch- und Trunkierungsstrategien viel langsamer als Transaktionen sind. – averell

7

Dies zu @ oben jamesdevar Antwort ähnlich ist, aber ich konnte einen Codeblock nicht hinzufügen, also muss ich einen separaten Eintrag machen.

Sie haben nicht die Änderung der Strategie für die gesamte Spec-Suite. Sie können weiterhin :transaction global verwenden und dann einfach :deletion oder :truncation (beide funktionieren) nach Bedarf verwenden. Fügen Sie einfach eine Flagge zur relevanten Spezifikation hinzu.

config.use_transactional_fixtures = false 

config.before(:suite) do 
    # The :transaction strategy prevents :after_commit hooks from running 
    DatabaseCleaner.strategy = :transaction 
    DatabaseCleaner.clean_with(:truncation) 
end 

config.before(:each, :with_after_commit => true) do 
    DatabaseCleaner.strategy = :truncation 
end 

dann in Ihren Angaben:

describe "some test requiring after_commit hooks", :with_after_commit => true do 
+0

Dies ist eine großartige Lösung, die nicht mit den 'run_callbacks' des Objekts verbunden ist und Sie müssen kein neues Juwel installieren! –