association_proxy.rb 3.0 KB
Newer Older
1 2 3 4
module ActiveRecord
  module Associations
    class AssociationProxy #:nodoc:
      alias_method :proxy_respond_to?, :respond_to?
5 6
      alias_method :proxy_extend, :extend
      instance_methods.each { |m| undef_method m unless m =~ /(^__|^nil\?|^proxy_respond_to\?|^proxy_extend|^send)/ }
7 8 9 10 11

      def initialize(owner, association_name, association_class_name, association_class_primary_key_name, options)
        @owner = owner
        @options = options
        @association_name = association_name
12
        @association_class = eval(association_class_name, nil, __FILE__, __LINE__)
13 14
        @association_class_primary_key_name = association_class_primary_key_name

15 16
        proxy_extend(options[:extend]) if options[:extend]

17 18 19
        reset
      end
      
20 21 22 23 24 25 26
      def respond_to?(symbol, include_priv = false)
        proxy_respond_to?(symbol, include_priv) || (load_target && @target.respond_to?(symbol, include_priv))
      end

      # Explicitly proxy === because the instance method removal above
      # doesn't catch it.
      def ===(other)
27
        load_target
28
        other === @target
29 30
      end

31 32 33
      def reload
        reset
        load_target
34 35 36 37 38
      end

      def loaded?
        @loaded
      end
39
      
40 41 42 43
      def loaded
        @loaded = true
      end
      
44 45 46 47
      def target
        @target
      end
      
48 49 50 51
      def target=(t)
        @target = t
        @loaded = true
      end
52
      
53
      protected
54 55 56 57
        def dependent?
          @options[:dependent] || false
        end
        
58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76
        def quoted_record_ids(records)
          records.map { |record| record.quoted_id }.join(',')
        end

        def interpolate_sql_options!(options, *keys)
          keys.each { |key| options[key] &&= interpolate_sql(options[key]) }
        end

        def interpolate_sql(sql, record = nil)
          @owner.send(:interpolate_sql, sql, record)
        end

        def sanitize_sql(sql)
          @association_class.send(:sanitize_sql, sql)
        end

        def extract_options_from_args!(args)
          @owner.send(:extract_options_from_args!, args)
        end
77
        
78
      private
79 80 81 82 83
        def method_missing(method, *args, &block)
          load_target
          @target.send(method, *args, &block)
        end

84
        def load_target
85
          if !@owner.new_record? || foreign_key_present
86 87 88 89 90 91
            begin
              @target = find_target if not loaded?
            rescue ActiveRecord::RecordNotFound
              reset
            end
          end
92
          @loaded = true if @target
93 94 95
          @target
        end

96 97 98 99 100 101
        # Can be overwritten by associations that might have the foreign key available for an association without
        # having the object itself (and still being a new record). Currently, only belongs_to present this scenario.
        def foreign_key_present
          false
        end

102 103 104 105 106
        def raise_on_type_mismatch(record)
          raise ActiveRecord::AssociationTypeMismatch, "#{@association_class} expected, got #{record.class}" unless record.is_a?(@association_class)
        end
    end
  end
107
end