-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathsql.rb
81 lines (61 loc) · 1.79 KB
/
sql.rb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
# frozen_string_literal: true
require "active_support"
require "rspec"
require_relative "sql/query_matcher"
# We are building within the RSpec namespace for consistency and convenience.
# We are not part of the RSpec team though.
module RSpec
# RSpec::Sql contains our code.
module Sql; end
Matchers.define :query_database do |expected = nil|
match do |block|
@queries = scribe_queries(&block)
@matcher = Sql::QueryMatcher.new(@queries, expected)
expected = matcher.expected
matcher.matches?
end
failure_message do |_block|
if expected.nil?
return "Expected at least one database query but observed none."
end
<<~MESSAGE
Expected database queries: #{expected}
Actual database queries: #{matcher.actual}
Diff: #{diff(matcher.actual, expected)}
Full query log:
#{query_descriptions.join("\n")}
MESSAGE
end
failure_message_when_negated do |_block|
<<~TXT
Expected no database queries but observed:
#{query_descriptions.join("\n")}
TXT
end
def supports_block_expectations?
true
end
def query_descriptions
@queries.map { |q| "#{q[:name]} #{q[:sql]}" }
end
def matcher
@matcher
end
def diff(actual, expected)
if expected.is_a?(Numeric)
change = actual - expected
format("%+d", change)
else
Expectations.differ.diff_as_object(actual, expected)
end
end
def scribe_queries(&)
queries = []
logger = lambda do |_name, _started, _finished, _unique_id, payload|
queries << payload unless %w[CACHE SCHEMA].include?(payload[:name])
end
ActiveSupport::Notifications.subscribed(logger, "sql.active_record", &)
queries
end
end
end