2018-06-03 06:31:41 -04:00
|
|
|
module ExceedQueryLimitHelpers
|
2017-09-14 05:40:19 -04:00
|
|
|
def with_threshold(threshold)
|
|
|
|
@threshold = threshold
|
|
|
|
self
|
|
|
|
end
|
|
|
|
|
2017-12-04 10:58:44 -05:00
|
|
|
def for_query(query)
|
|
|
|
@query = query
|
|
|
|
self
|
|
|
|
end
|
|
|
|
|
2017-09-14 05:40:19 -04:00
|
|
|
def threshold
|
|
|
|
@threshold.to_i
|
|
|
|
end
|
|
|
|
|
|
|
|
def expected_count
|
|
|
|
if expected.is_a?(ActiveRecord::QueryRecorder)
|
|
|
|
expected.count
|
|
|
|
else
|
|
|
|
expected
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def actual_count
|
2017-12-04 10:58:44 -05:00
|
|
|
@actual_count ||= if @query
|
|
|
|
recorder.log.select { |recorded| recorded =~ @query }.size
|
|
|
|
else
|
|
|
|
recorder.count
|
|
|
|
end
|
2016-12-20 09:48:04 -05:00
|
|
|
end
|
|
|
|
|
2017-12-04 10:58:44 -05:00
|
|
|
def recorder
|
2018-06-03 06:31:41 -04:00
|
|
|
@recorder ||= ActiveRecord::QueryRecorder.new(skip_cached: skip_cached, &@subject_block)
|
2016-12-20 09:48:04 -05:00
|
|
|
end
|
2017-09-14 05:40:19 -04:00
|
|
|
|
2017-11-22 06:12:47 -05:00
|
|
|
def count_queries(queries)
|
|
|
|
queries.each_with_object(Hash.new(0)) { |query, counts| counts[query] += 1 }
|
|
|
|
end
|
|
|
|
|
2017-09-14 05:40:19 -04:00
|
|
|
def log_message
|
|
|
|
if expected.is_a?(ActiveRecord::QueryRecorder)
|
2017-11-22 06:12:47 -05:00
|
|
|
counts = count_queries(expected.log)
|
|
|
|
extra_queries = @recorder.log.reject { |query| counts[query] -= 1 unless counts[query].zero? }
|
|
|
|
extra_queries_display = count_queries(extra_queries).map { |query, count| "[#{count}] #{query}" }
|
|
|
|
|
|
|
|
(['Extra queries:'] + extra_queries_display).join("\n\n")
|
2017-09-14 05:40:19 -04:00
|
|
|
else
|
|
|
|
@recorder.log_message
|
|
|
|
end
|
|
|
|
end
|
2018-06-03 06:31:41 -04:00
|
|
|
|
|
|
|
def skip_cached
|
|
|
|
true
|
|
|
|
end
|
|
|
|
|
|
|
|
def verify_count(&block)
|
|
|
|
@subject_block = block
|
|
|
|
actual_count > expected_count + threshold
|
|
|
|
end
|
|
|
|
|
|
|
|
def failure_message
|
|
|
|
threshold_message = threshold > 0 ? " (+#{@threshold})" : ''
|
|
|
|
counts = "#{expected_count}#{threshold_message}"
|
|
|
|
"Expected a maximum of #{counts} queries, got #{actual_count}:\n\n#{log_message}"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
RSpec::Matchers.define :exceed_all_query_limit do |expected|
|
|
|
|
supports_block_expectations
|
|
|
|
|
|
|
|
include ExceedQueryLimitHelpers
|
|
|
|
|
|
|
|
match do |block|
|
|
|
|
verify_count(&block)
|
|
|
|
end
|
|
|
|
|
|
|
|
failure_message_when_negated do |actual|
|
|
|
|
failure_message
|
|
|
|
end
|
|
|
|
|
|
|
|
def skip_cached
|
|
|
|
false
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# Excludes cached methods from the query count
|
|
|
|
RSpec::Matchers.define :exceed_query_limit do |expected|
|
|
|
|
supports_block_expectations
|
|
|
|
|
|
|
|
include ExceedQueryLimitHelpers
|
|
|
|
|
|
|
|
match do |block|
|
|
|
|
verify_count(&block)
|
|
|
|
end
|
|
|
|
|
|
|
|
failure_message_when_negated do |actual|
|
|
|
|
failure_message
|
|
|
|
end
|
2016-12-20 09:48:04 -05:00
|
|
|
end
|