Custom Rails configuration
The Rails apps I work with have varying levels of configuration. Some require credentials files while others shove a few variables in application.rb
. There are applications using .env
and some with constants littered across the codebase.
But there’s one common thread. They all require some environment-based configuration that isn’t secret. Configuration that doesn’t change often but is used throughout the codebase. Stuff we don’t mind checking in to source control.
I started using ActiveSupport::OrderedOptions
to configure a few variables in my Rails app. I like how the API gives me clean, explicit, and fail-fast feedback. Also, it’s built into Rails!
Let’s walk through different ways of adding custom configuration to a Rails app. We will use email addresses in our examples.
Basic configuration #
For single variables that are constant across environments you can add a variable directly to your Rails application file. Here I’m adding a support email address to be used in my mailers and the public support page.
# config/application.rb`
module MyApp
class Application < Rails::Application
config.support_email = "joe@masilotti.com"
end
end
Attaching configuration directly to the application is the most straightforward approach. It can be accessed directly, as well.
Rails.configuration.support_email # => "joe@masilotti.com"
You can also override these values in the environment specific configuration files like config/environments/development.rb
and production.rb
.
Nested configuration #
You can nest configuration to associate a few values under a single namespace. In this example I have two email addresses, one for support and one for marketing.
# config/application.rb`
module MyApp
class Application < Rails::Application
config.x.emails.support = "joe@masilotti.com"
config.x.emails.marketing = "hi@masilotti.com"
end
end
Accessing them follows the same approach as above, the addition of the x
.
Rails.configuration.x.emails.support # => "joe@masilotti.com"
Rails.configuration.x.emails.marketing # => "hi@masilotti.com"
More complex configuration #
This works great for a few variables. But what happens when you have 15 email addresses to configure? And they are all different based on the environment? All those lines of code would only muddy up application.rb
.
Enter Rails::Application.config_for
, a helper method to load entire configuration files. Here’s how it works. First, create a YAML file holding all of your configuration.
# config/emails.yml
shared:
support: joe@masilotti.com
marketing: hi@masilotti.com
# Many more email addresses...
Next, load the file with config_for
in your application file.
# config/application.rb
module MyApp
class Application < Rails::Application
config.emails = config_for(:emails)
end
end
You can now access these values like before. And without the additional x
.
Rails.configuration.emails.support # => "joe@masilotti.com"
Rails.configuration.emails.marketing # => "hi@masilotti.com"
Rails.configuration.emails.sales # => nil
Oh, wait a minute. Why did #sales
return nil
? Shouldn’t a missing value raise an exception or something?
Raise on nil
values #
Glad you asked! Add a bang at the end of the call and Rails will raise an error if the value doesn’t exist.
Rails.configuration.emails.sales! # => KeyError: :sales is blank
Environment-specific configuration #
The YAML file can also be extended to provide environment-specific configuration. Note the shared:
key at the top? That will apply across all environments. But providing environments as keys will override the shared values.
# config/emails.yml
shared:
support: joe@masilotti.com
production:
support: support@masilotti.com
Now, in development we will get joe@masilotti.com
but in production support@masilotti.com
. No changes are needed to application.rb
.
More information on Rails configuration can be found in the Configuring Rails Applications guide.
Real world example #
I’m using this in production on RailsDevs, my open source Rails app. I have a few email addresses configured and pull them in on support pages and mailers.
It works really nice for Rails configuration that changes infrequently, might differ across environments, and doesn’t need to be kept secure. Keeping these directly in the codebase makes onboarding new developers to the OSS project easier.