enum_state_machine 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 +9 -0
- data/.rvmrc +1 -0
- data/Appraisals +28 -0
- data/CHANGELOG.md +0 -0
- data/Gemfile +4 -0
- data/LICENSE +23 -0
- data/README.md +4 -0
- data/Rakefile +43 -0
- data/enum_state_machine.gemspec +25 -0
- data/gemfiles/active_model_4.0.4.gemfile +9 -0
- data/gemfiles/active_model_4.0.4.gemfile.lock +51 -0
- data/gemfiles/active_record_4.0.4.gemfile +11 -0
- data/gemfiles/active_record_4.0.4.gemfile.lock +61 -0
- data/gemfiles/default.gemfile +7 -0
- data/gemfiles/default.gemfile.lock +27 -0
- data/gemfiles/graphviz_1.0.9.gemfile +7 -0
- data/gemfiles/graphviz_1.0.9.gemfile.lock +30 -0
- data/lib/enum_state_machine/assertions.rb +36 -0
- data/lib/enum_state_machine/branch.rb +225 -0
- data/lib/enum_state_machine/callback.rb +232 -0
- data/lib/enum_state_machine/core.rb +12 -0
- data/lib/enum_state_machine/core_ext/class/state_machine.rb +5 -0
- data/lib/enum_state_machine/core_ext.rb +2 -0
- data/lib/enum_state_machine/error.rb +13 -0
- data/lib/enum_state_machine/eval_helpers.rb +87 -0
- data/lib/enum_state_machine/event.rb +257 -0
- data/lib/enum_state_machine/event_collection.rb +141 -0
- data/lib/enum_state_machine/extensions.rb +149 -0
- data/lib/enum_state_machine/graph.rb +92 -0
- data/lib/enum_state_machine/helper_module.rb +17 -0
- data/lib/enum_state_machine/initializers/rails.rb +22 -0
- data/lib/enum_state_machine/initializers.rb +4 -0
- data/lib/enum_state_machine/integrations/active_model/locale.rb +11 -0
- data/lib/enum_state_machine/integrations/active_model/observer.rb +33 -0
- data/lib/enum_state_machine/integrations/active_model/observer_update.rb +42 -0
- data/lib/enum_state_machine/integrations/active_model/versions.rb +31 -0
- data/lib/enum_state_machine/integrations/active_model.rb +585 -0
- data/lib/enum_state_machine/integrations/active_record/locale.rb +20 -0
- data/lib/enum_state_machine/integrations/active_record/versions.rb +123 -0
- data/lib/enum_state_machine/integrations/active_record.rb +548 -0
- data/lib/enum_state_machine/integrations/base.rb +100 -0
- data/lib/enum_state_machine/integrations.rb +97 -0
- data/lib/enum_state_machine/machine.rb +2292 -0
- data/lib/enum_state_machine/machine_collection.rb +86 -0
- data/lib/enum_state_machine/macro_methods.rb +518 -0
- data/lib/enum_state_machine/matcher.rb +123 -0
- data/lib/enum_state_machine/matcher_helpers.rb +54 -0
- data/lib/enum_state_machine/node_collection.rb +222 -0
- data/lib/enum_state_machine/path.rb +120 -0
- data/lib/enum_state_machine/path_collection.rb +90 -0
- data/lib/enum_state_machine/state.rb +297 -0
- data/lib/enum_state_machine/state_collection.rb +112 -0
- data/lib/enum_state_machine/state_context.rb +138 -0
- data/lib/enum_state_machine/state_enum.rb +23 -0
- data/lib/enum_state_machine/transition.rb +470 -0
- data/lib/enum_state_machine/transition_collection.rb +245 -0
- data/lib/enum_state_machine/version.rb +3 -0
- data/lib/enum_state_machine/yard/handlers/base.rb +32 -0
- data/lib/enum_state_machine/yard/handlers/event.rb +25 -0
- data/lib/enum_state_machine/yard/handlers/machine.rb +344 -0
- data/lib/enum_state_machine/yard/handlers/state.rb +25 -0
- data/lib/enum_state_machine/yard/handlers/transition.rb +47 -0
- data/lib/enum_state_machine/yard/handlers.rb +12 -0
- data/lib/enum_state_machine/yard/templates/default/class/html/setup.rb +30 -0
- data/lib/enum_state_machine/yard/templates/default/class/html/state_machines.erb +12 -0
- data/lib/enum_state_machine/yard/templates.rb +3 -0
- data/lib/enum_state_machine/yard.rb +8 -0
- data/lib/enum_state_machine.rb +9 -0
- data/lib/tasks/enum_state_machine.rake +1 -0
- data/lib/tasks/enum_state_machine.rb +24 -0
- data/lib/yard-enum_state_machine.rb +2 -0
- data/test/files/en.yml +9 -0
- data/test/files/switch.rb +15 -0
- data/test/functional/state_machine_test.rb +1066 -0
- data/test/test_helper.rb +7 -0
- data/test/unit/assertions_test.rb +40 -0
- data/test/unit/branch_test.rb +969 -0
- data/test/unit/callback_test.rb +704 -0
- data/test/unit/error_test.rb +43 -0
- data/test/unit/eval_helpers_test.rb +270 -0
- data/test/unit/event_collection_test.rb +398 -0
- data/test/unit/event_test.rb +1196 -0
- data/test/unit/graph_test.rb +98 -0
- data/test/unit/helper_module_test.rb +17 -0
- data/test/unit/integrations/active_model_test.rb +1245 -0
- data/test/unit/integrations/active_record_test.rb +2551 -0
- data/test/unit/integrations/base_test.rb +104 -0
- data/test/unit/integrations_test.rb +71 -0
- data/test/unit/invalid_event_test.rb +20 -0
- data/test/unit/invalid_parallel_transition_test.rb +18 -0
- data/test/unit/invalid_transition_test.rb +115 -0
- data/test/unit/machine_collection_test.rb +603 -0
- data/test/unit/machine_test.rb +3395 -0
- data/test/unit/matcher_helpers_test.rb +37 -0
- data/test/unit/matcher_test.rb +155 -0
- data/test/unit/node_collection_test.rb +362 -0
- data/test/unit/path_collection_test.rb +266 -0
- data/test/unit/path_test.rb +485 -0
- data/test/unit/state_collection_test.rb +352 -0
- data/test/unit/state_context_test.rb +441 -0
- data/test/unit/state_enum_test.rb +50 -0
- data/test/unit/state_machine_test.rb +31 -0
- data/test/unit/state_test.rb +1101 -0
- data/test/unit/transition_collection_test.rb +2168 -0
- data/test/unit/transition_test.rb +1558 -0
- metadata +249 -0
checksums.yaml
ADDED
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
---
|
|
2
|
+
SHA1:
|
|
3
|
+
metadata.gz: d195b0469b462e5fdb5257d7c552159dd81b9584
|
|
4
|
+
data.tar.gz: 9823967bda0300a02fad53e1b4db6e70b39f2cc6
|
|
5
|
+
SHA512:
|
|
6
|
+
metadata.gz: 0f8186afaf20c300450f3c494d069772b3fd2ba9282db5f386f651d3ac118d9a504eaa45662d66a4e5a2b9d61e26daeaf046547a2ffea7272dc0cbb7023d99bc
|
|
7
|
+
data.tar.gz: 1c6b94a1e170edeaa89af462f41df0b1665adf12d23730825b325cab3262f7a5f7ca45562c0bd2092aa27d43589d3353558786bed07f3f49d4c3d542ffa1b7aa
|
data/.gitignore
ADDED
data/.rvmrc
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
rvm use 2.0.0@enum_state_machine --create
|
data/Appraisals
ADDED
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
appraise "default" do
|
|
2
|
+
end
|
|
3
|
+
|
|
4
|
+
# GraphViz
|
|
5
|
+
appraise "graphviz_1.0.9" do
|
|
6
|
+
gem "ruby-graphviz", "~> 1.0.9"
|
|
7
|
+
end
|
|
8
|
+
|
|
9
|
+
# ActiveRecord integrations
|
|
10
|
+
if RUBY_VERSION > "1.9.2"
|
|
11
|
+
appraise "active_record_4.0.4" do
|
|
12
|
+
gem "sqlite3", "~> 1.3.9"
|
|
13
|
+
gem "activerecord", "~> 4.0.4"
|
|
14
|
+
gem "activerecord-deprecated_finders", "~> 1.0.3"
|
|
15
|
+
gem "protected_attributes", "~> 1.0.7"
|
|
16
|
+
gem "rails-observers", "~> 0.1.2"
|
|
17
|
+
end
|
|
18
|
+
end
|
|
19
|
+
|
|
20
|
+
# ActiveModel integrations
|
|
21
|
+
if RUBY_VERSION > "1.9.2"
|
|
22
|
+
appraise "active_model_4.0.4" do
|
|
23
|
+
gem "activemodel", "~> 4.0.4"
|
|
24
|
+
gem "rails-observers", "~> 0.1.2"
|
|
25
|
+
gem "protected_attributes", "~> 1.0.7"
|
|
26
|
+
end
|
|
27
|
+
end
|
|
28
|
+
|
data/CHANGELOG.md
ADDED
|
File without changes
|
data/Gemfile
ADDED
data/LICENSE
ADDED
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
Copyright (c) 2006-2012 Aaron Pfeifer
|
|
2
|
+
PowerEnum integration Copyright (c) 2014 Horns and Hooves team.
|
|
3
|
+
|
|
4
|
+
Permission is hereby granted, free of charge, to any person obtaining
|
|
5
|
+
a copy of this software and associated documentation files (the
|
|
6
|
+
"Software"), to deal in the Software without restriction, including
|
|
7
|
+
without limitation the rights to use, copy, modify, merge, publish,
|
|
8
|
+
distribute, sublicense, and/or sell copies of the Software, and to
|
|
9
|
+
permit persons to whom the Software is furnished to do so, subject to
|
|
10
|
+
the following conditions:
|
|
11
|
+
|
|
12
|
+
The above copyright notice and this permission notice shall be
|
|
13
|
+
included in all copies or substantial portions of the Software.
|
|
14
|
+
|
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
16
|
+
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
17
|
+
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
18
|
+
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
19
|
+
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
20
|
+
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
21
|
+
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
22
|
+
|
|
23
|
+
|
data/README.md
ADDED
data/Rakefile
ADDED
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
require 'rubygems'
|
|
2
|
+
require 'bundler'
|
|
3
|
+
Bundler.setup
|
|
4
|
+
|
|
5
|
+
require 'rake'
|
|
6
|
+
require 'rake/testtask'
|
|
7
|
+
|
|
8
|
+
require 'appraisal'
|
|
9
|
+
|
|
10
|
+
desc 'Default: run all tests.'
|
|
11
|
+
task :default => :test
|
|
12
|
+
|
|
13
|
+
desc "Test enum_state_machine."
|
|
14
|
+
Rake::TestTask.new(:test) do |t|
|
|
15
|
+
integration = %w(active_model active_record).detect do |name|
|
|
16
|
+
Bundler.default_gemfile.to_s.include?(name)
|
|
17
|
+
end
|
|
18
|
+
|
|
19
|
+
t.libs << 'lib'
|
|
20
|
+
t.test_files = integration ? Dir["test/unit/integrations/#{integration}_test.rb"] : Dir['test/{functional,unit}/*_test.rb'] + ['test/unit/integrations/base_test.rb']
|
|
21
|
+
t.verbose = true
|
|
22
|
+
t.warning = true if ENV['WARNINGS']
|
|
23
|
+
end
|
|
24
|
+
|
|
25
|
+
namespace :appraisal do
|
|
26
|
+
desc "Run the given task for a particular integration's appraisals"
|
|
27
|
+
task :integration do
|
|
28
|
+
integration = ENV['INTEGRATION']
|
|
29
|
+
|
|
30
|
+
Appraisal::File.each do |appraisal|
|
|
31
|
+
if appraisal.name.include?(integration)
|
|
32
|
+
appraisal.install
|
|
33
|
+
Appraisal::Command.from_args(appraisal.gemfile_path).run
|
|
34
|
+
end
|
|
35
|
+
end
|
|
36
|
+
|
|
37
|
+
exit
|
|
38
|
+
end
|
|
39
|
+
end
|
|
40
|
+
|
|
41
|
+
load File.dirname(__FILE__) + '/lib/tasks/enum_state_machine.rake'
|
|
42
|
+
|
|
43
|
+
Bundler::GemHelper.install_tasks
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
$LOAD_PATH.unshift File.expand_path('../lib', __FILE__)
|
|
2
|
+
require 'enum_state_machine/version'
|
|
3
|
+
|
|
4
|
+
Gem::Specification.new do |s|
|
|
5
|
+
s.name = "enum_state_machine"
|
|
6
|
+
s.version = EnumStateMachine::VERSION
|
|
7
|
+
s.authors = ["The HornsAndHooves Team"]
|
|
8
|
+
s.email = ["arthur.shagall@gmail.com"]
|
|
9
|
+
s.homepage = "https://github.com/HornsAndHooves/enum_state_machine"
|
|
10
|
+
s.description = "Adds support for creating enum state machines for attributes on any Ruby class"
|
|
11
|
+
s.summary = "Enum State machines for attributes"
|
|
12
|
+
s.require_paths = ["lib"]
|
|
13
|
+
ignores = File.read(".gitignore").split.map {|i| i.sub(/\/$/, "/*").sub(/^[^\/]/, "**/\\0")}
|
|
14
|
+
s.files = (Dir[".*"] + Dir["**/*"]).select {|f| File.file?(f) && !ignores.any? {|i| File.fnmatch(i, "/#{f}")}}
|
|
15
|
+
s.test_files = s.files.grep(/^test\//)
|
|
16
|
+
s.rdoc_options = %w(--line-numbers --inline-source --title enum_state_machine --main README.md)
|
|
17
|
+
s.extra_rdoc_files = %w(README.md CHANGELOG.md LICENSE)
|
|
18
|
+
s.license = 'MIT'
|
|
19
|
+
|
|
20
|
+
s.add_dependency("power_enum", "~> 2.4")
|
|
21
|
+
|
|
22
|
+
s.add_development_dependency("rake")
|
|
23
|
+
s.add_development_dependency("simplecov")
|
|
24
|
+
s.add_development_dependency("appraisal", "~> 0.5.0")
|
|
25
|
+
end
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
PATH
|
|
2
|
+
remote: ../
|
|
3
|
+
specs:
|
|
4
|
+
enum_state_machine (1.2.0)
|
|
5
|
+
|
|
6
|
+
GEM
|
|
7
|
+
remote: https://www.rubygems.org/
|
|
8
|
+
specs:
|
|
9
|
+
activemodel (4.0.4)
|
|
10
|
+
activesupport (= 4.0.4)
|
|
11
|
+
builder (~> 3.1.0)
|
|
12
|
+
activesupport (4.0.4)
|
|
13
|
+
i18n (~> 0.6, >= 0.6.9)
|
|
14
|
+
minitest (~> 4.2)
|
|
15
|
+
multi_json (~> 1.3)
|
|
16
|
+
thread_safe (~> 0.1)
|
|
17
|
+
tzinfo (~> 0.3.37)
|
|
18
|
+
appraisal (0.5.2)
|
|
19
|
+
bundler
|
|
20
|
+
rake
|
|
21
|
+
atomic (1.1.16)
|
|
22
|
+
builder (3.1.4)
|
|
23
|
+
docile (1.1.3)
|
|
24
|
+
i18n (0.6.9)
|
|
25
|
+
minitest (4.7.5)
|
|
26
|
+
multi_json (1.9.2)
|
|
27
|
+
protected_attributes (1.0.7)
|
|
28
|
+
activemodel (>= 4.0.1, < 5.0)
|
|
29
|
+
rails-observers (0.1.2)
|
|
30
|
+
activemodel (~> 4.0)
|
|
31
|
+
rake (10.2.2)
|
|
32
|
+
simplecov (0.8.2)
|
|
33
|
+
docile (~> 1.1.0)
|
|
34
|
+
multi_json
|
|
35
|
+
simplecov-html (~> 0.8.0)
|
|
36
|
+
simplecov-html (0.8.0)
|
|
37
|
+
thread_safe (0.3.1)
|
|
38
|
+
atomic (>= 1.1.7, < 2)
|
|
39
|
+
tzinfo (0.3.39)
|
|
40
|
+
|
|
41
|
+
PLATFORMS
|
|
42
|
+
ruby
|
|
43
|
+
|
|
44
|
+
DEPENDENCIES
|
|
45
|
+
activemodel (~> 4.0.4)
|
|
46
|
+
appraisal (~> 0.5.0)
|
|
47
|
+
enum_state_machine!
|
|
48
|
+
protected_attributes (~> 1.0.7)
|
|
49
|
+
rails-observers (~> 0.1.2)
|
|
50
|
+
rake
|
|
51
|
+
simplecov
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
# This file was generated by Appraisal
|
|
2
|
+
|
|
3
|
+
source "https://www.rubygems.org"
|
|
4
|
+
|
|
5
|
+
gem "sqlite3", "~> 1.3.9"
|
|
6
|
+
gem "activerecord", "~> 4.0.4"
|
|
7
|
+
gem "activerecord-deprecated_finders", "~> 1.0.3"
|
|
8
|
+
gem "protected_attributes", "~> 1.0.7"
|
|
9
|
+
gem "rails-observers", "~> 0.1.2"
|
|
10
|
+
|
|
11
|
+
gemspec :path=>"../"
|
|
@@ -0,0 +1,61 @@
|
|
|
1
|
+
PATH
|
|
2
|
+
remote: ../
|
|
3
|
+
specs:
|
|
4
|
+
enum_state_machine (1.2.0)
|
|
5
|
+
|
|
6
|
+
GEM
|
|
7
|
+
remote: https://www.rubygems.org/
|
|
8
|
+
specs:
|
|
9
|
+
activemodel (4.0.4)
|
|
10
|
+
activesupport (= 4.0.4)
|
|
11
|
+
builder (~> 3.1.0)
|
|
12
|
+
activerecord (4.0.4)
|
|
13
|
+
activemodel (= 4.0.4)
|
|
14
|
+
activerecord-deprecated_finders (~> 1.0.2)
|
|
15
|
+
activesupport (= 4.0.4)
|
|
16
|
+
arel (~> 4.0.0)
|
|
17
|
+
activerecord-deprecated_finders (1.0.3)
|
|
18
|
+
activesupport (4.0.4)
|
|
19
|
+
i18n (~> 0.6, >= 0.6.9)
|
|
20
|
+
minitest (~> 4.2)
|
|
21
|
+
multi_json (~> 1.3)
|
|
22
|
+
thread_safe (~> 0.1)
|
|
23
|
+
tzinfo (~> 0.3.37)
|
|
24
|
+
appraisal (0.5.2)
|
|
25
|
+
bundler
|
|
26
|
+
rake
|
|
27
|
+
arel (4.0.2)
|
|
28
|
+
atomic (1.1.16)
|
|
29
|
+
builder (3.1.4)
|
|
30
|
+
docile (1.1.3)
|
|
31
|
+
i18n (0.6.9)
|
|
32
|
+
minitest (4.7.5)
|
|
33
|
+
multi_json (1.9.2)
|
|
34
|
+
protected_attributes (1.0.7)
|
|
35
|
+
activemodel (>= 4.0.1, < 5.0)
|
|
36
|
+
rails-observers (0.1.2)
|
|
37
|
+
activemodel (~> 4.0)
|
|
38
|
+
rake (10.2.2)
|
|
39
|
+
simplecov (0.8.2)
|
|
40
|
+
docile (~> 1.1.0)
|
|
41
|
+
multi_json
|
|
42
|
+
simplecov-html (~> 0.8.0)
|
|
43
|
+
simplecov-html (0.8.0)
|
|
44
|
+
sqlite3 (1.3.9)
|
|
45
|
+
thread_safe (0.3.1)
|
|
46
|
+
atomic (>= 1.1.7, < 2)
|
|
47
|
+
tzinfo (0.3.39)
|
|
48
|
+
|
|
49
|
+
PLATFORMS
|
|
50
|
+
ruby
|
|
51
|
+
|
|
52
|
+
DEPENDENCIES
|
|
53
|
+
activerecord (~> 4.0.4)
|
|
54
|
+
activerecord-deprecated_finders (~> 1.0.3)
|
|
55
|
+
appraisal (~> 0.5.0)
|
|
56
|
+
enum_state_machine!
|
|
57
|
+
protected_attributes (~> 1.0.7)
|
|
58
|
+
rails-observers (~> 0.1.2)
|
|
59
|
+
rake
|
|
60
|
+
simplecov
|
|
61
|
+
sqlite3 (~> 1.3.9)
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
PATH
|
|
2
|
+
remote: ../
|
|
3
|
+
specs:
|
|
4
|
+
enum_state_machine (1.2.0)
|
|
5
|
+
|
|
6
|
+
GEM
|
|
7
|
+
remote: https://www.rubygems.org/
|
|
8
|
+
specs:
|
|
9
|
+
appraisal (0.5.1)
|
|
10
|
+
bundler
|
|
11
|
+
rake
|
|
12
|
+
multi_json (1.0.4)
|
|
13
|
+
rake (0.9.2.2)
|
|
14
|
+
simplecov (0.5.4)
|
|
15
|
+
multi_json (~> 1.0.3)
|
|
16
|
+
simplecov-html (~> 0.5.3)
|
|
17
|
+
simplecov-html (0.5.3)
|
|
18
|
+
|
|
19
|
+
PLATFORMS
|
|
20
|
+
java
|
|
21
|
+
ruby
|
|
22
|
+
|
|
23
|
+
DEPENDENCIES
|
|
24
|
+
appraisal (~> 0.5.0)
|
|
25
|
+
enum_state_machine!
|
|
26
|
+
rake
|
|
27
|
+
simplecov
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
PATH
|
|
2
|
+
remote: ../
|
|
3
|
+
specs:
|
|
4
|
+
enum_state_machine (1.2.0)
|
|
5
|
+
|
|
6
|
+
GEM
|
|
7
|
+
remote: https://www.rubygems.org/
|
|
8
|
+
specs:
|
|
9
|
+
appraisal (0.5.2)
|
|
10
|
+
bundler
|
|
11
|
+
rake
|
|
12
|
+
docile (1.1.3)
|
|
13
|
+
multi_json (1.9.2)
|
|
14
|
+
rake (10.2.2)
|
|
15
|
+
ruby-graphviz (1.0.9)
|
|
16
|
+
simplecov (0.8.2)
|
|
17
|
+
docile (~> 1.1.0)
|
|
18
|
+
multi_json
|
|
19
|
+
simplecov-html (~> 0.8.0)
|
|
20
|
+
simplecov-html (0.8.0)
|
|
21
|
+
|
|
22
|
+
PLATFORMS
|
|
23
|
+
ruby
|
|
24
|
+
|
|
25
|
+
DEPENDENCIES
|
|
26
|
+
appraisal (~> 0.5.0)
|
|
27
|
+
enum_state_machine!
|
|
28
|
+
rake
|
|
29
|
+
ruby-graphviz (~> 1.0.9)
|
|
30
|
+
simplecov
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
module EnumStateMachine
|
|
2
|
+
# Provides a set of helper methods for making assertions about the content
|
|
3
|
+
# of various objects
|
|
4
|
+
module Assertions
|
|
5
|
+
# Validates that the given hash *only* includes the specified valid keys.
|
|
6
|
+
# If any invalid keys are found, an ArgumentError will be raised.
|
|
7
|
+
#
|
|
8
|
+
# == Examples
|
|
9
|
+
#
|
|
10
|
+
# options = {:name => 'John Smith', :age => 30}
|
|
11
|
+
#
|
|
12
|
+
# assert_valid_keys(options, :name) # => ArgumentError: Invalid key(s): age
|
|
13
|
+
# assert_valid_keys(options, 'name', 'age') # => ArgumentError: Invalid key(s): age, name
|
|
14
|
+
# assert_valid_keys(options, :name, :age) # => nil
|
|
15
|
+
def assert_valid_keys(hash, *valid_keys)
|
|
16
|
+
invalid_keys = hash.keys - valid_keys
|
|
17
|
+
raise ArgumentError, "Invalid key(s): #{invalid_keys.join(', ')}" unless invalid_keys.empty?
|
|
18
|
+
end
|
|
19
|
+
|
|
20
|
+
# Validates that the given hash only includes at *most* one of a set of
|
|
21
|
+
# exclusive keys. If more than one key is found, an ArgumentError will be
|
|
22
|
+
# raised.
|
|
23
|
+
#
|
|
24
|
+
# == Examples
|
|
25
|
+
#
|
|
26
|
+
# options = {:only => :on, :except => :off}
|
|
27
|
+
# assert_exclusive_keys(options, :only) # => nil
|
|
28
|
+
# assert_exclusive_keys(options, :except) # => nil
|
|
29
|
+
# assert_exclusive_keys(options, :only, :except) # => ArgumentError: Conflicting keys: only, except
|
|
30
|
+
# assert_exclusive_keys(options, :only, :except, :with) # => ArgumentError: Conflicting keys: only, except
|
|
31
|
+
def assert_exclusive_keys(hash, *exclusive_keys)
|
|
32
|
+
conflicting_keys = exclusive_keys & hash.keys
|
|
33
|
+
raise ArgumentError, "Conflicting keys: #{conflicting_keys.join(', ')}" unless conflicting_keys.length <= 1
|
|
34
|
+
end
|
|
35
|
+
end
|
|
36
|
+
end
|
|
@@ -0,0 +1,225 @@
|
|
|
1
|
+
require 'enum_state_machine/matcher'
|
|
2
|
+
require 'enum_state_machine/eval_helpers'
|
|
3
|
+
require 'enum_state_machine/assertions'
|
|
4
|
+
|
|
5
|
+
module EnumStateMachine
|
|
6
|
+
# Represents a set of requirements that must be met in order for a transition
|
|
7
|
+
# or callback to occur. Branches verify that the event, from state, and to
|
|
8
|
+
# state of the transition match, in addition to if/unless conditionals for
|
|
9
|
+
# an object's state.
|
|
10
|
+
class Branch
|
|
11
|
+
include Assertions
|
|
12
|
+
include EvalHelpers
|
|
13
|
+
|
|
14
|
+
# The condition that must be met on an object
|
|
15
|
+
attr_reader :if_condition
|
|
16
|
+
|
|
17
|
+
# The condition that must *not* be met on an object
|
|
18
|
+
attr_reader :unless_condition
|
|
19
|
+
|
|
20
|
+
# The requirement for verifying the event being matched
|
|
21
|
+
attr_reader :event_requirement
|
|
22
|
+
|
|
23
|
+
# One or more requirements for verifying the states being matched. All
|
|
24
|
+
# requirements contain a mapping of {:from => matcher, :to => matcher}.
|
|
25
|
+
attr_reader :state_requirements
|
|
26
|
+
|
|
27
|
+
# A list of all of the states known to this branch. This will pull states
|
|
28
|
+
# from the following options (in the same order):
|
|
29
|
+
# * +from+ / +except_from+
|
|
30
|
+
# * +to+ / +except_to+
|
|
31
|
+
attr_reader :known_states
|
|
32
|
+
|
|
33
|
+
# Creates a new branch
|
|
34
|
+
def initialize(options = {}) #:nodoc:
|
|
35
|
+
# Build conditionals
|
|
36
|
+
@if_condition = options.delete(:if)
|
|
37
|
+
@unless_condition = options.delete(:unless)
|
|
38
|
+
|
|
39
|
+
# Build event requirement
|
|
40
|
+
@event_requirement = build_matcher(options, :on, :except_on)
|
|
41
|
+
|
|
42
|
+
if (options.keys - [:from, :to, :on, :except_from, :except_to, :except_on]).empty?
|
|
43
|
+
# Explicit from/to requirements specified
|
|
44
|
+
@state_requirements = [{:from => build_matcher(options, :from, :except_from), :to => build_matcher(options, :to, :except_to)}]
|
|
45
|
+
else
|
|
46
|
+
# Separate out the event requirement
|
|
47
|
+
options.delete(:on)
|
|
48
|
+
options.delete(:except_on)
|
|
49
|
+
|
|
50
|
+
# Implicit from/to requirements specified
|
|
51
|
+
@state_requirements = options.collect do |from, to|
|
|
52
|
+
from = WhitelistMatcher.new(from) unless from.is_a?(Matcher)
|
|
53
|
+
to = WhitelistMatcher.new(to) unless to.is_a?(Matcher)
|
|
54
|
+
{:from => from, :to => to}
|
|
55
|
+
end
|
|
56
|
+
end
|
|
57
|
+
|
|
58
|
+
# Track known states. The order that requirements are iterated is based
|
|
59
|
+
# on the priority in which tracked states should be added.
|
|
60
|
+
@known_states = []
|
|
61
|
+
@state_requirements.each do |state_requirement|
|
|
62
|
+
[:from, :to].each {|option| @known_states |= state_requirement[option].values}
|
|
63
|
+
end
|
|
64
|
+
end
|
|
65
|
+
|
|
66
|
+
# Determines whether the given object / query matches the requirements
|
|
67
|
+
# configured for this branch. In addition to matching the event, from state,
|
|
68
|
+
# and to state, this will also check whether the configured :if/:unless
|
|
69
|
+
# conditions pass on the given object.
|
|
70
|
+
#
|
|
71
|
+
# == Examples
|
|
72
|
+
#
|
|
73
|
+
# branch = EnumStateMachine::Branch.new(:parked => :idling, :on => :ignite)
|
|
74
|
+
#
|
|
75
|
+
# # Successful
|
|
76
|
+
# branch.matches?(object, :on => :ignite) # => true
|
|
77
|
+
# branch.matches?(object, :from => nil) # => true
|
|
78
|
+
# branch.matches?(object, :from => :parked) # => true
|
|
79
|
+
# branch.matches?(object, :to => :idling) # => true
|
|
80
|
+
# branch.matches?(object, :from => :parked, :to => :idling) # => true
|
|
81
|
+
# branch.matches?(object, :on => :ignite, :from => :parked, :to => :idling) # => true
|
|
82
|
+
#
|
|
83
|
+
# # Unsuccessful
|
|
84
|
+
# branch.matches?(object, :on => :park) # => false
|
|
85
|
+
# branch.matches?(object, :from => :idling) # => false
|
|
86
|
+
# branch.matches?(object, :to => :first_gear) # => false
|
|
87
|
+
# branch.matches?(object, :from => :parked, :to => :first_gear) # => false
|
|
88
|
+
# branch.matches?(object, :on => :park, :from => :parked, :to => :idling) # => false
|
|
89
|
+
def matches?(object, query = {})
|
|
90
|
+
!match(object, query).nil?
|
|
91
|
+
end
|
|
92
|
+
|
|
93
|
+
# Attempts to match the given object / query against the set of requirements
|
|
94
|
+
# configured for this branch. In addition to matching the event, from state,
|
|
95
|
+
# and to state, this will also check whether the configured :if/:unless
|
|
96
|
+
# conditions pass on the given object.
|
|
97
|
+
#
|
|
98
|
+
# If a match is found, then the event/state requirements that the query
|
|
99
|
+
# passed successfully will be returned. Otherwise, nil is returned if there
|
|
100
|
+
# was no match.
|
|
101
|
+
#
|
|
102
|
+
# Query options:
|
|
103
|
+
# * <tt>:from</tt> - One or more states being transitioned from. If none
|
|
104
|
+
# are specified, then this will always match.
|
|
105
|
+
# * <tt>:to</tt> - One or more states being transitioned to. If none are
|
|
106
|
+
# specified, then this will always match.
|
|
107
|
+
# * <tt>:on</tt> - One or more events that fired the transition. If none
|
|
108
|
+
# are specified, then this will always match.
|
|
109
|
+
# * <tt>:guard</tt> - Whether to guard matches with the if/unless
|
|
110
|
+
# conditionals defined for this branch. Default is true.
|
|
111
|
+
#
|
|
112
|
+
# == Examples
|
|
113
|
+
#
|
|
114
|
+
# branch = EnumStateMachine::Branch.new(:parked => :idling, :on => :ignite)
|
|
115
|
+
#
|
|
116
|
+
# branch.match(object, :on => :ignite) # => {:to => ..., :from => ..., :on => ...}
|
|
117
|
+
# branch.match(object, :on => :park) # => nil
|
|
118
|
+
def match(object, query = {})
|
|
119
|
+
assert_valid_keys(query, :from, :to, :on, :guard)
|
|
120
|
+
|
|
121
|
+
if (match = match_query(query)) && matches_conditions?(object, query)
|
|
122
|
+
match
|
|
123
|
+
end
|
|
124
|
+
end
|
|
125
|
+
|
|
126
|
+
# Draws a representation of this branch on the given graph. This will draw
|
|
127
|
+
# an edge between every state this branch matches *from* to either the
|
|
128
|
+
# configured to state or, if none specified, then a loopback to the from
|
|
129
|
+
# state.
|
|
130
|
+
#
|
|
131
|
+
# For example, if the following from states are configured:
|
|
132
|
+
# * +idling+
|
|
133
|
+
# * +first_gear+
|
|
134
|
+
# * +backing_up+
|
|
135
|
+
#
|
|
136
|
+
# ...and the to state is +parked+, then the following edges will be created:
|
|
137
|
+
# * +idling+ -> +parked+
|
|
138
|
+
# * +first_gear+ -> +parked+
|
|
139
|
+
# * +backing_up+ -> +parked+
|
|
140
|
+
#
|
|
141
|
+
# Each edge will be labeled with the name of the event that would cause the
|
|
142
|
+
# transition.
|
|
143
|
+
def draw(graph, event, valid_states)
|
|
144
|
+
state_requirements.each do |state_requirement|
|
|
145
|
+
# From states determined based on the known valid states
|
|
146
|
+
from_states = state_requirement[:from].filter(valid_states)
|
|
147
|
+
|
|
148
|
+
# If a to state is not specified, then it's a loopback and each from
|
|
149
|
+
# state maps back to itself
|
|
150
|
+
if state_requirement[:to].values.empty?
|
|
151
|
+
loopback = true
|
|
152
|
+
else
|
|
153
|
+
to_state = state_requirement[:to].values.first
|
|
154
|
+
to_state = to_state ? to_state.to_s : 'nil'
|
|
155
|
+
loopback = false
|
|
156
|
+
end
|
|
157
|
+
|
|
158
|
+
# Generate an edge between each from and to state
|
|
159
|
+
from_states.each do |from_state|
|
|
160
|
+
from_state = from_state ? from_state.to_s : 'nil'
|
|
161
|
+
graph.add_edges(from_state, loopback ? from_state : to_state, :label => event.to_s)
|
|
162
|
+
end
|
|
163
|
+
end
|
|
164
|
+
|
|
165
|
+
true
|
|
166
|
+
end
|
|
167
|
+
|
|
168
|
+
protected
|
|
169
|
+
# Builds a matcher strategy to use for the given options. If neither a
|
|
170
|
+
# whitelist nor a blacklist option is specified, then an AllMatcher is
|
|
171
|
+
# built.
|
|
172
|
+
def build_matcher(options, whitelist_option, blacklist_option)
|
|
173
|
+
assert_exclusive_keys(options, whitelist_option, blacklist_option)
|
|
174
|
+
|
|
175
|
+
if options.include?(whitelist_option)
|
|
176
|
+
value = options[whitelist_option]
|
|
177
|
+
value.is_a?(Matcher) ? value : WhitelistMatcher.new(options[whitelist_option])
|
|
178
|
+
elsif options.include?(blacklist_option)
|
|
179
|
+
value = options[blacklist_option]
|
|
180
|
+
raise ArgumentError, ":#{blacklist_option} option cannot use matchers; use :#{whitelist_option} instead" if value.is_a?(Matcher)
|
|
181
|
+
BlacklistMatcher.new(value)
|
|
182
|
+
else
|
|
183
|
+
AllMatcher.instance
|
|
184
|
+
end
|
|
185
|
+
end
|
|
186
|
+
|
|
187
|
+
# Verifies that all configured requirements (event and state) match the
|
|
188
|
+
# given query. If a match is found, then a hash containing the
|
|
189
|
+
# event/state requirements that passed will be returned; otherwise, nil.
|
|
190
|
+
def match_query(query)
|
|
191
|
+
query ||= {}
|
|
192
|
+
|
|
193
|
+
if match_event(query) && (state_requirement = match_states(query))
|
|
194
|
+
state_requirement.merge(:on => event_requirement)
|
|
195
|
+
end
|
|
196
|
+
end
|
|
197
|
+
|
|
198
|
+
# Verifies that the event requirement matches the given query
|
|
199
|
+
def match_event(query)
|
|
200
|
+
matches_requirement?(query, :on, event_requirement)
|
|
201
|
+
end
|
|
202
|
+
|
|
203
|
+
# Verifies that the state requirements match the given query. If a
|
|
204
|
+
# matching requirement is found, then it is returned.
|
|
205
|
+
def match_states(query)
|
|
206
|
+
state_requirements.detect do |state_requirement|
|
|
207
|
+
[:from, :to].all? {|option| matches_requirement?(query, option, state_requirement[option])}
|
|
208
|
+
end
|
|
209
|
+
end
|
|
210
|
+
|
|
211
|
+
# Verifies that an option in the given query matches the values required
|
|
212
|
+
# for that option
|
|
213
|
+
def matches_requirement?(query, option, requirement)
|
|
214
|
+
!query.include?(option) || requirement.matches?(query[option], query)
|
|
215
|
+
end
|
|
216
|
+
|
|
217
|
+
# Verifies that the conditionals for this branch evaluate to true for the
|
|
218
|
+
# given object
|
|
219
|
+
def matches_conditions?(object, query)
|
|
220
|
+
query[:guard] == false ||
|
|
221
|
+
Array(if_condition).all? {|condition| evaluate_method(object, condition)} &&
|
|
222
|
+
!Array(unless_condition).any? {|condition| evaluate_method(object, condition)}
|
|
223
|
+
end
|
|
224
|
+
end
|
|
225
|
+
end
|