module ActiveGraph::Shared # Attributes provides a set of class methods for defining an attributes # schema and instance methods for reading and writing attributes. # # @example Usage # class Person # include ActiveGraph::Shared::Attributes # attribute :name # end # # person = Person.new # person.name = "Ben Poweski" # # Originally part of ActiveAttr, https://github.com/cgriego/active_attr module Attributes extend ActiveSupport::Concern include ActiveModel::AttributeMethods # Methods deprecated on the Object class which can be safely overridden DEPRECATED_OBJECT_METHODS = %w(id type) included do attribute_method_suffix '' if attribute_method_matchers.none? { |matcher| matcher.prefix == '' && matcher.suffix == '' } attribute_method_suffix '=' attribute_method_suffix '?' end # Performs equality checking on the result of attributes and its type. # # @example Compare for equality. # model == other # # @param [ActiveAttr::Attributes, Object] other The other model to compare # # @return [true, false] True if attributes are equal and other is instance # of the same Class, false if not. def ==(other) return false unless other.instance_of? self.class attributes == other.attributes end # Returns a Hash of all attributes # # @example Get attributes # person.attributes # => {"name"=>"Ben Poweski"} # # @return [Hash{String => Object}] The Hash of all attributes def attributes attributes_map { |name| send name } end # Write a single attribute to the model's attribute hash. # # @example Write the attribute with write_attribute # person.write_attribute(:name, "Benjamin") # @example Write an attribute with bracket syntax # person[:name] = "Benjamin" # # @param [String, Symbol, #to_s] name The name of the attribute to update. # @param [Object] value The value to set for the attribute. # # @raise [UnknownAttributeError] if the attribute is unknown def write_attribute(name, value) fail ActiveGraph::UnknownAttributeError, "unknown attribute: #{name}" if !respond_to? "#{name}=" send "#{name}=", value end alias []= write_attribute def query_attribute(name) fail ActiveGraph::UnknownAttributeError, "unknown attribute: #{name}" if !respond_to? "#{name}?" send "#{name}?" end private # Read an attribute from the attributes hash def attribute(name) @attributes ||= ActiveGraph::AttributeSet.new({}, self.class.attributes.keys) @attributes.fetch_value(name.to_s) end # Write an attribute to the attributes hash def attribute=(name, value) @attributes ||= ActiveGraph::AttributeSet.new({}, self.class.attributes.keys) @attributes.write_cast_value(name, value) value end # Maps all attributes using the given block # # @example Stringify attributes # person.attributes_map { |name| send(name).to_s } # # @yield [name] block called to return hash value # @yieldparam [String] name The name of the attribute to map. # # @return [Hash{String => Object}] The Hash of mapped attributes def attributes_map Hash[self.class.attribute_names.map { |name| [name, yield(name)] }] end def attribute?(name) ActiveGraph::Shared::TypeConverters::BooleanConverter.to_ruby(read_attribute(name)) end module ClassMethods # Defines an attribute # # For each attribute that is defined, a getter and setter will be # added as an instance method to the model. An # {AttributeDefinition} instance will be added to result of the # attributes class method. # # @example Define an attribute. # attribute :name # # @param (see AttributeDefinition#initialize) # # @raise [DangerousAttributeError] if the attribute name conflicts with # existing methods # # @return [AttributeDefinition] Attribute's definition def attribute(name) fail ActiveGraph::DangerousAttributeError, %(an attribute method named "#{name}" would conflict with an existing method) if dangerous_attribute?(name) attribute!(name) end # Returns an Array of attribute names as Strings # # @example Get attribute names # Person.attribute_names # # @return [Array] The attribute names def attribute_names attributes.keys end # Returns a Hash of AttributeDefinition instances # # @example Get attribute definitions # Person.attributes # # @return [ActiveSupport::HashWithIndifferentAccess{String => ActiveGraph::Shared::AttributeDefinition}] # The Hash of AttributeDefinition instances def attributes @attributes ||= ActiveSupport::HashWithIndifferentAccess.new end # Determine if a given attribute name is dangerous # # Some attribute names can cause conflicts with existing methods # on an object. For example, an attribute named "timeout" would # conflict with the timeout method that Ruby's Timeout library # mixes into Object. # # @example Testing a harmless attribute # Person.dangerous_attribute? :name #=> false # # @example Testing a dangerous attribute # Person.dangerous_attribute? :nil #=> "nil?" # # @param name Attribute name # # @return [false, String] False or the conflicting method name def dangerous_attribute?(name) methods = instance_methods attribute_methods(name).detect do |method_name| !DEPRECATED_OBJECT_METHODS.include?(method_name.to_s) && methods.include?(method_name) end unless attribute_names.include? name.to_s end # Returns the class name plus its attribute names # # @example Inspect the model's definition. # Person.inspect # # @return [String] Human-readable presentation of the attributes def inspect inspected_attributes = attribute_names.sort attributes_list = "(#{inspected_attributes.join(', ')})" unless inspected_attributes.empty? "#{name}#{attributes_list}" end protected # Assign a set of attribute definitions, used when subclassing models # # @param [Array] attributes The Array of # AttributeDefinition instances def attributes=(attributes) @attributes = attributes end # Overrides ActiveModel::AttributeMethods to backport 3.2 fix def instance_method_already_implemented?(method_name) generated_attribute_methods.method_defined?(method_name) end private # Expand an attribute name into its generated methods names def attribute_methods(name) attribute_method_matchers.map { |matcher| matcher.method_name name } end # Ruby inherited hook to assign superclass attributes to subclasses def inherited(subclass) super subclass.attributes = attributes.dup end end end end