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

                     
                    



                           
          
            
         

       







                                                   






                                                                                                               
                      
                                            
                                            
 
                                                              
                                                                    

         

                                                                  
         
 
                                                               
 


            






                                                                                       
                     
                                            
                                            
 
                                                               




                                                                    

         
                                                               
 


            

     
require 'active_support/core_ext/object/blank'
require 'active_record/relation/merger'

module ActiveRecord
  module SpawnMethods
    def merge(other)
      if other.is_a?(Array)
        to_a & other
      elsif other
        clone.merge!(other)
      else
        self
      end
    end

    def merge!(other)
      if other.is_a?(Hash)
        Relation::HashMerger.new(self, other).merge
      else
        Relation::Merger.new(self, other).merge
      end
    end

    # Removes from the query the condition(s) specified in +skips+.
    #
    # Example:
    #
    #   Post.order('id asc').except(:order)                  # discards the order condition
    #   Post.where('id > 10').order('id asc').except(:where) # discards the where condition but keeps the order
    #
    def except(*skips)
      result = self.class.new(@klass, table)
      result.default_scoped = default_scoped

      (Relation::MULTI_VALUE_METHODS - skips).each do |method|
        result.send(:"#{method}_values=", send(:"#{method}_values"))
      end

      (Relation::SINGLE_VALUE_METHODS - skips).each do |method|
        result.send(:"#{method}_value=", send(:"#{method}_value"))
      end

      result.extend(*extending_values) if extending_values.any?

      result
    end

    # Removes any condition from the query other than the one(s) specified in +onlies+.
    #
    # Example:
    #
    #   Post.order('id asc').only(:where)         # discards the order condition
    #   Post.order('id asc').only(:where, :order) # uses the specified order
    #
    def only(*onlies)
      result = self.class.new(@klass, table)
      result.default_scoped = default_scoped

      (Relation::MULTI_VALUE_METHODS & onlies).each do |method|
        result.send(:"#{method}_values=", send(:"#{method}_values"))
      end

      (Relation::SINGLE_VALUE_METHODS & onlies).each do |method|
        result.send(:"#{method}_value=", send(:"#{method}_value"))
      end

      result.extend(*extending_values) if extending_values.any?

      result
    end

  end
end