• Stars
    star
    606
  • Rank 71,120 (Top 2 %)
  • Language
    Ruby
  • License
    MIT License
  • Created about 8 years ago
  • Updated over 1 year ago

Reviews

There are no reviews yet. Be the first to send feedback to the community and the maintainers!

Repository Details

A very simple state machine plugin built on top of ActiveRecord::Enum

StatefulEnum Build Status

stateful_enum is a state machine gem built on top of ActiveRecord's built-in ActiveRecord::Enum.

Installation

Add this line to your Rails app's Gemfile:

gem 'stateful_enum'

And bundle.

Motivation

You Ain't Gonna Need Abstraction

stateful_enum depends on ActiveRecord. If you prefer a "well-abstracted" state machine library that supports multiple datastores, or Plain Old Ruby Objects (who needs that feature?), I'm sorry but this gem is not for you.

I Hate Saving States in a VARCHAR Column

From a database design point of view, I prefer to save state data in an INTEGER column rather than saving the state name directly in a VARCHAR column.

❤️ ActiveRecord::Enum

ActiveRecord 4.1+ has a very simple and useful built-in Enum DSL that provides human-friendly API over integer values in DB.

Method Names Should be Verbs

AR::Enum automatically defines Ruby methods per each label. However, Enum labels are in most cases adjectives or past participle, which often creates weird method names. What we really want to define as methods are the transition events between states, and not the states themselves.

Usage

The stateful_enum gem extends AR::Enum definition to take a block with a similar DSL to the state_machine gem.

Example:

class Bug < ApplicationRecord
  enum status: {unassigned: 0, assigned: 1, resolved: 2, closed: 3} do
    event :assign do
      transition :unassigned => :assigned
    end

    event :resolve do
      before do
        self.resolved_at = Time.zone.now
      end

      transition [:unassigned, :assigned] => :resolved
    end

    event :close do
      after do
        Notifier.notify "Bug##{id} has been closed."
      end

      transition all - [:closed] => :closed
    end
  end
end

Defining the States

Just call the AR::Enum's enum method. The only difference from the original enum method is that our enum call takes a block. Please see the full API documentation of AR::Enum for more information.

Defining the Events

You can declare events through event method inside of an enum block. Then stateful_enum defines the following methods per each event:

An instance method to fire the event

@bug.assign  # does nothing and returns false if a valid transition for the current state is not defined

An instance method with ! to fire the event

@bug.assign!  # raises if a valid transition for the current state is not defined

A predicate method that returns if the event is fireable

@bug.can_assign?  # returns if the `assign` event can be called on this bug or not

An instance method that returns the state name after an event

@bug.assign_transition  #=> :assigned

Defining the Transitions

You can define state transitions through transition method inside of an event block.

There are a few important details to note regarding this feature:

  • The transition method takes a Hash each key of which is state "from" transitions to the Hash value.
  • The "from" states and the "to" states should both be given in Symbols.
  • The "from" state can be multiple states, in which case the key can be given as an Array of states, as shown in the usage example.
  • The "from" state can be all that means all defined states.
  • The "from" state can be an exception of Array of states, in this case the key can be a subtraction of all with the state to be excluded, as shown in the usage example.

:if and :unless Condition

The transition method takes an :if or :unless option as a Proc.

Example:

event :assign do
  transition :unassigned => :assigned, if: -> { !!assigned_to }
end

Event Hooks

You can define before and after event hooks inside of an event block.

Inspecting All Defined Events And Current Possible Events

You can get the list of defined events from the model class:

Bug.stateful_enum.events
#=> an Array of all defined StatefulEnum::Machine::Event objects

And you can get the list of possible event definitions from the model instance:

Bug.new(status: :assigned).stateful_enum.possible_events
#=> an Array of StatefulEnum::Machine::Event objects that are callable from the receiver object

Maybe what you really need for your app is the list of possible event "names":

Bug.new(status: :assigned).stateful_enum.possible_event_names
#=> [:resolve, :close]

You can get the list of next possible state names as well:

Bug.new(status: :assigned).stateful_enum.possible_states
#=> [:resolved, :closed]

These features would help some kind of metaprogramming over state transitions.

Generating State Machine Diagrams

stateful_enum includes a Rails generator that generates a state machine diagram. Note that you need to bundle the ruby-graphviz gem (and its dependencies) for the development env in order to run the generator.

% rails g stateful_enum:graph bug

You can specify relative or absolute output path via environment variable DEST_DIR.

% DEST_DIR=doc rails g stateful_enum:graph bug

TODO

  • Better Error handling

Support Rails Versions

  • Rails 4.1.x, 4.2.x, 5.0, 5.1, 5.2, 6.0, 6.1, 7.0, and 7.1 (edge)

Contributing

Pull requests are welcome on GitHub at https://github.com/amatsuda/stateful_enum.

License

The gem is available as open source under the terms of the MIT License.

More Repositories

1

jb

A simple and fast JSON API template engine for Ruby on Rails
Ruby
1,235
star
2

