The sea_food gem is a Ruby library designed to enhance the development of service and form objects in Ruby applications. Representing SErvice Objects And Form Object Design patterns, this gem facilitates the separation of business logic and data validations from ActiveRecord models. P.S. Jian Yang would be proud; Erlich Bachman, not so much.
Add this line to your application's Gemfile:
gem 'sea_food'
And then execute:
$ bundle
Or install it yourself as:
$ gem install sea_food
class Invoices::ApproveService < SeaFood::Service
def initialize(invoice:, user:)
@invoice = invoice
@user = user
end
def call
fail!("You are not authorized to approve invoices") unless authorized?
@invoice.update!(status: :approved)
success(invoice: @invoice)
end
end
result = Invoices::ApproveService.call(invoice: invoice, user: user)
result.success?
#=> true
result.invoice
#=> <Invoice(...>
If you want to enforce the interface and not allow users to use call
without arguments defined in the initialize
you can add an initializer.
# initializers/sea_food.rb
SeaFood.configure do |config|
config.enforce_interface = true
end
It will raise an ArgumentError
.
success(data):
Marks the service result as successful and optionally provides data.
fail(data):
Marks the service result as a failure but continues executing the call method.
fail!(data):
Marks the service result as a failure and immediately exits the call method.
class TestFailService < SeaFood::Service
def initialize(email:)
@email = email
end
def call
fail(email: '[email protected]')
success(email: @email)
end
end
result = TestFailService.call(email: '[email protected]')
puts result.success? # => true
puts result.email # => '[email protected]'
In this example:
The fail method sets the result to failure but allows the method to continue. The subsequent success call overrides the failure, resulting in a successful outcome.
class TestFailBangService < SeaFood::Service
def initialize(email:)
@email = email
end
def call
fail!(email: '[email protected]')
success(email: @email) # This line is not executed
end
end
result = TestFailBangService.call(email: '[email protected]')
puts result.success? # => false
puts result.email # => '[email protected]'
The fail!
method immediately exits the call method.
The service result is a failure, and subsequent code in call is not executed.
Nested Services
You can call other services within a service. The behavior depends on whether you use call
or call!
.
call
: Executes the service and returns the result. Does not raise an exception if the service fails.
call!
: Executes the service and raises an exception if the service fails. Useful for propagating failures in nested services.
Failures in nested services do not automatically propagate to the parent service.
class InnerService < SeaFood::Service
def initialize(email:)
@email = email
end
def call
fail!(email: @email)
end
end
class OuterService < SeaFood::Service
def initialize(email:)
@email = email
end
def call
InnerService.call(email: '[email protected]')
success(email: @email)
end
end
result = OuterService.call(email: '[email protected]')
puts result.success? # => true
puts result.email # => '[email protected]'
Explanation:
InnerService
fails using fail!
.
OuterService
calls InnerService
using call
.
Since call
does not raise an exception on failure, OuterService
continues and succeeds.
Failures in nested services propagate to the parent service when using call!
.
class InnerService < SeaFood::Service
def initialize(email:)
@email = email
end
def call
fail!(email: @email)
end
end
class OuterService < SeaFood::Service
def initialize(email:)
@email = email
end
def call
InnerService.call!(email: '[email protected]')
success(email: @email) # This line is not executed
end
end
result = OuterService.call(email: '[email protected]')
puts result.success? # => false
puts result.email # => '[email protected]'
Explanation:
InnerService
fails using fail!
.
OuterService
calls InnerService
using call!.
The failure from InnerService
propagates, causing OuterService
to fail.
You can handle failures from nested services by checking their result.
class InnerService < SeaFood::Service
def initialize(value:)
@value = value
end
def call
if @value < 0
fail!(error: 'Negative value')
else
success(value: @value)
end
end
end
class OuterService < SeaFood::Service
def initialize(value:)
@value = value
end
def call
result = InnerService.call(value: @value)
if result.fail?
fail!(error: result.error)
else
success(value: result.value * 2)
end
end
end
result = OuterService.call(value: -1)
puts result.success? # => false
puts result.error # => 'Negative value'
Explanation:
OuterService
checks the result of InnerService
.
If InnerService
fails, OuterService
handles it accordingly.
The result object allows you to access data provided in success or fail calls using method syntax.
class ExampleService < SeaFood::Service
def call
success(message: 'Operation successful', value: 42)
end
end
result = ExampleService.call
puts result.success? # => true
puts result.message # => 'Operation successful'
puts result.value # => 42
After checking out the repo, run bin/setup
to install dependencies. Then, run rake spec
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.
Bug reports and pull requests are welcome on GitHub at https://github.com/[USERNAME]/sea_food. 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.
The gem is available as open source under the terms of the MIT License.
Everyone interacting in the SeaFood project’s codebases, issue trackers, chat rooms and mailing lists is expected to follow the code of conduct.