ND::Enum
This gem allows you to create and use enums easily and quickly in your Rails project.
Installation
Install the gem and add to the application's Gemfile by executing:
$ bundle add nd-enum
If bundler is not being used to manage dependencies, install the gem by executing:
$ gem install nd-enum
Usage
Basic usage
Define your enum in an ActiveRecord model:
class User < ApplicationRecord
nd_enum(role: %i(user admin))
end
It creates a module for your enum that contains one constant per enum value. Say goodbye to magic strings!
In our example, the module is User::Role
, and the constants are User::Role::USER
and User::Role::ADMIN
.
irb(main)> User::Role::USER
=> "user"
irb(main)> User::Role::ADMIN
=> "admin"
irb(main)> User::Role.all
=> ["user", "admin"]
irb(main)> User::Role.length
=> 2
irb(main)> User::Role[1]
=> "admin"
irb(main)> User::Role[:user]
=> "user"
irb(main)> User::Role.include?('foobar')
=> false
irb(main)> User::Role.include?('user')
=> true
ND::Enum inheritates from Enumerable
, so it is possible to use all Enumerable
methods on the enum module: each
, map
, find
...
Configuration
Fine-tune ND::Enum
behaviour by creating an initializer (for example: config/initializers/nd_enum.rb
).
The values shown in the example below are the default values.
ND::Enum.configure do |c|
c.default_i18n_scope = :base
c.default_i18n_validation_mode = :ignore # Allowed values: ignore, log, enforce
end
I18n
Allows to translate your enum values. Add to your locale files:
en:
users: # Model.table_name
role: # attribute
base: # default scope
user: User
admin: Admin
foobar: # custom scope
user: The user
admin: The admin
Then call t
(or translate
) method:
irb(main)> User::Role.t(:user) # Or `translate` method (alias)
=> "translation missing: en.users.role.base.user"
Use a different scope to have several translations for a single value, depending on context:
irb(main)> User::Role.t(:user, :foobar)
=> "translation missing: en.users.role.foobar.user"
Please note that the default scope (base
) can be configured using the default_i18n_scope
initializer option.
Validate translations presence
Validate that your enum are translated. By default, this feature is disabled.
class User < ApplicationRecord
nd_enum(role: %i(user admin), i18n: { mode: :log })
end
It will log the missing translations for each scope & locale. For example:
I, [2022-08-10T21:17:53.931669 #67401] INFO -- : ND::Enum: User#role scopes=[:base, :short]
I, [2022-08-10T21:17:53.931669 #67401] INFO -- : ND::Enum: User#role locale=en missing_keys=[]
I, [2022-08-10T21:17:53.931669 #67401] INFO -- : ND::Enum: User#role locale=nl missing_keys=["users.role.base.user", "users.role.short.user"]
This mode can be used as default with the c.default_i18n_validation_mode = :log
initializer option.
Enforce translations presence
Raise an exception when some translations are missing.
class User < ApplicationRecord
nd_enum(role: %i(user admin), i18n: { mode: :enforce })
end
(irb):1:in `<main>': One or several translations are missing (ND::Enum::MissingTranslationError)
from bin/console:15:in `<main>'
This mode can be used as default with the c.default_i18n_validation_mode = :enforce
initializer option.
ActiveRecord
Enum
Add a wrapper to ActiveRecord
Enum by specifying the db: true
option.
class User < ApplicationRecord
nd_enum(role: %i(user admin), db: true)
end
# It does exactly the same thing than below, but shorter:
class User < ApplicationRecord
nd_enum(role: %i(user admin))
enum(role: User::Role.to_h) # Or `enum(role: { user: 'user', admin: 'admin '})`
end
It allows to use these methods:
user.admin!
user.admin? # => true
user.role # => "admin"
And these scopes:
User.admin
User.not_admin
User.user
User.not_user
# ...
Disable scope definition by setting scopes: false
to your enum:
class User < ApplicationRecord
nd_enum(role: %i(user admin), db: { scopes: false })
end
Set the default enum:
class User < ApplicationRecord
nd_enum(role: %i(user admin), db: { default: :admin })
end
Add a prefix
or suffix
option when you need to define multiple enums with same values. If the passed value is true, the methods are prefixed/suffixed with the name of the enum. It is also possible to supply a custom value:
class User < ApplicationRecord
nd_enum(role: %i(user admin), db: { prefix: true })
# Scopes: `User.role_admin`, `User.role_user` ...
# Methods: `User.role_admin!`, `User.role_user!` ...
nd_enum(role: %i(user admin), db: { suffix: true })
# Scopes: `User.admin_role`, `User.user_role` ...
# Methods: `User.admin_role!`, `User.user_role!` ...
nd_enum(role: %i(user admin), db: { prefix: 'foobar' })
# Scopes: `User.foobar_admin`, `User.foobar_user` ...
# Methods: `User.foobar_admin!`, `User.foobar_user!` ...
end
Development
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.
Guard is also installed: bundle exec guard
.
To install this gem onto your local machine, run bundle exec rake install
. To release a new version, run gem bump
(or manually 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 the created tag, and push the .gem
file to rubygems.org.
Contributing
Bug reports and pull requests are welcome on GitHub at https://github.com/rclavel/nd-enum.
License
The gem is available as open source under the terms of the MIT License.