inheritance.rb 9.5 KB
Newer Older
A
Akira Matsuda 已提交
1 2
require 'active_support/core_ext/hash/indifferent_access'

3
module ActiveRecord
4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
  # == Single table inheritance
  #
  # Active Record allows inheritance by storing the name of the class in a column that by
  # default is named "type" (can be changed by overwriting <tt>Base.inheritance_column</tt>).
  # This means that an inheritance looking like this:
  #
  #   class Company < ActiveRecord::Base; end
  #   class Firm < Company; end
  #   class Client < Company; end
  #   class PriorityClient < Client; end
  #
  # When you do <tt>Firm.create(name: "37signals")</tt>, this record will be saved in
  # the companies table with type = "Firm". You can then fetch this row again using
  # <tt>Company.where(name: '37signals').first</tt> and it will return a Firm object.
  #
19
  # Be aware that because the type column is an attribute on the record every new
20
  # subclass will instantly be marked as dirty and the type column will be included
21
  # in the list of changed attributes on the record. This is different from non
22 23 24
  # STI classes:
  #
  #   Company.new.changed? # => false
25 26
  #   Firm.new.changed?    # => true
  #   Firm.new.changes     # => {"type"=>["","Firm"]}
27 28 29 30 31 32 33 34
  #
  # If you don't have a type column defined in your table, single-table inheritance won't
  # be triggered. In that case, it'll work just like normal subclasses with no special magic
  # for differentiating between them or reloading the right type with find.
  #
  # Note, all the attributes for all the cases are kept in the same table. Read more:
  # http://www.martinfowler.com/eaaCatalog/singleTableInheritance.html
  #
35 36 37 38
  module Inheritance
    extend ActiveSupport::Concern

    included do
N
Neeraj Singh 已提交
39
      # Determines whether to store the full constant name including namespace when using STI.
J
Jon Leighton 已提交
40 41
      class_attribute :store_full_sti_class, instance_writer: false
      self.store_full_sti_class = true
42 43 44
    end

    module ClassMethods
45 46
      # Determines if one of the attributes passed in is the inheritance column,
      # and if the inheritance column is attr accessible, it initializes an
N
Neeraj Singh 已提交
47
      # instance of the given subclass instead of the base class.
48
      def new(*args, &block)
49
        if abstract_class? || self == Base
50
          raise NotImplementedError, "#{self} is an abstract class and cannot be instantiated."
51
        end
52 53 54 55 56 57 58 59 60 61

        attrs = args.first
        if subclass_from_attributes?(attrs)
          subclass = subclass_from_attributes(attrs)
        end

        if subclass
          subclass.new(*args, &block)
        else
          super
62 63 64
        end
      end

N
Neeraj Singh 已提交
65 66
      # Returns +true+ if this does not need STI type condition. Returns
      # +false+ if STI type condition needs to be applied.
67
      def descends_from_active_record?
J
Jon Leighton 已提交
68
        if self == Base
69
          false
J
Jon Leighton 已提交
70 71
        elsif superclass.abstract_class?
          superclass.descends_from_active_record?
72
        else
J
Jon Leighton 已提交
73
          superclass == Base || !columns_hash.include?(inheritance_column)
74 75 76 77 78 79 80 81 82
        end
      end

      def finder_needs_type_condition? #:nodoc:
        # This is like this because benchmarking justifies the strange :false stuff
        :true == (@finder_needs_type_condition ||= descends_from_active_record? ? :false : :true)
      end

      def symbolized_base_class
X
Xavier Noria 已提交
83
        ActiveSupport::Deprecation.warn('`ActiveRecord::Base.symbolized_base_class` is deprecated and will be removed without replacement.')
84 85 86 87
        @symbolized_base_class ||= base_class.to_s.to_sym
      end

      def symbolized_sti_name
X
Xavier Noria 已提交
88
        ActiveSupport::Deprecation.warn('`ActiveRecord::Base.symbolized_sti_name` is deprecated and will be removed without replacement.')
89 90 91
        @symbolized_sti_name ||= sti_name.present? ? sti_name.to_sym : symbolized_base_class
      end

J
Jon Leighton 已提交
92 93
      # Returns the class descending directly from ActiveRecord::Base, or
      # an abstract class, if any, in the inheritance hierarchy.
94 95
      #
      # If A extends AR::Base, A.base_class will return A. If B descends from A
96 97 98 99 100
      # through some arbitrarily deep hierarchy, B.base_class will return A.
      #
      # If B < A and C < B and if A is an abstract_class then both B.base_class
      # and C.base_class would return B as the answer since A is an abstract_class.
      def base_class
J
Jon Leighton 已提交
101
        unless self < Base
102 103 104
          raise ActiveRecordError, "#{name} doesn't belong in a hierarchy descending from ActiveRecord"
        end

J
Jon Leighton 已提交
105
        if superclass == Base || superclass.abstract_class?
106 107
          self
        else
J
Jon Leighton 已提交
108
          superclass.base_class
109
        end
110 111 112
      end

      # Set this to true if this is an abstract class (see <tt>abstract_class?</tt>).
