1
0
Fork 0
mirror of https://github.com/rails/rails.git synced 2022-11-09 12:12:34 -05:00

Merge pull request #28526 from kamipo/fix_log_subscriber_to_allow_legacy_binds

Fix `LogSubscriber` to allow legacy `binds`
This commit is contained in:
Rafael França 2017-03-22 19:10:49 -04:00 committed by GitHub
commit 0709b60fbe
3 changed files with 73 additions and 73 deletions

View file

@ -152,16 +152,15 @@ module ActiveRecord
"'#{quote_string(value.to_s)}'" "'#{quote_string(value.to_s)}'"
end end
private def type_casted_binds(binds) # :nodoc:
if binds.first.is_a?(Array)
def type_casted_binds(binds) binds.map { |column, value| type_cast(value, column) }
if binds.first.is_a?(Array) else
binds.map { |column, value| type_cast(value, column) } binds.map { |attr| type_cast(attr.value_for_database) }
else
binds.map { |attr| type_cast(attr.value_for_database) }
end
end end
end
private
def id_value_for_database(value) def id_value_for_database(value)
if primary_key = value.class.primary_key if primary_key = value.class.primary_key
value.instance_variable_get(:@attributes)[primary_key].value_for_database value.instance_variable_get(:@attributes)[primary_key].value_for_database

View file

@ -44,17 +44,17 @@ module ActiveRecord
private private
def type_casted_binds(binds, casted_binds) def type_casted_binds(binds, casted_binds)
casted_binds || binds.map { |attr| type_cast attr.value_for_database } casted_binds || ActiveRecord::Base.connection.type_casted_binds(binds)
end end
def render_bind(attr, type_casted_value) def render_bind(attr, value)
value = if attr.type.binary? && attr.value if attr.is_a?(Array)
"<#{attr.value_for_database.to_s.bytesize} bytes of binary data>" attr = attr.first
else elsif attr.type.binary? && attr.value
type_casted_value value = "<#{attr.value_for_database.to_s.bytesize} bytes of binary data>"
end end
[attr.name, value] [attr && attr.name, value]
end end
def colorize_payload_name(name, payload_name) def colorize_payload_name(name, payload_name)
@ -89,10 +89,6 @@ module ActiveRecord
def logger def logger
ActiveRecord::Base.logger ActiveRecord::Base.logger
end end
def type_cast(value)
ActiveRecord::Base.connection.type_cast(value)
end
end end
end end

View file

@ -3,36 +3,36 @@ require "models/topic"
require "models/author" require "models/author"
require "models/post" require "models/post"
module ActiveRecord if ActiveRecord::Base.connection.supports_statement_cache? &&
class BindParameterTest < ActiveRecord::TestCase ActiveRecord::Base.connection.prepared_statements
fixtures :topics, :authors, :posts module ActiveRecord
class BindParameterTest < ActiveRecord::TestCase
fixtures :topics, :authors, :posts
class LogListener class LogListener
attr_accessor :calls attr_accessor :calls
def initialize def initialize
@calls = [] @calls = []
end
def call(*args)
calls << args
end
end end
def call(*args) def setup
calls << args super
@connection = ActiveRecord::Base.connection
@subscriber = LogListener.new
@pk = Topic.columns_hash[Topic.primary_key]
@subscription = ActiveSupport::Notifications.subscribe("sql.active_record", @subscriber)
end end
end
def setup def teardown
super ActiveSupport::Notifications.unsubscribe(@subscription)
@connection = ActiveRecord::Base.connection end
@subscriber = LogListener.new
@pk = Topic.columns_hash[Topic.primary_key]
@subscription = ActiveSupport::Notifications.subscribe("sql.active_record", @subscriber)
end
teardown do
ActiveSupport::Notifications.unsubscribe(@subscription)
end
if ActiveRecord::Base.connection.supports_statement_cache? &&
ActiveRecord::Base.connection.prepared_statements
def test_bind_from_join_in_subquery def test_bind_from_join_in_subquery
subquery = Author.joins(:thinking_posts).where(name: "David") subquery = Author.joins(:thinking_posts).where(name: "David")
scope = Author.from(subquery, "authors").where(id: 1) scope = Author.from(subquery, "authors").where(id: 1)
@ -56,43 +56,48 @@ module ActiveRecord
assert message, "expected a message with binds" assert message, "expected a message with binds"
end end
def test_logs_bind_vars_after_type_cast def test_logs_binds_after_type_cast
binds = [Relation::QueryAttribute.new("id", "10", Type::Integer.new)] binds = [Relation::QueryAttribute.new("id", "10", Type::Integer.new)]
type_casted_binds = binds.map { |attr| type_cast(attr.value_for_database) } assert_logs_binds(binds)
payload = { end
name: "SQL",
sql: "select * from topics where id = ?",
binds: binds,
type_casted_binds: type_casted_binds
}
event = ActiveSupport::Notifications::Event.new(
"foo",
Time.now,
Time.now,
123,
payload)
logger = Class.new(ActiveRecord::LogSubscriber) { def test_logs_legacy_binds_after_type_cast
attr_reader :debugs binds = [[@pk, "10"]]
def initialize assert_logs_binds(binds)
super
@debugs = []
end
def debug(str)
@debugs << str
end
}.new
logger.sql event
assert_match([[@pk.name, 10]].inspect, logger.debugs.first)
end end
private private
def assert_logs_binds(binds)
payload = {
name: "SQL",
sql: "select * from topics where id = ?",
binds: binds,
type_casted_binds: @connection.type_casted_binds(binds)
}
def type_cast(value) event = ActiveSupport::Notifications::Event.new(
ActiveRecord::Base.connection.type_cast(value) "foo",
end Time.now,
Time.now,
123,
payload)
logger = Class.new(ActiveRecord::LogSubscriber) {
attr_reader :debugs
def initialize
super
@debugs = []
end
def debug(str)
@debugs << str
end
}.new
logger.sql(event)
assert_match([[@pk.name, 10]].inspect, logger.debugs.first)
end
end end
end end
end end