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:
commit
0709b60fbe
3 changed files with 73 additions and 73 deletions
|
@ -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
|
||||||
|
|
|
@ -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
|
||||||
|
|
||||||
|
|
|
@ -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
|
||||||
|
|
Loading…
Reference in a new issue