2016-12-20 09:48:04 -05:00
|
|
|
module ActiveRecord
|
|
|
|
class QueryRecorder
|
2017-04-04 17:52:59 -04:00
|
|
|
attr_reader :log, :cached
|
2016-12-20 09:48:04 -05:00
|
|
|
|
|
|
|
def initialize(&block)
|
|
|
|
@log = []
|
2017-04-04 17:52:59 -04:00
|
|
|
@cached = []
|
2016-12-20 09:48:04 -05:00
|
|
|
ActiveSupport::Notifications.subscribed(method(:callback), 'sql.active_record', &block)
|
|
|
|
end
|
|
|
|
|
|
|
|
def callback(name, start, finish, message_id, values)
|
2017-04-04 17:52:59 -04:00
|
|
|
if values[:name]&.include?("CACHE")
|
|
|
|
@cached << values[:sql]
|
|
|
|
elsif !values[:name]&.include?("SCHEMA")
|
|
|
|
@log << values[:sql]
|
|
|
|
end
|
2016-12-20 09:48:04 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def count
|
|
|
|
@log.count
|
|
|
|
end
|
|
|
|
|
2017-04-04 17:52:59 -04:00
|
|
|
def cached_count
|
|
|
|
@cached.count
|
|
|
|
end
|
|
|
|
|
2016-12-20 09:48:04 -05:00
|
|
|
def log_message
|
|
|
|
@log.join("\n\n")
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
RSpec::Matchers.define :exceed_query_limit do |expected|
|
|
|
|
supports_block_expectations
|
|
|
|
|
|
|
|
match do |block|
|
2017-09-14 05:40:19 -04:00
|
|
|
query_count(&block) > expected_count + threshold
|
2016-12-20 09:48:04 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
failure_message_when_negated do |actual|
|
2017-09-14 05:40:19 -04:00
|
|
|
threshold_message = threshold > 0 ? " (+#{@threshold})" : ''
|
|
|
|
counts = "#{expected_count}#{threshold_message}"
|
|
|
|
"Expected a maximum of #{counts} queries, got #{actual_count}:\n\n#{log_message}"
|
|
|
|
end
|
|
|
|
|
|
|
|
def with_threshold(threshold)
|
|
|
|
@threshold = threshold
|
|
|
|
self
|
|
|
|
end
|
|
|
|
|
|
|
|
def threshold
|
|
|
|
@threshold.to_i
|
|
|
|
end
|
|
|
|
|
|
|
|
def expected_count
|
|
|
|
if expected.is_a?(ActiveRecord::QueryRecorder)
|
|
|
|
expected.count
|
|
|
|
else
|
|
|
|
expected
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def actual_count
|
|
|
|
@recorder.count
|
2016-12-20 09:48:04 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def query_count(&block)
|
|
|
|
@recorder = ActiveRecord::QueryRecorder.new(&block)
|
|
|
|
@recorder.count
|
|
|
|
end
|
2017-09-14 05:40:19 -04:00
|
|
|
|
|
|
|
def log_message
|
|
|
|
if expected.is_a?(ActiveRecord::QueryRecorder)
|
|
|
|
extra_queries = (expected.log - @recorder.log).join("\n\n")
|
|
|
|
"Extra queries: \n\n #{extra_queries}"
|
|
|
|
else
|
|
|
|
@recorder.log_message
|
|
|
|
end
|
|
|
|
end
|
2016-12-20 09:48:04 -05:00
|
|
|
end
|