state_machines-activerecord 0.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/.gitignore +22 -0
- data/.travis.yml +20 -0
- data/Appraisals +11 -0
- data/Gemfile +6 -0
- data/LICENSE.txt +23 -0
- data/README.md +65 -0
- data/Rakefile +9 -0
- data/gemfiles/active_record_4.1.gemfile +13 -0
- data/gemfiles/active_record_4.2.gemfile +13 -0
- data/lib/state_machines-activerecord.rb +1 -0
- data/lib/state_machines/integrations/active_record.rb +588 -0
- data/lib/state_machines/integrations/active_record/locale.rb +12 -0
- data/lib/state_machines/integrations/active_record/version.rb +7 -0
- data/log/.gitkeep +0 -0
- data/state_machines-activerecord.gemspec +27 -0
- data/test/files/en.yml +5 -0
- data/test/files/models/post.rb +11 -0
- data/test/integration_test.rb +23 -0
- data/test/machine_by_default_test.rb +16 -0
- data/test/machine_errors_test.rb +19 -0
- data/test/machine_multiple_test.rb +17 -0
- data/test/machine_nested_action_test.rb +38 -0
- data/test/machine_unmigrated_test.rb +14 -0
- data/test/machine_with_aliased_attribute_test.rb +23 -0
- data/test/machine_with_callbacks_test.rb +172 -0
- data/test/machine_with_column_state_attribute_test.rb +44 -0
- data/test/machine_with_complex_pluralization_scopes_test.rb +16 -0
- data/test/machine_with_conflicting_predicate_test.rb +18 -0
- data/test/machine_with_conflicting_state_name_test.rb +29 -0
- data/test/machine_with_custom_attribute_test.rb +21 -0
- data/test/machine_with_default_scope_test.rb +18 -0
- data/test/machine_with_different_column_default_test.rb +27 -0
- data/test/machine_with_different_integer_column_default_test.rb +29 -0
- data/test/machine_with_dirty_attribute_and_custom_attributes_during_loopback_test.rb +24 -0
- data/test/machine_with_dirty_attribute_and_state_events_test.rb +20 -0
- data/test/machine_with_dirty_attributes_and_custom_attribute_test.rb +32 -0
- data/test/machine_with_dirty_attributes_during_loopback_test.rb +22 -0
- data/test/machine_with_dirty_attributes_test.rb +35 -0
- data/test/machine_with_dynamic_initial_state_test.rb +99 -0
- data/test/machine_with_event_attributes_on_autosave_test.rb +48 -0
- data/test/machine_with_event_attributes_on_custom_action_test.rb +41 -0
- data/test/machine_with_event_attributes_on_save_bang_test.rb +82 -0
- data/test/machine_with_event_attributes_on_save_test.rb +184 -0
- data/test/machine_with_event_attributes_on_validation_test.rb +126 -0
- data/test/machine_with_events_test.rb +13 -0
- data/test/machine_with_failed_action_test.rb +40 -0
- data/test/machine_with_failed_after_callbacks_test.rb +35 -0
- data/test/machine_with_failed_before_callbacks_test.rb +36 -0
- data/test/machine_with_initialized_state_test.rb +35 -0
- data/test/machine_with_internationalization_test.rb +180 -0
- data/test/machine_with_loopback_test.rb +22 -0
- data/test/machine_with_non_column_state_attribute_defined_test.rb +35 -0
- data/test/machine_with_non_column_state_attribute_undefined_test.rb +33 -0
- data/test/machine_with_same_column_default_test.rb +26 -0
- data/test/machine_with_scopes_and_joins_test.rb +37 -0
- data/test/machine_with_scopes_and_owner_subclass_test.rb +27 -0
- data/test/machine_with_scopes_test.rb +70 -0
- data/test/machine_with_state_driven_validations_test.rb +30 -0
- data/test/machine_with_states_test.rb +13 -0
- data/test/machine_with_static_initial_state_test.rb +166 -0
- data/test/machine_with_transactions_test.rb +26 -0
- data/test/machine_with_validations_and_custom_attribute_test.rb +21 -0
- data/test/machine_with_validations_test.rb +47 -0
- data/test/machine_without_database_test.rb +20 -0
- data/test/machine_without_transactions_test.rb +26 -0
- data/test/model_test.rb +12 -0
- data/test/test_helper.rb +42 -0
- metadata +264 -0
@@ -0,0 +1,16 @@
|
|
1
|
+
require_relative 'test_helper'
|
2
|
+
|
3
|
+
class MachineWithComplexPluralizationScopesTest < BaseTestCase
|
4
|
+
def setup
|
5
|
+
@model = new_model
|
6
|
+
@machine = StateMachines::Machine.new(@model, :status)
|
7
|
+
end
|
8
|
+
|
9
|
+
def test_should_create_singular_with_scope
|
10
|
+
assert @model.respond_to?(:with_status)
|
11
|
+
end
|
12
|
+
|
13
|
+
def test_should_create_plural_with_scope
|
14
|
+
assert @model.respond_to?(:with_statuses)
|
15
|
+
end
|
16
|
+
end
|
@@ -0,0 +1,18 @@
|
|
1
|
+
require_relative 'test_helper'
|
2
|
+
|
3
|
+
class MachineWithConflictingPredicateTest < BaseTestCase
|
4
|
+
def setup
|
5
|
+
@model = new_model do
|
6
|
+
def state?(*args)
|
7
|
+
true
|
8
|
+
end
|
9
|
+
end
|
10
|
+
|
11
|
+
@machine = StateMachines::Machine.new(@model)
|
12
|
+
@record = @model.new
|
13
|
+
end
|
14
|
+
|
15
|
+
def test_should_not_define_attribute_predicate
|
16
|
+
assert @record.state?
|
17
|
+
end
|
18
|
+
end
|
@@ -0,0 +1,29 @@
|
|
1
|
+
require_relative 'test_helper'
|
2
|
+
require 'stringio'
|
3
|
+
|
4
|
+
class MachineWithConflictingStateNameTest < BaseTestCase
|
5
|
+
def setup
|
6
|
+
@original_stderr, $stderr = $stderr, StringIO.new
|
7
|
+
|
8
|
+
@model = new_model
|
9
|
+
end
|
10
|
+
|
11
|
+
def test_should_output_warning_with_same_machine_name
|
12
|
+
@machine = StateMachines::Machine.new(@model)
|
13
|
+
@machine.state :state
|
14
|
+
|
15
|
+
assert_match(/^Instance method "state\?" is already defined in Foo, use generic helper instead.*\n$/, $stderr.string)
|
16
|
+
end
|
17
|
+
|
18
|
+
def test_should_output_warning_with_same_machine_attribute
|
19
|
+
@machine = StateMachines::Machine.new(@model, :public_state, :attribute => :state)
|
20
|
+
@machine.state :state
|
21
|
+
|
22
|
+
assert_match(/^Instance method "state\?" is already defined in Foo, use generic helper instead.*\n$/, $stderr.string)
|
23
|
+
end
|
24
|
+
|
25
|
+
def teardown
|
26
|
+
$stderr = @original_stderr
|
27
|
+
super
|
28
|
+
end
|
29
|
+
end
|
@@ -0,0 +1,21 @@
|
|
1
|
+
require_relative 'test_helper'
|
2
|
+
require 'stringio'
|
3
|
+
|
4
|
+
class MachineWithCustomAttributeTest < BaseTestCase
|
5
|
+
def setup
|
6
|
+
@original_stderr, $stderr = $stderr, StringIO.new
|
7
|
+
|
8
|
+
@model = new_model
|
9
|
+
@machine = StateMachines::Machine.new(@model, :public_state, :attribute => :state)
|
10
|
+
@record = @model.new
|
11
|
+
end
|
12
|
+
|
13
|
+
def test_should_not_delegate_attribute_predicate_with_different_attribute
|
14
|
+
assert_raise(ArgumentError) { @record.public_state? }
|
15
|
+
end
|
16
|
+
|
17
|
+
def teardown
|
18
|
+
$stderr = @original_stderr
|
19
|
+
super
|
20
|
+
end
|
21
|
+
end
|
@@ -0,0 +1,18 @@
|
|
1
|
+
require_relative 'test_helper'
|
2
|
+
|
3
|
+
class MachineWithDefaultScopeTest < BaseTestCase
|
4
|
+
def setup
|
5
|
+
@model = new_model
|
6
|
+
@machine = StateMachines::Machine.new(@model, :initial => :parked)
|
7
|
+
@machine.state :idling
|
8
|
+
|
9
|
+
@model.class_eval do
|
10
|
+
default_scope { with_state(:parked, :idling) }
|
11
|
+
end
|
12
|
+
end
|
13
|
+
|
14
|
+
def test_should_set_initial_state_on_created_object
|
15
|
+
object = @model.new
|
16
|
+
assert_equal 'parked', object.state
|
17
|
+
end
|
18
|
+
end
|
@@ -0,0 +1,27 @@
|
|
1
|
+
require_relative 'test_helper'
|
2
|
+
require 'stringio'
|
3
|
+
|
4
|
+
class MachineWithDifferentColumnDefaultTest < BaseTestCase
|
5
|
+
def setup
|
6
|
+
@original_stderr, $stderr = $stderr, StringIO.new
|
7
|
+
|
8
|
+
@model = new_model do
|
9
|
+
connection.add_column table_name, :status, :string, :default => 'idling'
|
10
|
+
end
|
11
|
+
@machine = StateMachines::Machine.new(@model, :status, :initial => :parked)
|
12
|
+
@record = @model.new
|
13
|
+
end
|
14
|
+
|
15
|
+
def test_should_use_machine_default
|
16
|
+
assert_equal 'parked', @record.status
|
17
|
+
end
|
18
|
+
|
19
|
+
def test_should_generate_a_warning
|
20
|
+
assert_match(/Both Foo and its :status machine have defined a different default for "status". Use only one or the other for defining defaults to avoid unexpected behaviors\./, $stderr.string)
|
21
|
+
end
|
22
|
+
|
23
|
+
def teardown
|
24
|
+
$stderr = @original_stderr
|
25
|
+
super
|
26
|
+
end
|
27
|
+
end
|
@@ -0,0 +1,29 @@
|
|
1
|
+
require_relative 'test_helper'
|
2
|
+
require 'stringio'
|
3
|
+
|
4
|
+
class MachineWithDifferentIntegerColumnDefaultTest < BaseTestCase
|
5
|
+
def setup
|
6
|
+
@original_stderr, $stderr = $stderr, StringIO.new
|
7
|
+
|
8
|
+
@model = new_model do
|
9
|
+
connection.add_column table_name, :status, :integer, :default => 0
|
10
|
+
end
|
11
|
+
@machine = StateMachines::Machine.new(@model, :status, :initial => :parked)
|
12
|
+
@machine.state :parked, :value => 1
|
13
|
+
@record = @model.new
|
14
|
+
end
|
15
|
+
|
16
|
+
def test_should_use_machine_default
|
17
|
+
assert_equal 1, @record.status
|
18
|
+
end
|
19
|
+
|
20
|
+
def test_should_generate_a_warning
|
21
|
+
assert_match(/Both Foo and its :status machine have defined a different default for "status". Use only one or the other for defining defaults to avoid unexpected behaviors\./, $stderr.string)
|
22
|
+
end
|
23
|
+
|
24
|
+
def teardown
|
25
|
+
$stderr = @original_stderr
|
26
|
+
super
|
27
|
+
end
|
28
|
+
end
|
29
|
+
|
@@ -0,0 +1,24 @@
|
|
1
|
+
require_relative 'test_helper'
|
2
|
+
|
3
|
+
class MachineWithDirtyAttributeAndCustomAttributesDuringLoopbackTest < BaseTestCase
|
4
|
+
def setup
|
5
|
+
@model = new_model do
|
6
|
+
connection.add_column table_name, :status, :string
|
7
|
+
end
|
8
|
+
@machine = StateMachines::Machine.new(@model, :status, :initial => :parked)
|
9
|
+
@machine.event :park
|
10
|
+
|
11
|
+
@record = @model.create
|
12
|
+
|
13
|
+
@transition = StateMachines::Transition.new(@record, @machine, :park, :parked, :parked)
|
14
|
+
@transition.perform(false)
|
15
|
+
end
|
16
|
+
|
17
|
+
def test_should_not_include_state_in_changed_attributes
|
18
|
+
assert_equal [], @record.changed
|
19
|
+
end
|
20
|
+
|
21
|
+
def test_should_not_track_attribute_changes
|
22
|
+
assert_equal nil, @record.changes['status']
|
23
|
+
end
|
24
|
+
end
|
@@ -0,0 +1,20 @@
|
|
1
|
+
require_relative 'test_helper'
|
2
|
+
|
3
|
+
class MachineWithDirtyAttributeAndStateEventsTest < BaseTestCase
|
4
|
+
def setup
|
5
|
+
@model = new_model
|
6
|
+
@machine = StateMachines::Machine.new(@model, :initial => :parked)
|
7
|
+
@machine.event :ignite
|
8
|
+
|
9
|
+
@record = @model.create
|
10
|
+
@record.state_event = 'ignite'
|
11
|
+
end
|
12
|
+
|
13
|
+
def test_should_not_include_state_in_changed_attributes
|
14
|
+
assert_equal [], @record.changed
|
15
|
+
end
|
16
|
+
|
17
|
+
def test_should_not_track_attribute_change
|
18
|
+
assert_equal nil, @record.changes['state']
|
19
|
+
end
|
20
|
+
end
|
@@ -0,0 +1,32 @@
|
|
1
|
+
require_relative 'test_helper'
|
2
|
+
|
3
|
+
class MachineWithDirtyAttributesAndCustomAttributeTest < BaseTestCase
|
4
|
+
def setup
|
5
|
+
@model = new_model do
|
6
|
+
connection.add_column table_name, :status, :string
|
7
|
+
end
|
8
|
+
@machine = StateMachines::Machine.new(@model, :status, :initial => :parked)
|
9
|
+
@machine.event :ignite
|
10
|
+
@machine.state :idling
|
11
|
+
|
12
|
+
@record = @model.create
|
13
|
+
|
14
|
+
@transition = StateMachines::Transition.new(@record, @machine, :ignite, :parked, :idling)
|
15
|
+
@transition.perform(false)
|
16
|
+
end
|
17
|
+
|
18
|
+
def test_should_include_state_in_changed_attributes
|
19
|
+
assert_equal %w(status), @record.changed
|
20
|
+
end
|
21
|
+
|
22
|
+
def test_should_track_attribute_change
|
23
|
+
assert_equal %w(parked idling), @record.changes['status']
|
24
|
+
end
|
25
|
+
|
26
|
+
def test_should_not_reset_changes_on_multiple_transitions
|
27
|
+
transition = StateMachines::Transition.new(@record, @machine, :ignite, :idling, :idling)
|
28
|
+
transition.perform(false)
|
29
|
+
|
30
|
+
assert_equal %w(parked idling), @record.changes['status']
|
31
|
+
end
|
32
|
+
end
|
@@ -0,0 +1,22 @@
|
|
1
|
+
require_relative 'test_helper'
|
2
|
+
|
3
|
+
class MachineWithDirtyAttributesDuringLoopbackTest < BaseTestCase
|
4
|
+
def setup
|
5
|
+
@model = new_model
|
6
|
+
@machine = StateMachines::Machine.new(@model, :initial => :parked)
|
7
|
+
@machine.event :park
|
8
|
+
|
9
|
+
@record = @model.create
|
10
|
+
|
11
|
+
@transition = StateMachines::Transition.new(@record, @machine, :park, :parked, :parked)
|
12
|
+
@transition.perform(false)
|
13
|
+
end
|
14
|
+
|
15
|
+
def test_should_not_include_state_in_changed_attributes
|
16
|
+
assert_equal [], @record.changed
|
17
|
+
end
|
18
|
+
|
19
|
+
def test_should_not_track_attribute_changes
|
20
|
+
assert_equal nil, @record.changes['state']
|
21
|
+
end
|
22
|
+
end
|
@@ -0,0 +1,35 @@
|
|
1
|
+
require_relative 'test_helper'
|
2
|
+
|
3
|
+
class MachineWithDirtyAttributesTest < BaseTestCase
|
4
|
+
def setup
|
5
|
+
@model = new_model
|
6
|
+
@machine = StateMachines::Machine.new(@model, :initial => :parked)
|
7
|
+
@machine.event :ignite
|
8
|
+
@machine.state :idling
|
9
|
+
|
10
|
+
@record = @model.create
|
11
|
+
|
12
|
+
@transition = StateMachines::Transition.new(@record, @machine, :ignite, :parked, :idling)
|
13
|
+
@transition.perform(false)
|
14
|
+
end
|
15
|
+
|
16
|
+
def test_should_include_state_in_changed_attributes
|
17
|
+
assert_equal %w(state), @record.changed
|
18
|
+
end
|
19
|
+
|
20
|
+
def test_should_track_attribute_change
|
21
|
+
assert_equal %w(parked idling), @record.changes['state']
|
22
|
+
end
|
23
|
+
|
24
|
+
def test_should_not_reset_changes_on_multiple_transitions
|
25
|
+
transition = StateMachines::Transition.new(@record, @machine, :ignite, :idling, :idling)
|
26
|
+
transition.perform(false)
|
27
|
+
|
28
|
+
assert_equal %w(parked idling), @record.changes['state']
|
29
|
+
end
|
30
|
+
|
31
|
+
def test_should_not_have_changes_when_loaded_from_database
|
32
|
+
record = @model.find(@record.id)
|
33
|
+
refute record.changed?
|
34
|
+
end
|
35
|
+
end
|
@@ -0,0 +1,99 @@
|
|
1
|
+
require_relative 'test_helper'
|
2
|
+
|
3
|
+
class MachineWithDynamicInitialStateTest < BaseTestCase
|
4
|
+
def setup
|
5
|
+
@model = new_model do
|
6
|
+
attr_accessor :value
|
7
|
+
end
|
8
|
+
@machine = StateMachines::Machine.new(@model, :initial => lambda { |object| :parked })
|
9
|
+
@machine.state :parked
|
10
|
+
end
|
11
|
+
|
12
|
+
def test_should_set_initial_state_on_created_object
|
13
|
+
record = @model.new
|
14
|
+
assert_equal 'parked', record.state
|
15
|
+
end
|
16
|
+
|
17
|
+
def test_should_still_set_attributes
|
18
|
+
record = @model.new(:value => 1)
|
19
|
+
assert_equal 1, record.value
|
20
|
+
end
|
21
|
+
|
22
|
+
def test_should_still_allow_initialize_blocks
|
23
|
+
block_args = nil
|
24
|
+
record = @model.new do |*args|
|
25
|
+
block_args = args
|
26
|
+
end
|
27
|
+
|
28
|
+
assert_equal [record], block_args
|
29
|
+
end
|
30
|
+
|
31
|
+
def test_should_set_attributes_prior_to_initialize_block
|
32
|
+
state = nil
|
33
|
+
@model.new do |record|
|
34
|
+
state = record.state
|
35
|
+
end
|
36
|
+
|
37
|
+
assert_equal 'parked', state
|
38
|
+
end
|
39
|
+
|
40
|
+
def test_should_set_attributes_prior_to_after_initialize_hook
|
41
|
+
state = nil
|
42
|
+
@model.after_initialize do |record|
|
43
|
+
state = record.state
|
44
|
+
end
|
45
|
+
@model.new
|
46
|
+
assert_equal 'parked', state
|
47
|
+
end
|
48
|
+
|
49
|
+
def test_should_set_initial_state_after_setting_attributes
|
50
|
+
@model.class_eval do
|
51
|
+
attr_accessor :state_during_setter
|
52
|
+
|
53
|
+
remove_method :value=
|
54
|
+
define_method(:value=) do |value|
|
55
|
+
self.state_during_setter = state || 'nil'
|
56
|
+
end
|
57
|
+
end
|
58
|
+
|
59
|
+
record = @model.new(:value => 1)
|
60
|
+
assert_equal 'nil', record.state_during_setter
|
61
|
+
end
|
62
|
+
|
63
|
+
def test_should_not_set_initial_state_after_already_initialized
|
64
|
+
record = @model.new(:value => 1)
|
65
|
+
assert_equal 'parked', record.state
|
66
|
+
|
67
|
+
record.state = 'idling'
|
68
|
+
record.attributes = {}
|
69
|
+
assert_equal 'idling', record.state
|
70
|
+
end
|
71
|
+
|
72
|
+
def test_should_persist_initial_state
|
73
|
+
record = @model.new
|
74
|
+
record.save
|
75
|
+
record.reload
|
76
|
+
assert_equal 'parked', record.state
|
77
|
+
end
|
78
|
+
|
79
|
+
def test_should_persist_initial_state_on_dup
|
80
|
+
record = @model.create.dup
|
81
|
+
record.save
|
82
|
+
record.reload
|
83
|
+
assert_equal 'parked', record.state
|
84
|
+
end
|
85
|
+
|
86
|
+
def test_should_use_stored_values_when_loading_from_database
|
87
|
+
@machine.state :idling
|
88
|
+
|
89
|
+
record = @model.find(@model.create(:state => 'idling').id)
|
90
|
+
assert_equal 'idling', record.state
|
91
|
+
end
|
92
|
+
|
93
|
+
def test_should_use_stored_values_when_loading_from_database_with_nil_state
|
94
|
+
@machine.state nil
|
95
|
+
|
96
|
+
record = @model.find(@model.create(:state => nil).id)
|
97
|
+
assert_nil record.state
|
98
|
+
end
|
99
|
+
end
|
@@ -0,0 +1,48 @@
|
|
1
|
+
require_relative 'test_helper'
|
2
|
+
|
3
|
+
class MachineWithEventAttributesOnAutosaveTest < BaseTestCase
|
4
|
+
def setup
|
5
|
+
@vehicle_model = new_model(:vehicle) do
|
6
|
+
connection.add_column table_name, :owner_id, :integer
|
7
|
+
end
|
8
|
+
MachineWithEventAttributesOnAutosaveTest.const_set('Vehicle', @vehicle_model)
|
9
|
+
|
10
|
+
@owner_model = new_model(:owner)
|
11
|
+
MachineWithEventAttributesOnAutosaveTest.const_set('Owner', @owner_model)
|
12
|
+
|
13
|
+
machine = StateMachines::Machine.new(@vehicle_model)
|
14
|
+
machine.event :ignite do
|
15
|
+
transition :parked => :idling
|
16
|
+
end
|
17
|
+
|
18
|
+
@owner = @owner_model.create
|
19
|
+
@vehicle = @vehicle_model.create(:state => 'parked', :owner_id => @owner.id)
|
20
|
+
end
|
21
|
+
|
22
|
+
def test_should_persist_has_one_autosave
|
23
|
+
@owner_model.has_one :vehicle, :class_name => 'MachineWithEventAttributesOnAutosaveTest::Vehicle', :autosave => true
|
24
|
+
@owner.vehicle.state_event = 'ignite'
|
25
|
+
@owner.save
|
26
|
+
|
27
|
+
@vehicle.reload
|
28
|
+
assert_equal 'idling', @vehicle.state
|
29
|
+
end
|
30
|
+
|
31
|
+
def test_should_persist_has_many_autosave
|
32
|
+
@owner_model.has_many :vehicles, :class_name => 'MachineWithEventAttributesOnAutosaveTest::Vehicle', :autosave => true
|
33
|
+
@owner.vehicles[0].state_event = 'ignite'
|
34
|
+
@owner.save
|
35
|
+
|
36
|
+
@vehicle.reload
|
37
|
+
assert_equal 'idling', @vehicle.state
|
38
|
+
end
|
39
|
+
|
40
|
+
def teardown
|
41
|
+
MachineWithEventAttributesOnAutosaveTest.class_eval do
|
42
|
+
remove_const('Vehicle')
|
43
|
+
remove_const('Owner')
|
44
|
+
end
|
45
|
+
ActiveSupport::Dependencies.clear if defined?(ActiveSupport::Dependencies)
|
46
|
+
super
|
47
|
+
end
|
48
|
+
end
|