require "spec_helper" require "statesman/adapters/shared_examples" require "statesman/exceptions" describe Statesman::Adapters::ActiveRecord, active_record: true do before do prepare_model_table prepare_transitions_table end before { MyActiveRecordModelTransition.serialize(:metadata, JSON) } let(:observer) { double(Statesman::Machine, execute: nil) } let(:model) { MyActiveRecordModel.create(current_state: :pending) } it_behaves_like "an adapter", described_class, MyActiveRecordModelTransition describe "#initialize" do context "with unserialized metadata and non json column type" do before do metadata_column = double allow(metadata_column).to receive_messages(sql_type: '') allow(MyActiveRecordModelTransition).to receive_messages(columns_hash: { 'metadata' => metadata_column }) if ::ActiveRecord.respond_to?(:gem_version) && ::ActiveRecord.gem_version >= Gem::Version.new('4.2.0.a') expect(MyActiveRecordModelTransition). to receive(:type_for_attribute).with("metadata"). and_return(ActiveRecord::Type::Value.new) else expect(MyActiveRecordModelTransition). to receive_messages(serialized_attributes: {}) end end it "raises an exception" do expect do described_class.new(MyActiveRecordModelTransition, MyActiveRecordModel, observer) end.to raise_exception(Statesman::UnserializedMetadataError) end end context "with serialized metadata and json column type" do before do metadata_column = double allow(metadata_column).to receive_messages(sql_type: 'json') allow(MyActiveRecordModelTransition).to receive_messages(columns_hash: { 'metadata' => metadata_column }) if ::ActiveRecord.respond_to?(:gem_version) && ::ActiveRecord.gem_version >= Gem::Version.new('4.2.0.a') serialized_type = ::ActiveRecord::Type::Serialized.new( '', ::ActiveRecord::Coders::JSON ) expect(MyActiveRecordModelTransition). to receive(:type_for_attribute).with("metadata"). and_return(serialized_type) else expect(MyActiveRecordModelTransition). to receive_messages(serialized_attributes: { 'metadata' => '' }) end end it "raises an exception" do expect do described_class.new(MyActiveRecordModelTransition, MyActiveRecordModel, observer) end.to raise_exception(Statesman::IncompatibleSerializationError) end end context "with serialized metadata and jsonb column type" do before do metadata_column = double allow(metadata_column).to receive_messages(sql_type: 'jsonb') allow(MyActiveRecordModelTransition).to receive_messages(columns_hash: { 'metadata' => metadata_column }) if ::ActiveRecord.respond_to?(:gem_version) && ::ActiveRecord.gem_version >= Gem::Version.new('4.2.0.a') serialized_type = ::ActiveRecord::Type::Serialized.new( '', ::ActiveRecord::Coders::JSON ) expect(MyActiveRecordModelTransition). to receive(:type_for_attribute).with("metadata"). and_return(serialized_type) else expect(MyActiveRecordModelTransition). to receive_messages(serialized_attributes: { 'metadata' => '' }) end end it "raises an exception" do expect do described_class.new(MyActiveRecordModelTransition, MyActiveRecordModel, observer) end.to raise_exception(Statesman::IncompatibleSerializationError) end end end describe "#create" do let!(:adapter) do described_class.new(MyActiveRecordModelTransition, model, observer) end let(:from) { :x } let(:to) { :y } let(:create) { adapter.create(from, to) } subject { -> { create } } context "when there is a race" do it "raises a TransitionConflictError" do adapter2 = adapter.dup adapter2.create(:x, :y) adapter.last adapter2.create(:y, :z) expect { adapter.create(:y, :z) }. to raise_exception(Statesman::TransitionConflictError) end end context "when other exceptions occur" do before do allow_any_instance_of(MyActiveRecordModelTransition). to receive(:save!).and_raise(error) end context "ActiveRecord::RecordNotUnique unrelated to this transition" do let(:error) { ActiveRecord::RecordNotUnique.new("unrelated", nil) } it { is_expected.to raise_exception(ActiveRecord::RecordNotUnique) } end context "other errors" do let(:error) { StandardError } it { is_expected.to raise_exception(StandardError) } end end describe "updating the most_recent column" do subject { create } context "with no previous transition" do its(:most_recent) { is_expected.to eq(true) } end context "with a previous transition" do let!(:previous_transition) { adapter.create(from, to) } its(:most_recent) { is_expected.to eq(true) } it "updates the previous transition's most_recent flag" do expect { create }. to change { previous_transition.reload.most_recent }. from(true).to be_falsey end context "and a query on the parent model's state is made" do context "in a before action" do it "still has the old state" do allow(observer).to receive(:execute) do |phase| next unless phase == :before expect( model.transitions.where(most_recent: true).first.to_state ).to eq("y") end adapter.create(:y, :z) end end context "in an after action" do it "still has the old state" do allow(observer).to receive(:execute) do |phase| next unless phase == :after expect( model.transitions.where(most_recent: true).first.to_state ).to eq("z") end adapter.create(:y, :z) end end end end context "with two previous transitions" do let!(:previous_transition) { adapter.create(from, to) } let!(:another_previous_transition) { adapter.create(from, to) } its(:most_recent) { is_expected.to eq(true) } it "updates the previous transition's most_recent flag" do expect { create }. to change { another_previous_transition.reload.most_recent }. from(true).to be_falsey end end end end describe "#last" do let(:adapter) do described_class.new(MyActiveRecordModelTransition, model, observer) end before { adapter.create(:x, :y) } context "with a previously looked up transition" do before { adapter.last } it "caches the transition" do expect_any_instance_of(MyActiveRecordModel). to receive(:my_active_record_model_transitions).never adapter.last end context "after then creating a new transition" do before { adapter.create(:y, :z, []) } it "retrieves the new transition from the database" do expect(adapter.last.to_state).to eq("z") end end context "when a new transition has been created elsewhere" do let(:alternate_adapter) do described_class.new(MyActiveRecordModelTransition, model, observer) end before { alternate_adapter.create(:y, :z, []) } it "still returns the cached value" do expect_any_instance_of(MyActiveRecordModel). to receive(:my_active_record_model_transitions).never expect(adapter.last.to_state).to eq("y") end context "when explitly not using the cache" do it "still returns the cached value" do expect(adapter.last(force_reload: true).to_state).to eq("z") end end end end context "with a pre-fetched transition history" do before { adapter.create(:x, :y) } before { model.my_active_record_model_transitions.load_target } it "doesn't query the database" do expect(MyActiveRecordModelTransition).not_to receive(:connection) expect(adapter.last.to_state).to eq("y") end end end context "with a namespaced model" do before do CreateNamespacedARModelMigration.migrate(:up) CreateNamespacedARModelTransitionMigration.migrate(:up) end before do MyNamespace::MyActiveRecordModelTransition.serialize(:metadata, JSON) end let(:observer) { double(Statesman::Machine, execute: nil) } let(:model) do MyNamespace::MyActiveRecordModel.create(current_state: :pending) end it_behaves_like "an adapter", described_class, MyNamespace::MyActiveRecordModelTransition, association_name: :my_active_record_model_transitions end end