Wisper is a Ruby library for decoupling and managing the dependencies of your Ruby objects using Pub/Sub.
Wisper was extracted from a Rails codebase but is not dependant on Rails.
It is commonly used as an alternative to ActiveRecord callbacks and Observers to reduce coupling between data and domain layers.
Add this line to your application's Gemfile:
gem 'wisper'
Any class with the Wisper::Publisher
module included can broadcast events
to subscribed listeners. Listeners subscribe, at runtime, to the publisher.
class MyPublisher
include Wisper::Publisher
def do_something
# ...
publish(:done_something)
end
end
When a publisher broadcasts an event it can pass any number of arguments which are to be passed on to the listeners.
publish(:done_something, 'hello', 'world')
Any object can be a listener and only receives events it can respond to.
my_publisher = MyPublisher.new
my_publisher.subscribe(MyListener.new)
Blocks are subscribed to single events only.
my_publisher = MyPublisher.new
my_publisher.on(:done_something) do |publisher|
# ...
end
Please refer to the wisper-async gem.
class Bid < ActiveRecord::Base
include Wisper::Publisher
validates :amount, :presence => true
def commit(_attrs = nil)
assign_attributes(_attrs) if _attrs.present?
if valid?
save!
publish(:create_bid_successful, self)
else
publish(:create_bid_failed, self)
end
end
end
class BidsController < ApplicationController
def new
@bid = Bid.new
end
def create
@bid = Bid.new(params[:bid])
@bid.subscribe(PusherListener.new)
@bid.subscribe(ActivityListener.new)
@bid.subscribe(StatisticsListener.new)
@bid.on(:create_bid_successful) { |bid| redirect_to bid }
@bid.on(:create_bid_failed) { |bid| render :action => :new }
@bid.commit
end
end
A full CRUD example is shown in the Wiki.
A Service object is useful when an operation is complex, interacts with more than one model, accesses an external API or would burden a model with too much responsibility.
class PlayerJoiningTeam
include Wisper::Publisher
attr_reader :player, :team
def initialize(player, team)
@player = player
@team = team
end
def execute
membership = Membership.new(player, team)
if membership.valid?
membership.save!
email_player
assign_first_mission
publish(:player_joining_team_successful, player, team)
else
publish(:player_joining_team_failed, player, team)
end
end
private
def email_player
# ...
end
def assign_first_mission
# ...
end
end
These are typical app wide listeners which have a method for pretty much every event which is broadcast.
class PusherListener
def create_thing_successful(thing)
# ...
end
end
class ActivityListener
def create_thing_successful(thing)
# ...
end
end
class StatisticsListener
def create_thing_successful(thing)
# ...
end
end
class CacheListener
def create_thing_successful(thing)
# ...
end
end
class IndexingListener
def create_thing_successful(thing)
# ...
end
end
If you become tired of adding the same listeners to every publisher you can add listeners globally. They receive all broadcast events which they can respond to.
Global listeners should be used with caution, the execution path becomes less obvious on reading the code and of course you are introducing global state and 'always on' behaviour. This may not desirable.
Wisper.add_listener(MyListener.new)
In a Rails app you might want to add your global listeners in an initalizer.
Global listeners are threadsafe.
You might want to globally subscribe a listener to publishers with a certain class.
Wisper.add_listener(MyListener.new, :scope => :MyPublisher)
This will subscribe the listener to all instances of MyPublisher
and its
subclasses.
Alternatively you can also do exactly the same with a publisher class:
MyPublisher.add_listener(MyListener.new)
You can also globally subscribe listeners for the duration of a block.
Wisper.with_listeners(MyListener.new, OtherListener.new) do
# do stuff
end
Any events broadcast within the block by any publisher will be sent to the listeners. This is useful if you have a child object which publishes an event which is not bubbled down to a parent publisher.
Temporary Global Listeners are threadsafe.
By default a listener will get notified of all events it can respond to. You
can limit which events a listener is notified of by passing an event or array
of events to :on
.
post_creater.subscribe(PusherListener.new, :on => :create_post_successful)
If you would prefer listeners to receive events with a prefix, for example
on
, you can do so by passing a string or symbol to :prefix
.
post_creater.subscribe(PusherListener.new, :prefix => :on)
If post_creater
where to broadcast the event post_created
the subscribed
listeners would receive on_post_created
. You can also pass true
which will
use the default prefix, "on".
By default the method called on the subscriber is the same as the event
broadcast. However it can be mapped to a different method using :with
.
report_creator.subscribe(MailResponder.new, :with => :successful)
This is pretty useless unless used in conjuction with :on
, since all events
will get mapped to :successful
. Instead you might do something like this:
report_creator.subscribe(MailResponder.new, :on => :create_report_successful,
:with => :successful)
If you pass an array of events to :on
each event will be mapped to the same
method when :with
is specified. If you need to listen for select events
and map each one to a different method subscribe the listener once for
each mapping:
report_creator.subscribe(MailResponder.new, :on => :create_report_successful,
:with => :successful)
report_creator.subscribe(MailResponder.new, :on => :create_report_failed,
:with => :failed)
post.on(:success) { |post| redirect_to post }
.on(:failure) { |post| render :action => :edit, :locals => :post => post }
Wisper comes with a method for stubbing event publishers so that you can create isolation tests that only care about reacting to events.
Given this piece of code:
class CodeThatReactsToEvents
def do_something
publisher = MyPublisher.new
publisher.on(:some_event) do |variable|
return "Hello with #{variable}!"
end
publisher.execute
end
end
You can test it like this:
require 'wisper/rspec/stub_wisper_publisher'
describe CodeThatReactsToEvents do
context "on some_event" do
before do
stub_wisper_publisher("MyPublisher", :execute, :some_event, "foo")
end
it "renders" do
response = CodeThatReactsToEvents.new.do_something
response.should == "Hello with foo!"
end
end
end
This becomes important when testing, for example, Rails controllers in isolation from the business logic. This technique is used at the controller layer to isolate testing the controller from testing the encapsulated business logic.
You can use any number of args to pass to the event:
stub_wisper_publisher("MyPublisher", :execute, :some_event, "foo1", "foo2", ...)
See spec/lib/rspec_extensions_spec.rb
for a runnable example.
Tested with MRI 1.9.x, MRI 2.0.0, JRuby (1.9 and 2.0 mode) and Rubinius (1.9 mode).
See the build status for details.
rspec spec
There is both a Rakefile
and Guardfile
, if you like you prefer to run the
specs using guard-rspec
or rake
.
(The MIT License)
Copyright (c) 2013 Kris Leech
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the 'Software'), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.