We talk a lot about error handling in Ruby. But we rarely talk about raising errors and creating helpful error messages that are actionable. A good error should tell the developer what went wrong and what to do about it.
One library is known for its excellent error messages. Consider a typical validation error from Mongoid.
#<Mongoid::Errors::Validations:
Problem:
Validation of User failed.
Summary:
The following errors were found: Password can't be blank
Resolution:
Try persisting the document with valid data or remove the validations.>
This error describes the problem, offers a detailed summary and provides a possible resolution!
Let’s implement a similar system for the ruby-enum gem that I live-coded at NYC.rb.
First, add a dependency on i18n and require “i18n”. Then, create a lib/config/locales folder and an en.yml file in it. English error messages will go there. This file will need to be loaded by our library, specifically in ruby-enum.rb.
require 'i18n'
I18n.load_path << File.join(File.dirname(__FILE__), "config", "locales", "en.yml")
Error descriptions inside en.yml contain the problem, summary and resolution. The YAML format supports multi-lines with \_\_
and can include values from parameters using the %{name}
syntax.
en:
ruby:
enum:
errors:
messages:
uninitialized_constant:
message: "Uninitialized constant."
summary: "The constant %{name}::%{key} has not been defined."
resolution: "The enumerated value could not be found in class %{name}.\n
\_Use 'define' to declare it.\n
\_Example:\n
\_\_module %{name}\n
\_\_\_include Ruby::Enum\n
\_\_\_define %{key}, 'value'\n
\_\_end"
The base error class, Ruby::Enum::Errors::Base takes care of the translation. I stripped the implementation details below – the important parts is the BASE_KEY value for localized error messages and the compose_message method. Get the full implementation here and modify it for your project.
module Ruby
module Enum
module Errors
class Base < StandardError
attr_reader :problem, :summary, :resolution
def compose_message(key, attributes = {})
@problem = create_problem(key, attributes)
@summary = create_summary(key, attributes)
@resolution = create_resolution(key, attributes)
"\nProblem:\n #{@problem}" +
"\nSummary:\n #{@summary}" +
"\nResolution:\n #{@resolution}"
end
private
BASE_KEY = "ruby.enum.errors.messages" #:nodoc:
# implementation of create_problem, summary and resolution
end
end
end
end
Specific errors derive from this class.
module Ruby
module Enum
module Errors
class UninitializedConstantError < Base
def initialize(attrs)
super(compose_message("uninitialized_constant", attrs))
end
end
end
end
end
When raising an UninitializedConstantError
, pass the values of key and name used in the en.yml file above.
raise Ruby::Enum::Errors::UninitializedConstantError.new({
:name => "Class",
:key => "CONSTANT"
})
Here’s the result.
1.9.3-p362 :002 > require 'ruby-enum'
=> true
1.9.3-p362 :003 > raise Ruby::Enum::Errors::UninitializedConstantError.new({
:name => "Class",
:key => "CONSTANT"
})
Ruby::Enum::Errors::UninitializedConstantError:
Problem:
Uninitialized constant.
Summary:
The constant Class::CONSTANT has not been defined.
Resolution:
The enumerated value could not be found in class Class.
Use 'define' to declare it.
Example:
module Class
include Ruby::Enum
define CONSTANT, 'value'
end
Beautiful.