113 114 115 116 117 118 119 120 121 122
      # If you are using inheritance with ActiveRecord and don't want child classes
      # to utilize the implied STI table name of the parent class, this will need to be true.
      # For example, given the following:
      #
      #   class SuperClass < ActiveRecord::Base
      #     self.abstract_class = true
      #   end
      #   class Child < SuperClass
      #     self.table_name = 'the_table_i_really_want'
      #   end
123
      #
124 125 126
      #
      # <tt>self.abstract_class = true</tt> is required to make <tt>Child<.find,.create, or any Arel method></tt> use <tt>the_table_i_really_want</tt> instead of a table called <tt>super_classes</tt>
      #
127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153
      attr_accessor :abstract_class

      # Returns whether this class is an abstract class or not.
      def abstract_class?
        defined?(@abstract_class) && @abstract_class == true
      end

      def sti_name
        store_full_sti_class ? name : name.demodulize
      end

      protected

      # Returns the class type of the record using the current module as a prefix. So descendants of
      # MyApp::Business::Account would appear as MyApp::Business::AccountSubclass.
      def compute_type(type_name)
        if type_name.match(/^::/)
          # If the type is prefixed with a scope operator then we assume that
          # the type_name is an absolute reference.
          ActiveSupport::Dependencies.constantize(type_name)
        else
          # Build a list of candidates to search for
          candidates = []
          name.scan(/::|$/) { candidates.unshift "#{$`}::#{type_name}" }
          candidates << type_name

          candidates.each do |candidate|
154 155
            constant = ActiveSupport::Dependencies.safe_constantize(candidate)
            return constant if candidate == constant.to_s
156 157
          end

C
Chulki Lee 已提交
158
          raise NameError.new("uninitialized constant #{candidates.first}", candidates.first)
159 160 161 162 163
        end
      end

      private

164 165 166 167
      # Called by +instantiate+ to decide which class to use for a new
      # record instance. For single-table inheritance, we check the record
      # for a +type+ column and return the corresponding class.
      def discriminate_class_for_record(record)
168 169
        if using_single_table_inheritance?(record)
          find_sti_class(record[inheritance_column])
170 171 172 173 174
        else
          super
        end
      end

175 176
      def using_single_table_inheritance?(record)
        record[inheritance_column].present? && columns_hash.include?(inheritance_column)
177 178
      end

179
      def find_sti_class(type_name)
180 181
        if store_full_sti_class
          ActiveSupport::Dependencies.constantize(type_name)
182
        else
183
          compute_type(type_name)
184
        end
185 186 187 188 189 190
      rescue NameError
        raise SubclassNotFound,
          "The single-table inheritance mechanism failed to locate the subclass: '#{type_name}'. " +
          "This error is raised because the column '#{inheritance_column}' is reserved for storing the class in case of inheritance. " +
          "Please rename this column if you didn't intend it to be used for storing the inheritance class " +
          "or overwrite #{name}.inheritance_column to use another column for that information."
191 192 193
      end

      def type_condition(table = arel_table)
A
Aaron Patterson 已提交
194
        sti_column = table[inheritance_column]
195
        sti_names  = ([self] + descendants).map(&:sti_name)
196 197 198

        sti_column.in(sti_names)
      end
199 200 201 202 203

      # Detect the subclass from the inheritance column of attrs. If the inheritance column value
      # is not self or a valid subclass, raises ActiveRecord::SubclassNotFound
      # If this is a StrongParameters hash, and access to inheritance_column is not permitted,
      # this will ignore the inheritance column and return nil
204 205 206 207 208
      def subclass_from_attributes?(attrs)
        columns_hash.include?(inheritance_column) && attrs.is_a?(Hash)
      end

      def subclass_from_attributes(attrs)
209
        subclass_name = attrs.with_indifferent_access[inheritance_column]
210 211 212 213

        if subclass_name.present? && subclass_name != self.name
          subclass = subclass_name.safe_constantize

214
          unless descendants.include?(subclass)
215 216 217 218
            raise ActiveRecord::SubclassNotFound.new("Invalid single-table inheritance type: #{subclass_name} is not a subclass of #{name}")
          end

          subclass
219 220
        end
      end
221 222
    end

223 224 225 226 227
    def initialize_dup(other)
      super
      ensure_proper_type
    end

228 229
    private

230 231 232 233 234
    def initialize_internals_callback
      super
      ensure_proper_type
    end

235 236 237 238 239 240 241 242 243 244 245 246 247
    # Sets the attribute used for single table inheritance to this class name if this is not the
    # ActiveRecord::Base descendant.
    # Considering the hierarchy Reply < Message < ActiveRecord::Base, this makes it possible to
    # do Reply.new without having to set <tt>Reply[Reply.inheritance_column] = "Reply"</tt> yourself.
    # No such attribute would be set for objects of the Message class in that example.
    def ensure_proper_type
      klass = self.class
      if klass.finder_needs_type_condition?
        write_attribute(klass.inheritance_column, klass.sti_name)
      end
    end
  end
end