active_decorator

ORM agnostic truly Object-Oriented view helper for Rails 4, 5, 6, and 7
Ruby
1,051
star
3

traceroute

A Rake task gem that helps you find the unused routes and controller actions for your Rails 3+ app
Ruby
877
star
4

heavens_door

Capybara test scenario recorder for Rails
JavaScript
863
star
5

database_rewinder

minimalist's tiny and ultra-fast database cleaner
Ruby
800
star
6

kaminari_themes

HTML
352
star
7

erd

A Rails engine for drawing your app's ER diagram
Ruby
319
star
8

html5_validators

A gem/plugin for Rails 3, Rails 4, Rails 5, and Rails 6 that enables client-side validation using ActiveModel + HTML5 Form Validation
Ruby
304
star
9

i18n_generators

A pack of Rails generators gem plugin that generates Rails 3 and Rails 2 I18n locale files for almost every known locale.
Ruby
285
star
10

himl

HTML-based Indented Markup Language for Ruby
Ruby
235
star
11

still_life

Rails upgrade's best friend
Ruby
216
star
12

gem-src

Gem.post_install { `git clone gem_source src` }
Ruby
208
star
13

motorhead

A Rails Engine framework that helps safe and rapid feature prototyping
Ruby
181
star
14

nested_scaffold

Nested scaffold generator for Rails 4.2 and 5
Ruby
176
star
15

roundabout

A Rails Engine that generates a page transition diagram for your Rails app from request specs
Ruby
153
star
16

rfd

Ruby on Files & Directories
Ruby
152
star
17

routes_lazy_routes

A boot time booster for Ruby on Rails that defers loading the whole bloody routes so the app can spin up quickly 🤘
Ruby
141
star
18

string_template

A template engine for Rails, focusing on speed, using Ruby's String interpolation syntax
Ruby
125
star
19

kawaii_validation

An ActiveRecord extension that adds more kawaii validation syntax
Ruby
117
star
20

interactive_rspec

RSpec on IRB
Ruby
86
star
21

hocus_pocus

A magical isolated engine gem for Rails 3.1+
Ruby
82
star
22

ljax_rails

render :partial lazy-loader for Rails
Ruby
67
star
23

everywhere

Hash condition syntax for AR query everywhere!
Ruby
58
star
24

kaminari_example

A tutorial project for the basic and advanced usage of Kaminari paginator
Ruby
45
star
25

async_partial

Ruby
33
star
26

turbo_partial

Ruby
27
star
27

future_records

Ruby
25
star
28

lightweight_attributes

Ruby
24
star
29

more_optimized_resolver

Ruby
23
star
30

turbo_urls

Ruby
22
star
31

teriyaki

Automatically imports *_path definitions from config/routes.rb for acceptance testing
Ruby
22
star
32

kawaii_association

An ActiveRecord DSL extension that provides kawaii association syntax
Ruby
21
star
33

arel_ruby

ARel Ruby visitor
Ruby
20
star
34

activerecord-refinements

ActiveRecord + Ruby 2.0 refinements
Ruby
20
star
35

polymorphic_url_cache

Ruby
17
star
36

gem_i

A RubyGems plugin that explicitly aliases `gem i` to `gem install` to avoid ambiguity
Ruby
17
star
37

speed_king

Ruby
14
star
38

nested_layouts

The only fork of "nested_layouts" Rails plugin in Github that correctly bug fixed for Rails 2.3
Ruby
13
star
39

rspec-refinements

RSpec + Ruby 2.0 refinements
Ruby
10
star
40

arenai

Ruby
8
star
41

activecalendar

Rails 2.2.2 ready javascript calendar date renderer
JavaScript
8
star
42

bundler-squash

Ruby
8
star
43

rbenv-gem-shared

Ruby
8
star
44

tatsuzine

Live coded app at Rails勉強会@東京#59
Ruby
8
star
45

bot_for_ruby-lang

Ruby
7
star
46

factory_factory

a script that transfers existing AR models into factories for factory_girl
Shell
6
star
47

automagic

Ruby
6
star
48

snowman_meltdown

A simple middleware for Rails 3 to vanish _snowman parameter☃☃☃
Ruby
5
star
49

activesupport-refinements

Ruby
4
star
50

webdb073_tutorial

WEB+DB Press Vol. 73 特集2「詳解Rails 4」のチュートリアルのサンプルコード
Ruby
4
star
51

gitrockets

Ruby
3
star
52

rails3_hands_on

東京Ruby会議03のワークショップ「Rails 3ハンズオン」のサンプルアプリケーション
Ruby
3
star
53

prsnt

prsnt prttyp
3
star
54

git_commands

3
star
55

internationalization

Ruby
3
star
56

qwik

qwik
2
star
57

hfrails

hfrails
2
star
58

atode_yomu

A gem plugin that cleverly installs rdoc and ri for the latest versions of already installed gems
Ruby
2
star
59

gem-diet

Ruby
2
star
60

action_args_with_rbs

Ruby
1
star