aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorrick <technoweenie@gmail.com>2008-06-28 09:19:44 -0700
committerrick <technoweenie@gmail.com>2008-06-28 09:19:44 -0700
commit74cb05698684f237a7eb91afadec0020d8910c70 (patch)
treeff961d9f9c2119a6608db90bc41d713a191a9b9f
parentb9528ad3c5379896b00772cb44faf1db0fd882d7 (diff)
downloadrails-74cb05698684f237a7eb91afadec0020d8910c70.tar.gz
rails-74cb05698684f237a7eb91afadec0020d8910c70.tar.bz2
rails-74cb05698684f237a7eb91afadec0020d8910c70.zip
add basic events and transitions. still more tests to convert
-rw-r--r--activemodel/lib/active_model/state_machine.rb11
-rw-r--r--activemodel/lib/active_model/state_machine/event.rb67
-rw-r--r--activemodel/lib/active_model/state_machine/machine.rb44
-rw-r--r--activemodel/lib/active_model/state_machine/state.rb18
-rw-r--r--activemodel/lib/active_model/state_machine/state_transition.rb40
-rw-r--r--activemodel/test/state_machine/event_test.rb51
-rw-r--r--activemodel/test/state_machine/machine_test.rb13
-rw-r--r--activemodel/test/state_machine/state_test.rb5
-rw-r--r--activemodel/test/state_machine/state_transition_test.rb88
-rw-r--r--activemodel/test/state_machine_test.rb166
10 files changed, 395 insertions, 108 deletions
diff --git a/activemodel/lib/active_model/state_machine.rb b/activemodel/lib/active_model/state_machine.rb
index bb038f6b7a..2a5ac95a3e 100644
--- a/activemodel/lib/active_model/state_machine.rb
+++ b/activemodel/lib/active_model/state_machine.rb
@@ -5,6 +5,9 @@ end
module ActiveModel
module StateMachine
+ class InvalidTransition < Exception
+ end
+
def self.included(base)
base.extend ClassMethods
end
@@ -34,10 +37,14 @@ module ActiveModel
end
end
- def current_state(name = nil)
+ def current_state(name = nil, new_state = nil)
sm = self.class.state_machine(name)
ivar = "@#{sm.name}_current_state"
- instance_variable_get(ivar) || instance_variable_set(ivar, sm.initial_state)
+ if name && new_state
+ instance_variable_set(ivar, new_state)
+ else
+ instance_variable_get(ivar) || instance_variable_set(ivar, sm.initial_state)
+ end
end
end
end \ No newline at end of file
diff --git a/activemodel/lib/active_model/state_machine/event.rb b/activemodel/lib/active_model/state_machine/event.rb
new file mode 100644
index 0000000000..cc7d563214
--- /dev/null
+++ b/activemodel/lib/active_model/state_machine/event.rb
@@ -0,0 +1,67 @@
+module ActiveModel
+ module StateMachine
+ class Event
+ attr_reader :name, :success
+
+ def initialize(name, options = {}, &block)
+ @name, @transitions = name, []
+ machine = options.delete(:machine)
+ if machine
+ machine.klass.send(:define_method, "#{name.to_s}!") do |*args|
+ machine.fire_event(name, self, true, *args)
+ end
+
+ machine.klass.send(:define_method, "#{name.to_s}") do |*args|
+ machine.fire_event(name, self, false, *args)
+ end
+ end
+ update(options, &block)
+ end
+
+ def fire(obj, to_state = nil, *args)
+ transitions = @transitions.select { |t| t.from == obj.current_state }
+ raise InvalidTransition if transitions.size == 0
+
+ next_state = nil
+ transitions.each do |transition|
+ next if to_state && !Array(transition.to).include?(to_state)
+ if transition.perform(obj)
+ next_state = to_state || Array(transition.to).first
+ transition.execute(obj, *args)
+ break
+ end
+ end
+ next_state
+ end
+
+ def transitions_from_state?(state)
+ @transitions.any? { |t| t.from? state }
+ end
+
+ def success?
+ !!@success
+ end
+
+ def ==(event)
+ if event.is_a? Symbol
+ name == event
+ else
+ name == event.name
+ end
+ end
+
+ def update(options = {}, &block)
+ if options.key?(:success) then @success = options[:success] end
+ if block then instance_eval(&block) end
+ self
+ end
+
+ private
+ def transitions(trans_opts)
+ Array(trans_opts[:from]).each do |s|
+ @transitions << StateTransition.new(trans_opts.merge({:from => s.to_sym}))
+ end
+ end
+ end
+ end
+end
diff --git a/activemodel/lib/active_model/state_machine/machine.rb b/activemodel/lib/active_model/state_machine/machine.rb
index 75ed8f8b65..1da48290c5 100644
--- a/activemodel/lib/active_model/state_machine/machine.rb
+++ b/activemodel/lib/active_model/state_machine/machine.rb
@@ -1,29 +1,49 @@
module ActiveModel
module StateMachine
class Machine
- attr_accessor :initial_state, :states, :event
+ attr_accessor :initial_state, :states, :events, :state_index
attr_reader :klass, :name
- def initialize(klass, name)
- @klass, @name, @states, @events = klass, name, [], {}
+ def initialize(klass, name, options = {}, &block)
+ @klass, @name, @states, @state_index, @events = klass, name, [], {}, {}
+ update(options, &block)
+ end
+
+ def initial_state
+ @initial_state ||= (states.first ? states.first.name : nil)
+ end
+
+ def update(options = {}, &block)
+ if options.key?(:initial) then @initial_state = options[:initial] end
+ if block then instance_eval(&block) end
+ self
+ end
+
+ def fire_event(name, record, persist, *args)
+ state_index[record.current_state].call_action(:exit, record)
+ if new_state = @events[name].fire(record, *args)
+ state_index[new_state].call_action(:enter, record)
+ record.current_state(@name, new_state)
+ else
+ false
+ end
end
def states_for_select
states.map { |st| [st.display_name, st.name.to_s] }
end
- def state(name, options = {})
- @states << State.new(self, name, options)
+ def events_for(state)
+ events = @events.values.select { |event| event.transitions_from_state?(state) }
+ events.map! { |event| event.name }
end
-
- def initial_state
- @initial_state ||= (states.first ? states.first.name : nil)
+ private
+ def state(name, options = {})
+ @states << (state_index[name] ||= State.new(name, :machine => self)).update(options)
end
- def update(options = {}, &block)
- @initial_state = options[:initial]
- instance_eval(&block)
- self
+ def event(name, options = {}, &block)
+ (@events[name] ||= Event.new(name, :machine => self)).update(options, &block)
end
end
end
diff --git a/activemodel/lib/active_model/state_machine/state.rb b/activemodel/lib/active_model/state_machine/state.rb
index 5851a6fb79..68eb2aa34a 100644
--- a/activemodel/lib/active_model/state_machine/state.rb
+++ b/activemodel/lib/active_model/state_machine/state.rb
@@ -3,11 +3,15 @@ module ActiveModel
class State
attr_reader :name, :options
- def initialize(machine, name, options={})
- @machine, @name, @options, @display_name = machine, name, options, options.delete(:display)
- machine.klass.send(:define_method, "#{name}?") do
- current_state.to_s == name.to_s
+ def initialize(name, options = {})
+ @name = name
+ machine = options.delete(:machine)
+ if machine
+ machine.klass.send(:define_method, "#{name}?") do
+ current_state.to_s == name.to_s
+ end
end
+ update(options)
end
def ==(state)
@@ -35,6 +39,12 @@ module ActiveModel
def for_select
[display_name, name.to_s]
end
+
+ def update(options = {})
+ if options.key?(:display) then @display_name = options.delete(:display) end
+ @options = options
+ self
+ end
end
end
end
diff --git a/activemodel/lib/active_model/state_machine/state_transition.rb b/activemodel/lib/active_model/state_machine/state_transition.rb
new file mode 100644
index 0000000000..f9df998ea4
--- /dev/null
+++ b/activemodel/lib/active_model/state_machine/state_transition.rb
@@ -0,0 +1,40 @@
+module ActiveModel
+ module StateMachine
+ class StateTransition
+ attr_reader :from, :to, :options
+
+ def initialize(opts)
+ @from, @to, @guard, @on_transition = opts[:from], opts[:to], opts[:guard], opts[:on_transition]
+ @options = opts
+ end
+
+ def perform(obj)
+ case @guard
+ when Symbol, String
+ obj.send(@guard)
+ when Proc
+ @guard.call(obj)
+ else
+ true
+ end
+ end
+
+ def execute(obj, *args)
+ case @on_transition
+ when Symbol, String
+ obj.send(@on_transition, *args)
+ when Proc
+ @on_transition.call(obj, *args)
+ end
+ end
+
+ def ==(obj)
+ @from == obj.from && @to == obj.to
+ end
+
+ def from?(value)
+ @from == value
+ end
+ end
+ end
+end
diff --git a/activemodel/test/state_machine/event_test.rb b/activemodel/test/state_machine/event_test.rb
new file mode 100644
index 0000000000..01f3464cf2
--- /dev/null
+++ b/activemodel/test/state_machine/event_test.rb
@@ -0,0 +1,51 @@
+require File.expand_path(File.join(File.dirname(__FILE__), '..', 'test_helper'))
+
+class EventTest < ActiveModel::TestCase
+ def setup
+ @name = :close_order
+ @success = :success_callback
+ end
+
+ def new_event
+ @event = ActiveModel::StateMachine::Event.new(@name, {:success => @success}) do
+ transitions :to => :closed, :from => [:open, :received]
+ end
+ end
+
+ test 'should set the name' do
+ assert_equal @name, new_event.name
+ end
+
+ test 'should set the success option' do
+ assert new_event.success?
+ end
+
+ uses_mocha 'StateTransition creation' do
+ test 'should create StateTransitions' do
+ ActiveModel::StateMachine::StateTransition.expects(:new).with(:to => :closed, :from => :open)
+ ActiveModel::StateMachine::StateTransition.expects(:new).with(:to => :closed, :from => :received)
+ new_event
+ end
+ end
+end
+
+class EventBeingFiredTest < ActiveModel::TestCase
+ test 'should raise an AASM::InvalidTransition error if the transitions are empty' do
+ event = ActiveModel::StateMachine::Event.new(:event)
+
+ assert_raises ActiveModel::StateMachine::InvalidTransition do
+ event.fire(nil)
+ end
+ end
+
+ test 'should return the state of the first matching transition it finds' do
+ event = ActiveModel::StateMachine::Event.new(:event) do
+ transitions :to => :closed, :from => [:open, :received]
+ end
+
+ obj = stub
+ obj.stubs(:current_state).returns(:open)
+
+ assert_equal :closed, event.fire(obj)
+ end
+end
diff --git a/activemodel/test/state_machine/machine_test.rb b/activemodel/test/state_machine/machine_test.rb
index 34a4b384ce..64dea42b1f 100644
--- a/activemodel/test/state_machine/machine_test.rb
+++ b/activemodel/test/state_machine/machine_test.rb
@@ -4,9 +4,18 @@ class MachineTestSubject
include ActiveModel::StateMachine
state_machine do
+ state :open
+ state :closed
end
state_machine :initial => :foo do
+ event :shutdown do
+ transitions :from => :open, :to => :closed
+ end
+
+ event :timeout do
+ transitions :from => :open, :to => :closed
+ end
end
state_machine :extra, :initial => :bar do
@@ -25,4 +34,8 @@ class StateMachineMachineTest < ActiveModel::TestCase
test "accesses non-default state machine" do
assert_kind_of ActiveModel::StateMachine::Machine, MachineTestSubject.state_machine(:extra)
end
+
+ test "finds events for given state" do
+ assert_equal [:shutdown, :timeout], MachineTestSubject.state_machine.events_for(:open)
+ end
end \ No newline at end of file
diff --git a/activemodel/test/state_machine/state_test.rb b/activemodel/test/state_machine/state_test.rb
index 444435d271..22d0d9eb93 100644
--- a/activemodel/test/state_machine/state_test.rb
+++ b/activemodel/test/state_machine/state_test.rb
@@ -10,12 +10,12 @@ end
class StateTest < ActiveModel::TestCase
def setup
@name = :astate
- @options = { :crazy_custom_key => 'key' }
@machine = StateTestSubject.state_machine
+ @options = { :crazy_custom_key => 'key', :machine => @machine }
end
def new_state(options={})
- ActiveModel::StateMachine::State.new(options.delete(:machine) || @machine, @name, @options.merge(options))
+ ActiveModel::StateMachine::State.new(@name, @options.merge(options))
end
test 'sets the name' do
@@ -31,6 +31,7 @@ class StateTest < ActiveModel::TestCase
end
test 'sets the options and expose them as options' do
+ @options.delete(:machine)
assert_equal @options, new_state.options
end
diff --git a/activemodel/test/state_machine/state_transition_test.rb b/activemodel/test/state_machine/state_transition_test.rb
new file mode 100644
index 0000000000..9a9e7f60c5
--- /dev/null
+++ b/activemodel/test/state_machine/state_transition_test.rb
@@ -0,0 +1,88 @@
+require File.expand_path(File.join(File.dirname(__FILE__), '..', 'test_helper'))
+
+class StateTransitionTest < ActiveModel::TestCase
+ test 'should set from, to, and opts attr readers' do
+ opts = {:from => 'foo', :to => 'bar', :guard => 'g'}
+ st = ActiveModel::StateMachine::StateTransition.new(opts)
+
+ assert_equal opts[:from], st.from
+ assert_equal opts[:to], st.to
+ assert_equal opts, st.options
+ end
+
+ uses_mocha 'checking ActiveModel StateMachine transitions' do
+ test 'should pass equality check if from and to are the same' do
+ opts = {:from => 'foo', :to => 'bar', :guard => 'g'}
+ st = ActiveModel::StateMachine::StateTransition.new(opts)
+
+ obj = stub
+ obj.stubs(:from).returns(opts[:from])
+ obj.stubs(:to).returns(opts[:to])
+
+ assert_equal st, obj
+ end
+
+ test 'should fail equality check if from are not the same' do
+ opts = {:from => 'foo', :to => 'bar', :guard => 'g'}
+ st = ActiveModel::StateMachine::StateTransition.new(opts)
+
+ obj = stub
+ obj.stubs(:from).returns('blah')
+ obj.stubs(:to).returns(opts[:to])
+
+ assert_not_equal st, obj
+ end
+
+ test 'should fail equality check if to are not the same' do
+ opts = {:from => 'foo', :to => 'bar', :guard => 'g'}
+ st = ActiveModel::StateMachine::StateTransition.new(opts)
+
+ obj = stub
+ obj.stubs(:from).returns(opts[:from])
+ obj.stubs(:to).returns('blah')
+
+ assert_not_equal st, obj
+ end
+ end
+end
+
+class StateTransitionGuardCheckTest < ActiveModel::TestCase
+ test 'should return true of there is no guard' do
+ opts = {:from => 'foo', :to => 'bar'}
+ st = ActiveModel::StateMachine::StateTransition.new(opts)
+
+ assert st.perform(nil)
+ end
+
+ uses_mocha 'checking ActiveModel StateMachine transition guard checks' do
+ test 'should call the method on the object if guard is a symbol' do
+ opts = {:from => 'foo', :to => 'bar', :guard => :test_guard}
+ st = ActiveModel::StateMachine::StateTransition.new(opts)
+
+ obj = stub
+ obj.expects(:test_guard)
+
+ st.perform(obj)
+ end
+
+ test 'should call the method on the object if guard is a string' do
+ opts = {:from => 'foo', :to => 'bar', :guard => 'test_guard'}
+ st = ActiveModel::StateMachine::StateTransition.new(opts)
+
+ obj = stub
+ obj.expects(:test_guard)
+
+ st.perform(obj)
+ end
+
+ test 'should call the proc passing the object if the guard is a proc' do
+ opts = {:from => 'foo', :to => 'bar', :guard => Proc.new {|o| o.test_guard}}
+ st = ActiveModel::StateMachine::StateTransition.new(opts)
+
+ obj = stub
+ obj.expects(:test_guard)
+
+ st.perform(obj)
+ end
+ end
+end
diff --git a/activemodel/test/state_machine_test.rb b/activemodel/test/state_machine_test.rb
index e906744c77..963ce84248 100644
--- a/activemodel/test/state_machine_test.rb
+++ b/activemodel/test/state_machine_test.rb
@@ -7,22 +7,22 @@ class StateMachineSubject
state :open, :exit => :exit
state :closed, :enter => :enter
- #event :close, :success => :success_callback do
- # transitions :to => :closed, :from => [:open]
- #end
- #
- #event :null do
- # transitions :to => :closed, :from => [:open], :guard => :always_false
- #end
+ event :close, :success => :success_callback do
+ transitions :to => :closed, :from => [:open]
+ end
+
+ event :null do
+ transitions :to => :closed, :from => [:open], :guard => :always_false
+ end
end
state_machine :bar do
state :read
state :ended
- #event :foo do
- # transitions :to => :ended, :from => [:read]
- #end
+ event :foo do
+ transitions :to => :ended, :from => [:read]
+ end
end
def always_false
@@ -80,9 +80,13 @@ class StateMachineInstanceLevelTest < ActiveModel::TestCase
assert @foo.respond_to?(:open?)
end
- #test 'should define an event! inance method' do
- # assert @foo.respond_to?(:close!)
- #end
+ test 'defines an event! instance method' do
+ assert @foo.respond_to?(:close!)
+ end
+
+ test 'defines an event instance method' do
+ assert @foo.respond_to?(:close)
+ end
end
class StateMachineInitialStatesTest < ActiveModel::TestCase
@@ -90,19 +94,19 @@ class StateMachineInitialStatesTest < ActiveModel::TestCase
@foo = StateMachineSubject.new
end
- test 'should set the initial state' do
+ test 'sets the initial state' do
assert_equal :open, @foo.current_state
end
- #test '#open? should be initially true' do
- # @foo.open?.should be_true
- #end
- #
- #test '#closed? should be initially false' do
- # @foo.closed?.should be_false
- #end
+ test '#open? should be initially true' do
+ assert @foo.open?
+ end
+
+ test '#closed? should be initially false' do
+ assert !@foo.closed?
+ end
- test 'should use the first state defined if no initial state is given' do
+ test 'uses the first state defined if no initial state is given' do
assert_equal :read, @foo.current_state(:bar)
end
end
@@ -141,34 +145,36 @@ end
# foo.close!
# end
#end
-#
-#describe AASM, '- event firing without persistence' do
-# it 'should fire the Event' do
-# foo = Foo.new
-#
-# Foo.aasm_events[:close].should_receive(:fire).with(foo)
-# foo.close
-# end
-#
-# it 'should update the current state' do
-# foo = Foo.new
-# foo.close
-#
-# foo.aasm_current_state.should == :closed
-# end
-#
-# it 'should attempt to persist if aasm_write_state is defined' do
-# foo = Foo.new
-#
-# def foo.aasm_write_state
-# end
-#
-# foo.should_receive(:aasm_write_state_without_persistence)
-#
-# foo.close
-# end
-#end
-#
+
+class StateMachineEventFiringWithoutPersistence < ActiveModel::TestCase
+ test 'updates the current state' do
+ subj = StateMachineSubject.new
+ assert_equal :open, subj.current_state
+ subj.close
+ assert_equal :closed, subj.current_state
+ end
+
+ uses_mocha 'StateMachineEventFiringWithoutPersistence' do
+ test 'fires the Event' do
+ subj = StateMachineSubject.new
+
+ StateMachineSubject.state_machine.events[:close].expects(:fire).with(subj)
+ subj.close
+ end
+
+ test 'should attempt to persist if aasm_write_state is defined' do
+ subj = StateMachineSubject.new
+
+ def subj.aasm_write_state
+ end
+
+ subj.expects(:aasm_write_state_without_persistence)
+
+ subj.close
+ end
+ end
+end
+
#describe AASM, '- persistence' do
# it 'should read the state if it has not been set and aasm_read_state is defined' do
# foo = Foo.new
@@ -180,22 +186,7 @@ end
# foo.aasm_current_state
# end
#end
-#
-#describe AASM, '- getting events for a state' do
-# it '#aasm_events_for_current_state should use current state' do
-# foo = Foo.new
-# foo.should_receive(:aasm_current_state)
-# foo.aasm_events_for_current_state
-# end
-#
-# it '#aasm_events_for_current_state should use aasm_events_for_state' do
-# foo = Foo.new
-# foo.stub!(:aasm_current_state).and_return(:foo)
-# foo.should_receive(:aasm_events_for_state).with(:foo)
-# foo.aasm_events_for_current_state
-# end
-#end
-#
+
#describe AASM, '- event callbacks' do
# it 'should call aasm_event_fired if defined and successful for bang fire' do
# foo = Foo.new
@@ -237,32 +228,31 @@ end
# foo.null
# end
#end
-#
-#describe AASM, '- state actions' do
-# it "should call enter when entering state" do
-# foo = Foo.new
-# foo.should_receive(:enter)
-#
-# foo.close
-# end
-#
-# it "should call exit when exiting state" do
-# foo = Foo.new
-# foo.should_receive(:exit)
-#
-# foo.close
-# end
-#end
-#
-#
+
+uses_mocha 'StateMachineStateActionsTest' do
+ class StateMachineStateActionsTest < ActiveModel::TestCase
+ test "calls enter when entering state" do
+ subj = StateMachineSubject.new
+ subj.expects(:enter)
+ subj.close
+ end
+
+ test "calls exit when exiting state" do
+ subj = StateMachineSubject.new
+ subj.expects(:exit)
+ subj.close
+ end
+ end
+end
+
class StateMachineInheritanceTest < ActiveModel::TestCase
- test "should have the same states as it's parent" do
+ test "has the same states as its parent" do
assert_equal StateMachineSubject.state_machine.states, StateMachineSubjectSubclass.state_machine.states
end
- #test "should have the same events as it's parent" do
- # StateMachineSubjectSubclass.aasm_events.should == Bar.aasm_events
- #end
+ test "has the same events as its parent" do
+ assert_equal StateMachineSubject.state_machine.events, StateMachineSubjectSubclass.state_machine.events
+ end
end
#
#