Code Ruby like you build Lego
There is a version with refactoring example here
What do I mean?
Every Lego is an assembly of bricks.
Every piece of software is an assembly of parts.
Ruby Bricks?
The Service Object pattern lets you define classes which represent the parts of your business logic.
By convention each Service Object is named out of a verb: it's an action.
Service Objects are bricks.
What is a service object?
It's a simple Ruby class.
It must have a precise, action name (LaunchProject, CreateContact)
It's scope is usually limited: specialised object, reusable and easy to test.
Example
You want to process an order.
Steps are:
- remove line items from stock
- complete order
- prepare for shipment
=> Service Objects represent business logic
Service Objects are:
- UpdateStock
- CompleteOrder
- PrepareForShipment
Overengineering?
You might wonder what is the point to create an object like CompleteOrder.
This object has something models lack: Context.
No more (conditional) callbacks.
All logic gathered in its own (relevant!) box.
No more fat models nor controllers.
Assemble Service Objects?
class ProcessOrder < Struct.new(:order)
class RollbackTriggerError < StandardError; end
def call
ActiveRecord::Base.transaction do
update_stock = UpdateStock.new(order).call
raise_error unless update_stock.success?
complete_order = CompleteOrder.new(order).call
raise_error unless complete_order.success?
prepare_for_shipment = PrepareForShipment.new(order).call
raise_error unless prepare_for_shipment.success?
end
rescue RollbackTriggerError
@success = false
ensure
self
end
def success?; @success != false; end
def raise_error; raise RollbackTriggerError; end
end
this feels like
Our Goal
Process
UpdateStock
CompleteOrder
PrepareForShipment
How to elegantly assemble?
Unix has pipes
Elixir has pipes and streams
What Ruby has to offer?
Assemble with Ruby
I've created Waterfall, a dedicated gem, because I needed this feature.
It keeps on chaining service objects when everything goes well.
It stops the assembly if something goes wrong.
Assemble with Waterfall
class ProcessOrder < Struct.new(:order)
include Waterfall
class RollbackTriggerError < StandardError; end
def call
ActiveRecord::Base.transaction do
self
.chain { UpdateStock.new(order) }
.chain { CompleteOrder.new(order) }
.chain { PrepareForShipment.new(order) }
.on_dam { raise RollbackTriggerError }
end
rescue RollbackTriggerError
self
end
end
# we assume all services include Waterfall
Chaining API
.chain(:foo) {
# add `foo` to local outflow, which value will
# be what the block returns
}
.chain(bar: :baz) {
# block returns a waterfall which exposes `baz`
# and we rename it `bar` in the local outflow
}
.chain { |outflow|
# outflow is a mere OpenStruct
# in our example, we could use outflow.foo and outflow.bar
}
.when_falsy { some_expression }
.dam {
# you can dam with anything you want, but let it be explanatory
}
.when_truthy { some_other_expression }
.dam {}
.on_dam {|error_pool|
# the error pool contains the value returned by a `dam` in the chain
# or even by any dam from the chained waterfalls
}
Want to know more?
Check the github repository.
Contains full api details, examples of code, examples of how to spec.
Ping me @apneadiving
Happy Building!
code ruby like you build lego
By Benjamin Roth
code ruby like you build lego
Split your app based on business logic bricks and assemble them
- 9,160