2014-04-07 21:52:21 -04:00
|
|
|
require 'active_support/core_ext/object/deep_dup'
|
|
|
|
|
2013-11-02 15:01:31 -04:00
|
|
|
module ActiveRecord
|
2013-12-30 11:47:11 -05:00
|
|
|
# Declare an enum attribute where the values map to integers in the database,
|
|
|
|
# but can be queried by name. Example:
|
2013-11-02 15:01:31 -04:00
|
|
|
#
|
|
|
|
# class Conversation < ActiveRecord::Base
|
2013-11-02 21:32:08 -04:00
|
|
|
# enum status: [ :active, :archived ]
|
2013-11-02 15:01:31 -04:00
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # conversation.update! status: 0
|
|
|
|
# conversation.active!
|
|
|
|
# conversation.active? # => true
|
2013-11-06 04:49:22 -05:00
|
|
|
# conversation.status # => "active"
|
2013-11-02 16:08:36 -04:00
|
|
|
#
|
2013-11-02 15:01:31 -04:00
|
|
|
# # conversation.update! status: 1
|
|
|
|
# conversation.archived!
|
|
|
|
# conversation.archived? # => true
|
2013-11-06 04:49:22 -05:00
|
|
|
# conversation.status # => "archived"
|
2013-11-02 16:08:36 -04:00
|
|
|
#
|
2013-11-02 15:01:31 -04:00
|
|
|
# # conversation.update! status: 1
|
2013-11-06 04:49:22 -05:00
|
|
|
# conversation.status = "archived"
|
2013-11-02 15:01:31 -04:00
|
|
|
#
|
2013-12-20 07:12:59 -05:00
|
|
|
# # conversation.update! status: nil
|
|
|
|
# conversation.status = nil
|
|
|
|
# conversation.status.nil? # => true
|
|
|
|
# conversation.status # => nil
|
|
|
|
#
|
2013-12-30 11:47:11 -05:00
|
|
|
# Scopes based on the allowed values of the enum field will be provided
|
2014-05-09 16:46:22 -04:00
|
|
|
# as well. With the above example:
|
|
|
|
#
|
|
|
|
# Conversation.active
|
|
|
|
# Conversation.archived
|
2013-12-30 11:47:11 -05:00
|
|
|
#
|
2013-11-02 15:01:31 -04:00
|
|
|
# You can set the default value from the database declaration, like:
|
|
|
|
#
|
2013-11-02 16:08:36 -04:00
|
|
|
# create_table :conversations do |t|
|
2013-11-02 15:01:31 -04:00
|
|
|
# t.column :status, :integer, default: 0
|
|
|
|
# end
|
2013-11-02 16:08:36 -04:00
|
|
|
#
|
2013-11-02 15:01:31 -04:00
|
|
|
# Good practice is to let the first declared status be the default.
|
2013-11-02 21:32:08 -04:00
|
|
|
#
|
2013-12-05 03:41:09 -05:00
|
|
|
# Finally, it's also possible to explicitly map the relation between attribute and
|
|
|
|
# database integer with a +Hash+:
|
2013-11-02 21:32:08 -04:00
|
|
|
#
|
|
|
|
# class Conversation < ActiveRecord::Base
|
|
|
|
# enum status: { active: 0, archived: 1 }
|
|
|
|
# end
|
2013-11-06 09:25:04 -05:00
|
|
|
#
|
2013-12-05 03:41:09 -05:00
|
|
|
# Note that when an +Array+ is used, the implicit mapping from the values to database
|
|
|
|
# integers is derived from the order the values appear in the array. In the example,
|
2013-12-06 14:21:12 -05:00
|
|
|
# <tt>:active</tt> is mapped to +0+ as it's the first element, and <tt>:archived</tt>
|
2013-12-05 03:41:09 -05:00
|
|
|
# is mapped to +1+. In general, the +i+-th element is mapped to <tt>i-1</tt> in the
|
|
|
|
# database.
|
|
|
|
#
|
|
|
|
# Therefore, once a value is added to the enum array, its position in the array must
|
|
|
|
# be maintained, and new values should only be added to the end of the array. To
|
|
|
|
# remove unused values, the explicit +Hash+ syntax should be used.
|
|
|
|
#
|
2013-11-06 09:25:04 -05:00
|
|
|
# In rare circumstances you might need to access the mapping directly.
|
2014-01-14 18:41:44 -05:00
|
|
|
# The mappings are exposed through a class method with the pluralized attribute
|
|
|
|
# name:
|
2013-11-06 09:25:04 -05:00
|
|
|
#
|
2014-01-14 06:58:22 -05:00
|
|
|
# Conversation.statuses # => { "active" => 0, "archived" => 1 }
|
2013-11-06 09:25:04 -05:00
|
|
|
#
|
2014-01-14 18:41:44 -05:00
|
|
|
# Use that class method when you need to know the ordinal value of an enum:
|
2013-11-06 09:25:04 -05:00
|
|
|
#
|
2014-01-14 06:58:22 -05:00
|
|
|
# Conversation.where("status <> ?", Conversation.statuses[:archived])
|
2014-02-04 12:27:57 -05:00
|
|
|
#
|
|
|
|
# Where conditions on an enum attribute must use the ordinal value of an enum.
|
2013-11-02 15:01:31 -04:00
|
|
|
module Enum
|
2014-07-16 13:44:41 -04:00
|
|
|
def self.extended(base) # :nodoc:
|
2014-04-07 10:01:03 -04:00
|
|
|
base.class_attribute(:defined_enums)
|
|
|
|
base.defined_enums = {}
|
|
|
|
end
|
2014-01-20 18:59:20 -05:00
|
|
|
|
2014-07-16 13:44:41 -04:00
|
|
|
def inherited(base) # :nodoc:
|
2014-04-07 21:52:21 -04:00
|
|
|
base.defined_enums = defined_enums.deep_dup
|
|
|
|
super
|
2014-01-20 18:59:20 -05:00
|
|
|
end
|
|
|
|
|
2015-02-11 16:56:26 -05:00
|
|
|
class EnumType < Type::Value
|
|
|
|
def initialize(name, mapping)
|
|
|
|
@name = name
|
|
|
|
@mapping = mapping
|
|
|
|
end
|
|
|
|
|
|
|
|
def type_cast_from_user(value)
|
|
|
|
return if value.blank?
|
|
|
|
|
|
|
|
if mapping.has_key?(value)
|
|
|
|
value.to_s
|
|
|
|
elsif mapping.has_value?(value)
|
|
|
|
mapping.key(value)
|
|
|
|
else
|
|
|
|
raise ArgumentError, "'#{value}' is not a valid #{name}"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def type_cast_from_database(value)
|
|
|
|
mapping.key(value)
|
|
|
|
end
|
|
|
|
|
|
|
|
def type_cast_for_database(value)
|
|
|
|
mapping.fetch(value, value)
|
|
|
|
end
|
|
|
|
|
|
|
|
protected
|
|
|
|
|
|
|
|
attr_reader :name, :mapping
|
|
|
|
end
|
|
|
|
|
2013-11-02 15:01:31 -04:00
|
|
|
def enum(definitions)
|
2013-11-04 13:36:22 -05:00
|
|
|
klass = self
|
2013-11-02 15:01:31 -04:00
|
|
|
definitions.each do |name, values|
|
2014-01-14 06:58:22 -05:00
|
|
|
# statuses = { }
|
|
|
|
enum_values = ActiveSupport::HashWithIndifferentAccess.new
|
2013-11-04 18:55:29 -05:00
|
|
|
name = name.to_sym
|
2013-11-02 15:01:31 -04:00
|
|
|
|
2014-01-14 06:58:22 -05:00
|
|
|
# def self.statuses statuses end
|
2014-01-27 04:39:52 -05:00
|
|
|
detect_enum_conflict!(name, name.to_s.pluralize, true)
|
2014-01-14 06:58:22 -05:00
|
|
|
klass.singleton_class.send(:define_method, name.to_s.pluralize) { enum_values }
|
|
|
|
|
2015-02-11 16:56:26 -05:00
|
|
|
detect_enum_conflict!(name, name)
|
|
|
|
detect_enum_conflict!(name, "#{name}=")
|
|
|
|
|
|
|
|
attribute name, EnumType.new(name, enum_values)
|
2014-01-11 06:57:09 -05:00
|
|
|
|
2015-02-11 16:56:26 -05:00
|
|
|
_enum_methods_module.module_eval do
|
2013-11-04 13:36:22 -05:00
|
|
|
pairs = values.respond_to?(:each_pair) ? values.each_pair : values.each_with_index
|
|
|
|
pairs.each do |value, i|
|
2013-11-06 09:25:04 -05:00
|
|
|
enum_values[value] = i
|
2013-11-02 15:01:31 -04:00
|
|
|
|
2013-12-22 06:53:39 -05:00
|
|
|
# def active?() status == 0 end
|
2014-01-27 04:39:52 -05:00
|
|
|
klass.send(:detect_enum_conflict!, name, "#{value}?")
|
2015-02-11 16:56:26 -05:00
|
|
|
define_method("#{value}?") { self[name] == value.to_s }
|
2013-11-02 15:01:31 -04:00
|
|
|
|
2013-12-23 10:18:06 -05:00
|
|
|
# def active!() update! status: :active end
|
2014-01-27 04:39:52 -05:00
|
|
|
klass.send(:detect_enum_conflict!, name, "#{value}!")
|
2013-11-06 09:25:04 -05:00
|
|
|
define_method("#{value}!") { update! name => value }
|
2014-01-27 04:39:52 -05:00
|
|
|
|
|
|
|
# scope :active, -> { where status: 0 }
|
|
|
|
klass.send(:detect_enum_conflict!, name, value, true)
|
2015-02-11 16:56:26 -05:00
|
|
|
klass.scope value, -> { klass.where name => value }
|
2013-11-04 13:36:22 -05:00
|
|
|
end
|
2013-11-02 15:01:31 -04:00
|
|
|
end
|
2014-04-07 10:01:03 -04:00
|
|
|
defined_enums[name.to_s] = enum_values
|
2013-11-02 15:01:31 -04:00
|
|
|
end
|
|
|
|
end
|
2013-11-04 13:36:22 -05:00
|
|
|
|
2013-12-05 21:12:42 -05:00
|
|
|
private
|
|
|
|
def _enum_methods_module
|
|
|
|
@_enum_methods_module ||= begin
|
2015-02-11 16:56:26 -05:00
|
|
|
mod = Module.new
|
2013-12-05 21:12:42 -05:00
|
|
|
include mod
|
|
|
|
mod
|
|
|
|
end
|
2013-11-04 13:36:22 -05:00
|
|
|
end
|
2014-01-27 04:39:52 -05:00
|
|
|
|
|
|
|
ENUM_CONFLICT_MESSAGE = \
|
|
|
|
"You tried to define an enum named \"%{enum}\" on the model \"%{klass}\", but " \
|
|
|
|
"this will generate a %{type} method \"%{method}\", which is already defined " \
|
|
|
|
"by %{source}."
|
|
|
|
|
|
|
|
def detect_enum_conflict!(enum_name, method_name, klass_method = false)
|
|
|
|
if klass_method && dangerous_class_method?(method_name)
|
|
|
|
raise ArgumentError, ENUM_CONFLICT_MESSAGE % {
|
|
|
|
enum: enum_name,
|
|
|
|
klass: self.name,
|
|
|
|
type: 'class',
|
|
|
|
method: method_name,
|
|
|
|
source: 'Active Record'
|
|
|
|
}
|
|
|
|
elsif !klass_method && dangerous_attribute_method?(method_name)
|
|
|
|
raise ArgumentError, ENUM_CONFLICT_MESSAGE % {
|
|
|
|
enum: enum_name,
|
|
|
|
klass: self.name,
|
|
|
|
type: 'instance',
|
|
|
|
method: method_name,
|
|
|
|
source: 'Active Record'
|
|
|
|
}
|
|
|
|
elsif !klass_method && method_defined_within?(method_name, _enum_methods_module, Module)
|
|
|
|
raise ArgumentError, ENUM_CONFLICT_MESSAGE % {
|
|
|
|
enum: enum_name,
|
|
|
|
klass: self.name,
|
|
|
|
type: 'instance',
|
|
|
|
method: method_name,
|
|
|
|
source: 'another enum'
|
|
|
|
}
|
|
|
|
end
|
|
|
|
end
|
2013-11-02 15:01:31 -04:00
|
|
|
end
|
|
|
|
end
|