aboutsummaryrefslogblamecommitdiffstats
path: root/activerecord/lib/active_record/relation/delegation.rb
blob: dbfa92bbbdfb81f04af9270061fcb44ace542c10 (plain) (tree)
1
2
3
4
5
6
7
8
9
                

                   
                             
                                                                       
                                                                                                       


                                                                                          

                                  
                                             
                                               













                                                                     





                                                      


                                             
                                   





                                                                       
                                                      
                                         





                                                                    
                                        
                                    





                                                                    





                                        
   
require 'thread'

module ActiveRecord
  module Delegation # :nodoc:
    # Set up common delegations for performance (avoids method_missing)
    delegate :to_xml, :to_yaml, :length, :collect, :map, :each, :all?, :include?, :to_ary, :to => :to_a
    delegate :table_name, :quoted_table_name, :primary_key, :quoted_primary_key,
             :connection, :columns_hash, :auto_explain_threshold_in_seconds, :to => :klass

    @@delegation_mutex = Mutex.new

    def self.delegate_to_scoped_klass(method)
      if method.to_s =~ /\A[a-zA-Z_]\w*[!?]?\z/
        module_eval <<-RUBY, __FILE__, __LINE__ + 1
          def #{method}(*args, &block)
            scoping { @klass.#{method}(*args, &block) }
          end
        RUBY
      else
        module_eval <<-RUBY, __FILE__, __LINE__ + 1
          def #{method}(*args, &block)
            scoping { @klass.send(#{method.inspect}, *args, &block) }
          end
        RUBY
      end
    end

    def respond_to?(method, include_private = false)
      super || Array.method_defined?(method) ||
        @klass.respond_to?(method, include_private) ||
        arel.respond_to?(method, include_private)
    end

    protected

    def method_missing(method, *args, &block)
      if @klass.respond_to?(method)
        @@delegation_mutex.synchronize do
          unless ::ActiveRecord::Delegation.method_defined?(method)
            ::ActiveRecord::Delegation.delegate_to_scoped_klass(method)
          end
        end

        scoping { @klass.send(method, *args, &block) }
      elsif Array.method_defined?(method)
        @@delegation_mutex.synchronize do
          unless ::ActiveRecord::Delegation.method_defined?(method)
            ::ActiveRecord::Delegation.delegate method, :to => :to_a
          end
        end

        to_a.send(method, *args, &block)
      elsif arel.respond_to?(method)
        @@delegation_mutex.synchronize do
          unless ::ActiveRecord::Delegation.method_defined?(method)
            ::ActiveRecord::Delegation.delegate method, :to => :arel
          end
        end

        arel.send(method, *args, &block)
      else
        super
      end
    end
  end
end