query_cache.rb 2.6 KB
Newer Older
1 2 3 4
module ActiveRecord
  module ConnectionAdapters # :nodoc:
    module QueryCache
      class << self
5
        def included(base) #:nodoc:
6 7 8 9 10
          dirties_query_cache base, :insert, :update, :delete
        end

        def dirties_query_cache(base, *method_names)
          method_names.each do |method_name|
11
            base.class_eval <<-end_code, __FILE__, __LINE__ + 1
12 13 14 15
              def #{method_name}(*)
                clear_query_cache if @query_cache_enabled
                super
              end
16 17 18 19 20
            end_code
          end
        end
      end

21
      attr_reader :query_cache, :query_cache_enabled
N
Nick Sieger 已提交
22

23 24 25 26 27 28
      def initialize(*)
        super
        @query_cache         = Hash.new { |h,sql| h[sql] = {} }
        @query_cache_enabled = false
      end

29 30
      # Enable the query cache within the block.
      def cache
31
        old, @query_cache_enabled = @query_cache_enabled, true
32 33
        yield
      ensure
34
        @query_cache_enabled = old
35
        clear_query_cache unless @query_cache_enabled
36 37
      end

38 39 40 41 42 43 44 45
      def enable_query_cache!
        @query_cache_enabled = true
      end

      def disable_query_cache!
        @query_cache_enabled = false
      end

46 47
      # Disable the query cache within the block.
      def uncached
48
        old, @query_cache_enabled = @query_cache_enabled, false
49 50
        yield
      ensure
51
        @query_cache_enabled = old
52 53
      end

P
Pratik Naik 已提交
54 55 56 57 58 59
      # Clears the query cache.
      #
      # One reason you may wish to call this method explicitly is between queries
      # that ask the database to randomize results. Otherwise the cache would see
      # the same SQL query and repeatedly return the same result each time, silently
      # undermining the randomness you were expecting.
60
      def clear_query_cache
61
        @query_cache.clear
62 63
      end

64
      def select_all(arel, name = nil, binds = [])
65
        if @query_cache_enabled && !locked?(arel)
66
          sql = to_sql(arel, binds)
67
          cache_sql(sql, binds) { super(sql, name, binds) }
68
        else
69
          super
70 71 72 73
        end
      end

      private
74

75 76 77 78 79 80 81 82 83
      def cache_sql(sql, binds)
        result =
          if @query_cache[sql].key?(binds)
            ActiveSupport::Notifications.instrument("sql.active_record",
              :sql => sql, :binds => binds, :name => "CACHE", :connection_id => object_id)
            @query_cache[sql][binds]
          else
            @query_cache[sql][binds] = yield
          end
84
        result.dup
85
      end
86

87 88
      # If arel is locked this is a SELECT ... FOR UPDATE or somesuch. Such
      # queries should not be cached.
89
      def locked?(arel)
90
        arel.respond_to?(:locked) && arel.locked
91
      end
92 93 94
    end
  end
end