spree-state_machine 2.0.0.beta1
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 +8 -0
- data/.travis.yml +12 -0
- data/.yardopts +5 -0
- data/CHANGELOG.md +502 -0
- data/Gemfile +3 -0
- data/LICENSE +20 -0
- data/README.md +1246 -0
- data/Rakefile +20 -0
- data/examples/AutoShop_state.png +0 -0
- data/examples/Car_state.png +0 -0
- data/examples/Gemfile +5 -0
- data/examples/Gemfile.lock +14 -0
- data/examples/TrafficLight_state.png +0 -0
- data/examples/Vehicle_state.png +0 -0
- data/examples/auto_shop.rb +13 -0
- data/examples/car.rb +21 -0
- data/examples/doc/AutoShop.html +2856 -0
- data/examples/doc/AutoShop_state.png +0 -0
- data/examples/doc/Car.html +919 -0
- data/examples/doc/Car_state.png +0 -0
- data/examples/doc/TrafficLight.html +2230 -0
- data/examples/doc/TrafficLight_state.png +0 -0
- data/examples/doc/Vehicle.html +7921 -0
- data/examples/doc/Vehicle_state.png +0 -0
- data/examples/doc/_index.html +136 -0
- data/examples/doc/class_list.html +47 -0
- data/examples/doc/css/common.css +1 -0
- data/examples/doc/css/full_list.css +55 -0
- data/examples/doc/css/style.css +322 -0
- data/examples/doc/file_list.html +46 -0
- data/examples/doc/frames.html +13 -0
- data/examples/doc/index.html +136 -0
- data/examples/doc/js/app.js +205 -0
- data/examples/doc/js/full_list.js +173 -0
- data/examples/doc/js/jquery.js +16 -0
- data/examples/doc/method_list.html +734 -0
- data/examples/doc/top-level-namespace.html +105 -0
- data/examples/merb-rest/controller.rb +51 -0
- data/examples/merb-rest/model.rb +28 -0
- data/examples/merb-rest/view_edit.html.erb +24 -0
- data/examples/merb-rest/view_index.html.erb +23 -0
- data/examples/merb-rest/view_new.html.erb +13 -0
- data/examples/merb-rest/view_show.html.erb +17 -0
- data/examples/rails-rest/controller.rb +43 -0
- data/examples/rails-rest/migration.rb +7 -0
- data/examples/rails-rest/model.rb +23 -0
- data/examples/rails-rest/view__form.html.erb +34 -0
- data/examples/rails-rest/view_edit.html.erb +6 -0
- data/examples/rails-rest/view_index.html.erb +25 -0
- data/examples/rails-rest/view_new.html.erb +5 -0
- data/examples/rails-rest/view_show.html.erb +19 -0
- data/examples/traffic_light.rb +9 -0
- data/examples/vehicle.rb +33 -0
- data/lib/state_machine/assertions.rb +36 -0
- data/lib/state_machine/branch.rb +225 -0
- data/lib/state_machine/callback.rb +236 -0
- data/lib/state_machine/core.rb +7 -0
- data/lib/state_machine/core_ext/class/state_machine.rb +5 -0
- data/lib/state_machine/core_ext.rb +2 -0
- data/lib/state_machine/error.rb +13 -0
- data/lib/state_machine/eval_helpers.rb +87 -0
- data/lib/state_machine/event.rb +257 -0
- data/lib/state_machine/event_collection.rb +141 -0
- data/lib/state_machine/extensions.rb +149 -0
- data/lib/state_machine/graph.rb +92 -0
- data/lib/state_machine/helper_module.rb +17 -0
- data/lib/state_machine/initializers/rails.rb +25 -0
- data/lib/state_machine/initializers.rb +4 -0
- data/lib/state_machine/integrations/active_model/locale.rb +11 -0
- data/lib/state_machine/integrations/active_model/observer.rb +33 -0
- data/lib/state_machine/integrations/active_model/observer_update.rb +42 -0
- data/lib/state_machine/integrations/active_model/versions.rb +31 -0
- data/lib/state_machine/integrations/active_model.rb +585 -0
- data/lib/state_machine/integrations/active_record/locale.rb +20 -0
- data/lib/state_machine/integrations/active_record/versions.rb +123 -0
- data/lib/state_machine/integrations/active_record.rb +525 -0
- data/lib/state_machine/integrations/base.rb +100 -0
- data/lib/state_machine/integrations.rb +121 -0
- data/lib/state_machine/machine.rb +2287 -0
- data/lib/state_machine/machine_collection.rb +74 -0
- data/lib/state_machine/macro_methods.rb +522 -0
- data/lib/state_machine/matcher.rb +123 -0
- data/lib/state_machine/matcher_helpers.rb +54 -0
- data/lib/state_machine/node_collection.rb +222 -0
- data/lib/state_machine/path.rb +120 -0
- data/lib/state_machine/path_collection.rb +90 -0
- data/lib/state_machine/state.rb +297 -0
- data/lib/state_machine/state_collection.rb +112 -0
- data/lib/state_machine/state_context.rb +138 -0
- data/lib/state_machine/transition.rb +470 -0
- data/lib/state_machine/transition_collection.rb +245 -0
- data/lib/state_machine/version.rb +3 -0
- data/lib/state_machine/yard/handlers/base.rb +32 -0
- data/lib/state_machine/yard/handlers/event.rb +25 -0
- data/lib/state_machine/yard/handlers/machine.rb +344 -0
- data/lib/state_machine/yard/handlers/state.rb +25 -0
- data/lib/state_machine/yard/handlers/transition.rb +47 -0
- data/lib/state_machine/yard/handlers.rb +12 -0
- data/lib/state_machine/yard/templates/default/class/html/setup.rb +30 -0
- data/lib/state_machine/yard/templates/default/class/html/state_machines.erb +12 -0
- data/lib/state_machine/yard/templates.rb +3 -0
- data/lib/state_machine/yard.rb +8 -0
- data/lib/state_machine.rb +8 -0
- data/lib/yard-state_machine.rb +2 -0
- data/state_machine.gemspec +22 -0
- data/test/files/en.yml +17 -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_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 +264 -0
@@ -0,0 +1,100 @@
|
|
1
|
+
module StateMachine
|
2
|
+
module Integrations
|
3
|
+
# Provides a set of base helpers for managing individual integrations
|
4
|
+
module Base
|
5
|
+
module ClassMethods
|
6
|
+
# The default options to use for state machines using this integration
|
7
|
+
attr_reader :defaults
|
8
|
+
|
9
|
+
# The name of the integration
|
10
|
+
def integration_name
|
11
|
+
@integration_name ||= begin
|
12
|
+
name = self.name.split('::').last
|
13
|
+
name.gsub!(/([A-Z]+)([A-Z][a-z])/,'\1_\2')
|
14
|
+
name.gsub!(/([a-z\d])([A-Z])/,'\1_\2')
|
15
|
+
name.downcase!
|
16
|
+
name.to_sym
|
17
|
+
end
|
18
|
+
end
|
19
|
+
|
20
|
+
# Whether this integration is available for the current library. This
|
21
|
+
# is only true if the ORM that the integration is for is currently
|
22
|
+
# defined.
|
23
|
+
def available?
|
24
|
+
matching_ancestors.any? && Object.const_defined?(matching_ancestors[0].split('::')[0])
|
25
|
+
end
|
26
|
+
|
27
|
+
# The list of ancestor names that cause this integration to matched.
|
28
|
+
def matching_ancestors
|
29
|
+
[]
|
30
|
+
end
|
31
|
+
|
32
|
+
# Whether the integration should be used for the given class.
|
33
|
+
def matches?(klass)
|
34
|
+
matches_ancestors?(klass.ancestors.map {|ancestor| ancestor.name})
|
35
|
+
end
|
36
|
+
|
37
|
+
# Whether the integration should be used for the given list of ancestors.
|
38
|
+
def matches_ancestors?(ancestors)
|
39
|
+
(ancestors & matching_ancestors).any?
|
40
|
+
end
|
41
|
+
|
42
|
+
# Tracks the various version overrides for an integration
|
43
|
+
def versions
|
44
|
+
@versions ||= []
|
45
|
+
end
|
46
|
+
|
47
|
+
# Creates a new version override for an integration. When this
|
48
|
+
# integration is activated, each version that is marked as active will
|
49
|
+
# also extend the integration.
|
50
|
+
#
|
51
|
+
# == Example
|
52
|
+
#
|
53
|
+
# module StateMachine
|
54
|
+
# module Integrations
|
55
|
+
# module ORMLibrary
|
56
|
+
# version '0.2.x - 0.3.x' do
|
57
|
+
# def self.active?
|
58
|
+
# ::ORMLibrary::VERSION >= '0.2.0' && ::ORMLibrary::VERSION < '0.4.0'
|
59
|
+
# end
|
60
|
+
#
|
61
|
+
# def invalidate(object, attribute, message, values = [])
|
62
|
+
# # Override here...
|
63
|
+
# end
|
64
|
+
# end
|
65
|
+
# end
|
66
|
+
# end
|
67
|
+
# end
|
68
|
+
#
|
69
|
+
# In the above example, a version override is defined for the ORMLibrary
|
70
|
+
# integration when the version is between 0.2.x and 0.3.x.
|
71
|
+
def version(name, &block)
|
72
|
+
versions << mod = Module.new(&block)
|
73
|
+
mod
|
74
|
+
end
|
75
|
+
|
76
|
+
# The path to the locale file containing translations for this
|
77
|
+
# integration. This file will only exist for integrations that actually
|
78
|
+
# support i18n.
|
79
|
+
def locale_path
|
80
|
+
path = "#{File.dirname(__FILE__)}/#{integration_name}/locale.rb"
|
81
|
+
path if File.exists?(path)
|
82
|
+
end
|
83
|
+
|
84
|
+
# Extends the given object with any version overrides that are currently
|
85
|
+
# active
|
86
|
+
def extended(base)
|
87
|
+
versions.each do |version|
|
88
|
+
base.extend(version) if version.active?
|
89
|
+
end
|
90
|
+
end
|
91
|
+
end
|
92
|
+
|
93
|
+
extend ClassMethods
|
94
|
+
|
95
|
+
def self.included(base) #:nodoc:
|
96
|
+
base.class_eval { extend ClassMethods }
|
97
|
+
end
|
98
|
+
end
|
99
|
+
end
|
100
|
+
end
|
@@ -0,0 +1,121 @@
|
|
1
|
+
# Load each available integration
|
2
|
+
require 'state_machine/integrations/base'
|
3
|
+
Dir["#{File.dirname(__FILE__)}/integrations/*.rb"].sort.each do |path|
|
4
|
+
require "state_machine/integrations/#{File.basename(path)}"
|
5
|
+
end
|
6
|
+
|
7
|
+
require 'state_machine/error'
|
8
|
+
|
9
|
+
module StateMachine
|
10
|
+
# An invalid integration was specified
|
11
|
+
class IntegrationNotFound < Error
|
12
|
+
def initialize(name)
|
13
|
+
super(nil, "#{name.inspect} is an invalid integration")
|
14
|
+
end
|
15
|
+
end
|
16
|
+
|
17
|
+
# Integrations allow state machines to take advantage of features within the
|
18
|
+
# context of a particular library. This is currently most useful with
|
19
|
+
# database libraries. For example, the various database integrations allow
|
20
|
+
# state machines to hook into features like:
|
21
|
+
# * Saving
|
22
|
+
# * Transactions
|
23
|
+
# * Observers
|
24
|
+
# * Scopes
|
25
|
+
# * Callbacks
|
26
|
+
# * Validation errors
|
27
|
+
#
|
28
|
+
# This type of integration allows the user to work with state machines in a
|
29
|
+
# fashion similar to other object models in their application.
|
30
|
+
#
|
31
|
+
# The integration interface is loosely defined by various unimplemented
|
32
|
+
# methods in the StateMachine::Machine class. See that class or the various
|
33
|
+
# built-in integrations for more information about how to define additional
|
34
|
+
# integrations.
|
35
|
+
module Integrations
|
36
|
+
# Attempts to find an integration that matches the given class. This will
|
37
|
+
# look through all of the built-in integrations under the StateMachine::Integrations
|
38
|
+
# namespace and find one that successfully matches the class.
|
39
|
+
#
|
40
|
+
# == Examples
|
41
|
+
#
|
42
|
+
# class Vehicle
|
43
|
+
# end
|
44
|
+
#
|
45
|
+
# class ActiveModelVehicle
|
46
|
+
# include ActiveModel::Observing
|
47
|
+
# include ActiveModel::Validations
|
48
|
+
# end
|
49
|
+
#
|
50
|
+
# class ActiveRecordVehicle < ActiveRecord::Base
|
51
|
+
# end
|
52
|
+
#
|
53
|
+
# class DataMapperVehicle
|
54
|
+
# include DataMapper::Resource
|
55
|
+
# end
|
56
|
+
#
|
57
|
+
# class MongoidVehicle
|
58
|
+
# include Mongoid::Document
|
59
|
+
# end
|
60
|
+
#
|
61
|
+
# class MongoMapperVehicle
|
62
|
+
# include MongoMapper::Document
|
63
|
+
# end
|
64
|
+
#
|
65
|
+
# class SequelVehicle < Sequel::Model
|
66
|
+
# end
|
67
|
+
#
|
68
|
+
# StateMachine::Integrations.match(Vehicle) # => nil
|
69
|
+
# StateMachine::Integrations.match(ActiveModelVehicle) # => StateMachine::Integrations::ActiveModel
|
70
|
+
# StateMachine::Integrations.match(ActiveRecordVehicle) # => StateMachine::Integrations::ActiveRecord
|
71
|
+
# StateMachine::Integrations.match(DataMapperVehicle) # => StateMachine::Integrations::DataMapper
|
72
|
+
# StateMachine::Integrations.match(MongoidVehicle) # => StateMachine::Integrations::Mongoid
|
73
|
+
# StateMachine::Integrations.match(MongoMapperVehicle) # => StateMachine::Integrations::MongoMapper
|
74
|
+
# StateMachine::Integrations.match(SequelVehicle) # => StateMachine::Integrations::Sequel
|
75
|
+
def self.match(klass)
|
76
|
+
all.detect {|integration| integration.matches?(klass)}
|
77
|
+
end
|
78
|
+
|
79
|
+
# Attempts to find an integration that matches the given list of ancestors.
|
80
|
+
# This will look through all of the built-in integrations under the StateMachine::Integrations
|
81
|
+
# namespace and find one that successfully matches one of the ancestors.
|
82
|
+
#
|
83
|
+
# == Examples
|
84
|
+
#
|
85
|
+
# StateMachine::Integrations.match([]) # => nil
|
86
|
+
# StateMachine::Integrations.match(['ActiveRecord::Base') # => StateMachine::Integrations::ActiveModel
|
87
|
+
def self.match_ancestors(ancestors)
|
88
|
+
all.detect {|integration| integration.matches_ancestors?(ancestors)}
|
89
|
+
end
|
90
|
+
|
91
|
+
# Finds an integration with the given name. If the integration cannot be
|
92
|
+
# found, then a NameError exception will be raised.
|
93
|
+
#
|
94
|
+
# == Examples
|
95
|
+
#
|
96
|
+
# StateMachine::Integrations.find_by_name(:active_record) # => StateMachine::Integrations::ActiveRecord
|
97
|
+
# StateMachine::Integrations.find_by_name(:active_model) # => StateMachine::Integrations::ActiveModel
|
98
|
+
# StateMachine::Integrations.find_by_name(:data_mapper) # => StateMachine::Integrations::DataMapper
|
99
|
+
# StateMachine::Integrations.find_by_name(:mongoid) # => StateMachine::Integrations::Mongoid
|
100
|
+
# StateMachine::Integrations.find_by_name(:mongo_mapper) # => StateMachine::Integrations::MongoMapper
|
101
|
+
# StateMachine::Integrations.find_by_name(:sequel) # => StateMachine::Integrations::Sequel
|
102
|
+
# StateMachine::Integrations.find_by_name(:invalid) # => StateMachine::IntegrationNotFound: :invalid is an invalid integration
|
103
|
+
def self.find_by_name(name)
|
104
|
+
all.detect {|integration| integration.integration_name == name} || raise(IntegrationNotFound.new(name))
|
105
|
+
end
|
106
|
+
|
107
|
+
# Gets a list of all of the available integrations for use. This will
|
108
|
+
# always list the ActiveModel integration last.
|
109
|
+
#
|
110
|
+
# == Example
|
111
|
+
#
|
112
|
+
# StateMachine::Integrations.all
|
113
|
+
# # => [StateMachine::Integrations::ActiveRecord, StateMachine::Integrations::DataMapper
|
114
|
+
# # StateMachine::Integrations::Mongoid, StateMachine::Integrations::MongoMapper,
|
115
|
+
# # StateMachine::Integrations::Sequel, StateMachine::Integrations::ActiveModel]
|
116
|
+
def self.all
|
117
|
+
constants = self.constants.map {|c| c.to_s}.select {|c| c != 'ActiveModel'}.sort << 'ActiveModel'
|
118
|
+
constants.map {|c| const_get(c)}
|
119
|
+
end
|
120
|
+
end
|
121
|
+
end
|