No commit activity in last 3 years
No release in over 3 years
A StatefulController for Rails that uses a state machine to manage transitions between views.
2005
2006
2007
2008
2009
2010
2011
2012
2013
2014
2015
2016
2017
2018
2019
2020
2021
2022
2023
2024
2025
 Dependencies

Development

~> 1.10
~> 10.0
>= 0

Runtime

>= 0
>= 0
 Project Readme

StatefulController

StatefulController combines the idea of a Rails controller with a state machine that controls navigation through a sequence of views and actions (i.e. a user 'flow').

StatefulController uses the aasm gem to implement the state machine part of things, so we use its terminology to define the state machine. A state machine can be thought of as a graph of states (vertices) connected by event transitions (edges). A StatefulController treats states as Rails' views and events as Rails' actions. A StatefulController is a Rails controller, so you can call actions directly. But, more interestingly, you can call two special actions: start and next.

Example

Start with a simple controller in your Rails app:

class ExampleController < ApplicationController
  include StatefulController

  class ExampleState < StatefulController::State
  end

  # states are 'views' and events are 'actions'
  aasm do 
    view :sleeping, initial: true
    view :running
    view :cleaning
    
    action :run do
      transitions from: :sleeping, to: :running
    end

    action :clean do 
      transitions from: :running, to: :cleaning
    end

    action :sleep do 
      transitions from: [:running, :cleaning], to: :sleeping
    end
  end

  # controller actions can be defined here just like normal Rails actions:
  
  def run
    # NOTE: unlike Rails, the default render will not be 'run.html.erb', but 
    # instead will be defined by the aasm block above!

    # StatefulController acts like a dynamic dispatcher to your 
    # views, but instead of hand coding manual 'redirect_to' or 
    # 'render view' in various actions, the dispatcher is governed by the state machine.
  end

  def clean
  end

  def sleep
  end

  private

  def load_state
    session[:state] || ExampleState.new
  end

  def save_state(s)
    session[:state] = s
  end

end

Add some views:

app/
  views/
    example/
      cleaning.html.erb
      running.html.erb
      sleeping.html.erb

Add some routes:

  
  # don't forget to add the two special routes!
  get "example/start"
  get "example/next"

  # and your defined action routes:
  get "example/run"
  get "example/clean"
  get "example/sleep"

What does that get us?

Try the following:

http://localhost:3000/example/start   # starts in sleeping.html.erb
http://localhost:3000/example/next    # finds and fires 'run' and shows running.html.erb
http://localhost:3000/example/next    # finds and fires 'clean' and shows cleaning.html.erb
http://localhost:3000/example/next    # finds and fires 'sleep' and shows sleeping.html.erb

So, the special next action dispatches whatever the next permitted action is (according to the state machine definition above).

What if we don't follow the rules?

http://localhost:3000/example/start   # starts in sleeping.html.erb
http://localhost:3000/example/clean   # guard prevents 'clean' and shows current state (sleeping) 

Summary

StatefulController extends AASM by adding aliases for 'state' (view) and 'event' (action) in order to make the StatefulController context more readable.

You can define the legal transitions in your user flow in the state machine and StatefulController will enforce that flow for you, which greatly simplifies the Rails controller logic for complex flows. See the source for the dummy Rails application which has more detail and interactive examples.

Installation

Add this line to your application's Gemfile:

gem 'stateful_controller'

And then execute:

$ bundle

Or install it yourself as:

$ gem install stateful_controller

Development

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.

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 tags, and push the .gem file to rubygems.org.

Contributing

Bug reports and pull requests are welcome on GitHub at https://github.com/coldnebo/stateful_controller. This project is intended to be a safe, welcoming space for collaboration, and contributors are expected to adhere to the Contributor Covenant code of conduct.

License

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