state_machine_enum 0.1.0

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 ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA256:
3
+ metadata.gz: c62f9ef7d3062c2cfec2c914d2b4bdcb8dbd38621e7317af107d0ce65edf88ef
4
+ data.tar.gz: 98f294f99cb3943dd145874afef82add8533816a863a35411511cc32a7447826
5
+ SHA512:
6
+ metadata.gz: 213b039f7fe5b20f9a68306f17ae10d044bf56107d33d4cd0ad0c03c86d3c01a3bb963df8ccc117b3dda9a1cfe324dc5647adb77da0c0f33c57f293153a079ec
7
+ data.tar.gz: 490ec1ce157ce84481dd36e0973fa2601cc1638503b2769f04d60010527b81db4c9f208b1d6dfe285b81e65e143cfbaec802e55b8b6ee047af9a672bd0bedc3d
data/.standard.yml ADDED
@@ -0,0 +1,3 @@
1
+ # For available configuration options, see:
2
+ # https://github.com/standardrb/standard
3
+ ruby_version: 3.0
data/LICENSE.txt ADDED
@@ -0,0 +1,21 @@
1
+ The MIT License (MIT)
2
+
3
+ Copyright (c) 2024 Stanislav Katkov
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in
13
+ all copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21
+ THE SOFTWARE.
data/README.md ADDED
@@ -0,0 +1,60 @@
1
+ # StateMachineEnum
2
+
3
+ This concern adds a method called "state_machine_enum" useful for defining an enum using string values along with valid state transitions. Validations will be added for the state transitions and a proper enum is going to be defined. For example:
4
+
5
+ ```ruby
6
+ state_machine_enum :state do |states|
7
+ states.permit_transition(:created, :approved_pending_settlement)
8
+ states.permit_transition(:approved_pending_settlement, :rejected)
9
+ states.permit_transition(:created, :rejected)
10
+ states.permit_transition(:approved_pending_settlement, :settled)
11
+ end
12
+ ```
13
+
14
+ ## Installation
15
+
16
+ Install the gem and add to the application's Gemfile by executing:
17
+
18
+ $ bundle add state_machine_enum
19
+
20
+ If bundler is not being used to manage dependencies, install the gem by executing:
21
+
22
+ $ gem install state_machine_enum
23
+
24
+ ## Usage
25
+
26
+ StateMachineEnum needs to be extended and then it could be used, as example in AR model.
27
+
28
+ ```
29
+ class User < ApplicationRecord
30
+ include StateMachineEnum
31
+
32
+ state_machine_enum :state do |s|
33
+ s.permit_transition(:registered, :active)
34
+ s.permit_transition(:active, :banned)
35
+ s.permit_transition(:banned, :active)
36
+ s.permit_transition(:active, :deleted)
37
+ end
38
+ end
39
+ ```
40
+
41
+ And then it will offer bunch of convenient methods and callbacks that ensure proper state transitions.
42
+
43
+ ```
44
+ user = User.new(state: 'registered')
45
+ user.active?
46
+ user.registered! # throws InvalidState error, because state can not transition to "registered".
47
+ ```
48
+ ## Development
49
+
50
+ After checking out the repo, run `bin/setup` to install dependencies. Then, run `rake test` to run the tests. You can also run `bin/console` for an interactive prompt that will allow you to experiment.
51
+
52
+ To install this gem onto your local machine, run `bundle exec rake install`. To release a new version, update the version number in `version.rb`, and then run `bundle exec rake release`, which will create a git tag for the version, push git commits and the created tag, and push the `.gem` file to [rubygems.org](https://rubygems.org).
53
+
54
+ ## Contributing
55
+
56
+ Bug reports and pull requests are welcome on GitHub at https://github.com/cheddar-me/state_machine_enum.
57
+
58
+ ## License
59
+
60
+ The gem is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT).
data/Rakefile ADDED
@@ -0,0 +1,10 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "bundler/gem_tasks"
4
+ require "minitest/test_task"
5
+
6
+ Minitest::TestTask.create
7
+
8
+ require "standard/rake"
9
+
10
+ task default: %i[test standard]
@@ -0,0 +1,5 @@
1
+ # frozen_string_literal: true
2
+
3
+ module StateMachineEnum
4
+ VERSION = "0.1.0"
5
+ end
@@ -0,0 +1,127 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative "state_machine_enum/version"
4
+ require "active_support/concern"
5
+
6
+ # This concern adds a method called "state_machine_enum" useful for defining an enum using
7
+ # string values along with valid state transitions. Validations will be added for the
8
+ # state transitions and a proper enum is going to be defined. For example:
9
+ #
10
+ # state_machine_enum :state do |states|
11
+ # states.permit_transition(:created, :approved_pending_settlement)
12
+ # states.permit_transition(:approved_pending_settlement, :rejected)
13
+ # states.permit_transition(:created, :rejected)
14
+ # states.permit_transition(:approved_pending_settlement, :settled)
15
+ # end
16
+
17
+ module StateMachineEnum
18
+ extend ActiveSupport::Concern
19
+
20
+ class StatesCollector
21
+ attr_reader :states
22
+ attr_reader :after_commit_hooks
23
+ attr_reader :common_after_commit_hooks
24
+ attr_reader :after_attribute_write_hooks
25
+ attr_reader :common_after_write_hooks
26
+
27
+ def initialize
28
+ @transitions = Set.new
29
+ @states = Set.new
30
+ @after_commit_hooks = {}
31
+ @common_after_commit_hooks = []
32
+ @after_attribute_write_hooks = {}
33
+ @common_after_write_hooks = []
34
+ end
35
+
36
+ def permit_transition(from, to)
37
+ @states << from.to_s << to.to_s
38
+ @transitions << [from.to_s, to.to_s]
39
+ end
40
+
41
+ def may_transition?(from, to)
42
+ @transitions.include?([from.to_s, to.to_s])
43
+ end
44
+
45
+ def after_inline_transition_to(target_state, &blk)
46
+ @after_attribute_write_hooks[target_state.to_s] ||= []
47
+ @after_attribute_write_hooks[target_state.to_s] << blk.to_proc
48
+ end
49
+
50
+ def after_committed_transition_to(target_state, &blk)
51
+ @after_commit_hooks[target_state.to_s] ||= []
52
+ @after_commit_hooks[target_state.to_s] << blk.to_proc
53
+ end
54
+
55
+ def after_any_committed_transition(&blk)
56
+ @common_after_commit_hooks << blk.to_proc
57
+ end
58
+
59
+ def validate(model, attribute_name)
60
+ return unless model.persisted?
61
+
62
+ was = model.attribute_was(attribute_name)
63
+ is = model[attribute_name]
64
+
65
+ unless was == is || @transitions.include?([was, is])
66
+ model.errors.add(attribute_name, "Invalid transition from #{was} to #{is}")
67
+ end
68
+ end
69
+ end
70
+
71
+ class InvalidState < StandardError
72
+ end
73
+
74
+ class_methods do
75
+ def state_machine_enum(attribute_name, **options_for_enum)
76
+ # Collect the states
77
+ collector = StatesCollector.new
78
+ yield(collector).tap do
79
+ # Define the enum using labels, with string values
80
+ enum_map = collector.states.map(&:to_sym).zip(collector.states.to_a).to_h
81
+ enum(attribute_name, enum_map, **options_for_enum)
82
+
83
+ # Define validations for transitions
84
+ validates attribute_name, presence: true
85
+ validate { |model| collector.validate(model, attribute_name) }
86
+
87
+ # Define inline hooks
88
+ before_save do |model|
89
+ _value_was, value_has_become = model.changes[attribute_name]
90
+ next unless value_has_become
91
+ hook_procs = collector.after_attribute_write_hooks[value_has_become].to_a + collector.common_after_write_hooks.to_a
92
+ hook_procs.each do |hook_proc|
93
+ hook_proc.call(model)
94
+ end
95
+ end
96
+
97
+ # Define after commit hooks
98
+ after_commit do |model|
99
+ _value_was, value_has_become = model.previous_changes[attribute_name]
100
+ next unless value_has_become
101
+ hook_procs = collector.after_commit_hooks[value_has_become].to_a + collector.common_after_commit_hooks.to_a
102
+ hook_procs.each do |hook_proc|
103
+ hook_proc.call(model)
104
+ end
105
+ end
106
+
107
+ # Define the check methods
108
+ define_method(:"ensure_#{attribute_name}_one_of!") do |*allowed_states|
109
+ val = self[attribute_name]
110
+ return if Set.new(allowed_states.map(&:to_s)).include?(val)
111
+ raise InvalidState, "#{attribute_name} must be one of #{allowed_states.inspect} but was #{val.inspect}"
112
+ end
113
+
114
+ define_method(:"ensure_#{attribute_name}_may_transition_to!") do |next_state|
115
+ val = self[attribute_name]
116
+ raise InvalidState, "#{attribute_name} already is #{val.inspect}" if next_state.to_s == val
117
+ end
118
+
119
+ define_method(:"#{attribute_name}_may_transition_to?") do |next_state|
120
+ val = self[attribute_name]
121
+ return false if val == next_state.to_s
122
+ collector.may_transition?(val, next_state)
123
+ end
124
+ end
125
+ end
126
+ end
127
+ end
@@ -0,0 +1,4 @@
1
+ module StateMachineEnum
2
+ VERSION: String
3
+ # See the writing guide of rbs: https://github.com/ruby/rbs#guides
4
+ end
metadata ADDED
@@ -0,0 +1,67 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: state_machine_enum
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.1.0
5
+ platform: ruby
6
+ authors:
7
+ - Stanislav Katkov
8
+ autorequire:
9
+ bindir: exe
10
+ cert_chain: []
11
+ date: 2024-06-07 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: activesupport
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - ">="
18
+ - !ruby/object:Gem::Version
19
+ version: '6.0'
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - ">="
25
+ - !ruby/object:Gem::Version
26
+ version: '6.0'
27
+ description: Concern that makes it easy to define and enforce possibe state transitions
28
+ for a field/object
29
+ email:
30
+ - skatkov@cheddar.me
31
+ executables: []
32
+ extensions: []
33
+ extra_rdoc_files: []
34
+ files:
35
+ - ".standard.yml"
36
+ - LICENSE.txt
37
+ - README.md
38
+ - Rakefile
39
+ - lib/state_machine_enum.rb
40
+ - lib/state_machine_enum/version.rb
41
+ - sig/state_machine_enum.rbs
42
+ homepage: https://cheddar.me
43
+ licenses:
44
+ - MIT
45
+ metadata:
46
+ homepage_uri: https://cheddar.me
47
+ source_code_uri: https://github.com/cheddar-me/state_machine_enum
48
+ post_install_message:
49
+ rdoc_options: []
50
+ require_paths:
51
+ - lib
52
+ required_ruby_version: !ruby/object:Gem::Requirement
53
+ requirements:
54
+ - - ">="
55
+ - !ruby/object:Gem::Version
56
+ version: 3.0.0
57
+ required_rubygems_version: !ruby/object:Gem::Requirement
58
+ requirements:
59
+ - - ">="
60
+ - !ruby/object:Gem::Version
61
+ version: '0'
62
+ requirements: []
63
+ rubygems_version: 3.5.3
64
+ signing_key:
65
+ specification_version: 4
66
+ summary: Define possible state transitions for a field
67
+ test_files: []