提交 7d1dc984 编写于 作者: R Ryuta Kamizono

Fix explain logging with binds

`binds` is an array of a query attribute since Active Record 5.0.
上级 a8a3a8cc
...@@ -16,15 +16,14 @@ def collecting_queries_for_explain # :nodoc: ...@@ -16,15 +16,14 @@ def collecting_queries_for_explain # :nodoc:
# Makes the adapter execute EXPLAIN for the tuples of queries and bindings. # Makes the adapter execute EXPLAIN for the tuples of queries and bindings.
# Returns a formatted string ready to be logged. # Returns a formatted string ready to be logged.
def exec_explain(queries) # :nodoc: def exec_explain(queries) # :nodoc:
str = queries.map do |sql, bind| str = queries.map do |sql, binds|
[].tap do |msg| msg = "EXPLAIN for: #{sql}"
msg << "EXPLAIN for: #{sql}" unless binds.empty?
unless bind.empty? msg << " "
bind_msg = bind.map {|col, val| [col.name, val]}.inspect msg << binds.map { |attr| render_bind(attr) }.inspect
msg.last << " #{bind_msg}" end
end msg << "\n"
msg << connection.explain(sql, bind) msg << connection.explain(sql, binds)
end.join("\n")
end.join("\n") end.join("\n")
# Overriding inspect to be more human readable, especially in the console. # Overriding inspect to be more human readable, especially in the console.
...@@ -34,5 +33,17 @@ def str.inspect ...@@ -34,5 +33,17 @@ def str.inspect
str str
end end
private
def render_bind(attr)
value = if attr.type.binary? && attr.value
"<#{attr.value_for_database.to_s.bytesize} bytes of binary data>"
else
connection.type_cast(attr.value_for_database)
end
[attr.name, value]
end
end end
end end
...@@ -7,14 +7,14 @@ class PostgreSQLExplainTest < ActiveRecord::PostgreSQLTestCase ...@@ -7,14 +7,14 @@ class PostgreSQLExplainTest < ActiveRecord::PostgreSQLTestCase
def test_explain_for_one_query def test_explain_for_one_query
explain = Developer.where(id: 1).explain explain = Developer.where(id: 1).explain
assert_match %r(EXPLAIN for: SELECT "developers".* FROM "developers" WHERE "developers"."id" = \$?1), explain assert_match %r(EXPLAIN for: SELECT "developers".* FROM "developers" WHERE "developers"."id" = (?:\$1 \[\["id", 1\]\]|1)), explain
assert_match %(QUERY PLAN), explain assert_match %(QUERY PLAN), explain
end end
def test_explain_with_eager_loading def test_explain_with_eager_loading
explain = Developer.where(id: 1).includes(:audit_logs).explain explain = Developer.where(id: 1).includes(:audit_logs).explain
assert_match %(QUERY PLAN), explain assert_match %(QUERY PLAN), explain
assert_match %r(EXPLAIN for: SELECT "developers".* FROM "developers" WHERE "developers"."id" = \$?1), explain assert_match %r(EXPLAIN for: SELECT "developers".* FROM "developers" WHERE "developers"."id" = (?:\$1 \[\["id", 1\]\]|1)), explain
assert_match %(EXPLAIN for: SELECT "audit_logs".* FROM "audit_logs" WHERE "audit_logs"."developer_id" = 1), explain assert_match %(EXPLAIN for: SELECT "audit_logs".* FROM "audit_logs" WHERE "audit_logs"."developer_id" = 1), explain
end end
end end
...@@ -7,13 +7,13 @@ class SQLite3ExplainTest < ActiveRecord::SQLite3TestCase ...@@ -7,13 +7,13 @@ class SQLite3ExplainTest < ActiveRecord::SQLite3TestCase
def test_explain_for_one_query def test_explain_for_one_query
explain = Developer.where(id: 1).explain explain = Developer.where(id: 1).explain
assert_match %r(EXPLAIN for: SELECT "developers".* FROM "developers" WHERE "developers"."id" = (?:\?|1)), explain assert_match %r(EXPLAIN for: SELECT "developers".* FROM "developers" WHERE "developers"."id" = (?:\? \[\["id", 1\]\]|1)), explain
assert_match(/(SEARCH )?TABLE developers USING (INTEGER )?PRIMARY KEY/, explain) assert_match(/(SEARCH )?TABLE developers USING (INTEGER )?PRIMARY KEY/, explain)
end end
def test_explain_with_eager_loading def test_explain_with_eager_loading
explain = Developer.where(id: 1).includes(:audit_logs).explain explain = Developer.where(id: 1).includes(:audit_logs).explain
assert_match %r(EXPLAIN for: SELECT "developers".* FROM "developers" WHERE "developers"."id" = (?:\?|1)), explain assert_match %r(EXPLAIN for: SELECT "developers".* FROM "developers" WHERE "developers"."id" = (?:\? \[\["id", 1\]\]|1)), explain
assert_match(/(SEARCH )?TABLE developers USING (INTEGER )?PRIMARY KEY/, explain) assert_match(/(SEARCH )?TABLE developers USING (INTEGER )?PRIMARY KEY/, explain)
assert_match %(EXPLAIN for: SELECT "audit_logs".* FROM "audit_logs" WHERE "audit_logs"."developer_id" = 1), explain assert_match %(EXPLAIN for: SELECT "audit_logs".* FROM "audit_logs" WHERE "audit_logs"."developer_id" = 1), explain
assert_match(/(SCAN )?TABLE audit_logs/, explain) assert_match(/(SCAN )?TABLE audit_logs/, explain)
......
...@@ -46,11 +46,8 @@ def test_exec_explain_with_no_binds ...@@ -46,11 +46,8 @@ def test_exec_explain_with_no_binds
end end
def test_exec_explain_with_binds def test_exec_explain_with_binds
object = Struct.new(:name)
cols = [object.new('wadus'), object.new('chaflan')]
sqls = %w(foo bar) sqls = %w(foo bar)
binds = [[[cols[0], 1]], [[cols[1], 2]]] binds = [[bind_param('wadus', 1)], [bind_param('chaflan', 2)]]
queries = sqls.zip(binds) queries = sqls.zip(binds)
stub_explain_for_query_plans(["query plan foo\n", "query plan bar\n"]) do stub_explain_for_query_plans(["query plan foo\n", "query plan bar\n"]) do
...@@ -83,5 +80,8 @@ def stub_explain_for_query_plans(query_plans = ['query plan foo', 'query plan ba ...@@ -83,5 +80,8 @@ def stub_explain_for_query_plans(query_plans = ['query plan foo', 'query plan ba
end end
end end
def bind_param(name, value)
ActiveRecord::Relation::QueryAttribute.new(name, value, ActiveRecord::Type::Value.new)
end
end end
end end
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册