Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Redis instrumentation: support older redis-client #1673

Merged
merged 6 commits into from
Dec 7, 2022
Merged
Show file tree
Hide file tree
Changes from 5 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions .rubocop.yml
Original file line number Diff line number Diff line change
Expand Up @@ -1341,6 +1341,7 @@ Style/MethodCallWithArgsParentheses:
- skip
- source
- stub
- stub_const
AllowedPatterns: [^assert, ^refute]

Style/MethodCallWithoutArgsParentheses:
Expand Down
6 changes: 5 additions & 1 deletion CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@

## v8.14.0

Version 8.14.0 of the agent restores desired Capistrano-based changelog lookup functionalty when a deployment is performed and delivers support for instrumenting Rails custom event notifications.
Version 8.14.0 of the agent restores desired Capistrano-based changelog lookup functionalty when a deployment is performed, delivers support for instrumenting Rails custom event notifications, and fixes potential compatibility issues with Redis gems.

* **Deployment Recipe: Restore desired Capistrano-based changelog lookup behavior**

Expand All @@ -12,6 +12,10 @@

When the new `active_support_custom_events_names` configuration parameter is set equal to an array of custom event names to subscribe to, the agent will now subscribe to each of the names specified and report instrumentation for the events when they take place. [Creating custom events](https://guides.rubyonrails.org/active_support_instrumentation.html#creating-custom-events) is simple and now reporting instrumentation for them to New Relic is simple as well. [PR#1659](https://github.com/newrelic/newrelic-ruby-agent/pull/1659)

* **Bugfix: Support older versions of the redis-client gem, handle unknown Redis database index
fallwith marked this conversation as resolved.
Show resolved Hide resolved

With version 8.13.0 of the agent, support was added for `redis` gem v5+ and the new `redis-client` gem. With versions of `redis-client` older than v0.11, the agent could cause the monitored application to crash when attempting to determine the Redis database index. Version 8.14.0 adds two related improvements. Firstly, support for `redis-client` versions older than v0.11 has been added to get at the database index value. Secondly, the agent will no longer crash or impact the monitored application in the event that the database index cannot be obtained. [Issue#1650](https://github.com/newrelic/newrelic-ruby-agent/issues/1650) [PR#1673](https://github.com/newrelic/newrelic-ruby-agent/pull/1673)
fallwith marked this conversation as resolved.
Show resolved Hide resolved


## v8.13.1

Expand Down
35 changes: 28 additions & 7 deletions lib/new_relic/agent/instrumentation/redis/instrumentation.rb
Original file line number Diff line number Diff line change
Expand Up @@ -27,10 +27,17 @@ def call_pipeline_with_tracing(pipeline)

# Used for Redis 5.x+
def call_pipelined_with_tracing(pipeline)
db = begin
_nr_redis_client_config.db
rescue StandardError => e
NewRelic::Agent.logger.error("Failed to determine configured Redis db value: #{e.class} - #{e.message}")
nil
end

operation = pipeline.flatten.include?('MULTI') ? Constants::MULTI_OPERATION : Constants::PIPELINE_OPERATION
statement = ::NewRelic::Agent::Datastores::Redis.format_pipeline_commands(pipeline)

with_tracing(operation, statement: statement, database: client.config.db) { yield }
with_tracing(operation, statement: statement, database: db) { yield }
end

private
Expand All @@ -52,21 +59,35 @@ def with_tracing(operation, statement: nil, database: nil)
end

def _nr_hostname
_nr_client.path ? Constants::LOCALHOST : _nr_client.host
_nr_redis_client_config.path ? Constants::LOCALHOST : _nr_redis_client_config.host
rescue => e
NewRelic::Agent.logger.debug("Failed to retrieve Redis host: #{e}")
NewRelic::Agent.logger.debug("Failed to retrieve Redis host: #{e.class} - #{e.message}")
Constants::UNKNOWN
end

def _nr_port_path_or_id
_nr_client.path || _nr_client.port
_nr_redis_client_config.path || _nr_redis_client_config.port
rescue => e
NewRelic::Agent.logger.debug("Failed to retrieve Redis port_path_or_id: #{e}")
NewRelic::Agent.logger.debug("Failed to retrieve Redis port_path_or_id: #{e.class} - #{e.message}")
Constants::UNKNOWN
end

def _nr_client
@nr_client ||= self.is_a?(::Redis::Client) ? self : client.config
def _nr_redis_client_config
@nr_config ||= begin
fallwith marked this conversation as resolved.
Show resolved Hide resolved
# redis gem
config = if defined?(::Redis::Client) && self.is_a?(::Redis::Client)
self
# redis-client gem v0.11+ (self is a RedisClient::Middlewares)
elsif respond_to?(:client)
client && client.config
# redis-client gem <0.11 (self is a RedisClient::Middlewares)
elsif defined?(::RedisClient)
::RedisClient.config if ::RedisClient.respond_to?(:config)
end
raise 'Unable to locate the underlying Redis client configuration.' unless config

config
end
end
end
end
69 changes: 69 additions & 0 deletions test/multiverse/suites/redis/redis_instrumentation_test.rb
Original file line number Diff line number Diff line change
Expand Up @@ -4,12 +4,17 @@

require 'redis'
require_relative '../../../helpers/docker'
require_relative '../../../helpers/misc'

if NewRelic::Agent::Datastores::Redis.is_supported_version?
class NewRelic::Agent::Instrumentation::RedisInstrumentationTest < Minitest::Test
include MultiverseHelpers
setup_and_teardown_agent

class FakeClient
include ::NewRelic::Agent::Instrumentation::Redis
end

def after_setup
super
# Default timeout is 5 secs; a flushall takes longer on a busy box (i.e. CI)
Expand Down Expand Up @@ -388,6 +393,70 @@ def test_instrumentation_returns_expected_values
assert_equal 2, @redis.del('foo', 'baz')
end

def test__nr_redis_client_config_with_redis
skip_unless_minitest5_or_above

client = FakeClient.new

client.stub :is_a?, true, [::Redis::Client] do
assert_equal client, client.send(:_nr_redis_client_config)
end
end

def test__nr_redis_client_config_with_redis_client_v0_11
skip_unless_minitest5_or_above

client = FakeClient.new
config = 'the config'
mock_client = MiniTest::Mock.new
mock_client.expect :config, config
client.stub :respond_to?, true, [:client] do
client.stub :client, mock_client do
assert_equal config, client.send(:_nr_redis_client_config)
end
end
end

def test__nr_redis_client_config_with_redis_client_below_v0_11
skip_unless_minitest5_or_above

client = FakeClient.new
config = 'the config'
mock_client = MiniTest::Mock.new
mock_client.expect :config, config

Object.stub_const :RedisClient, mock_client do
assert_equal config, client.send(:_nr_redis_client_config)
end
end

def test__nr_redis_client_config_with_some_unknown_context
skip_unless_minitest5_or_above

client = FakeClient.new

Object.stub_const :RedisClient, nil do
assert_raises StandardError do
client.send(:_nr_redis_client_config)
end
end
end

def test_call_pipelined_with_tracing_uses_a_nil_db_value_if_it_must
client = FakeClient.new
with_tracing_validator = proc do |*args|
assert_equal 2, args.size
assert args.last.key?(:database)
refute args.last[:database]
end

Object.stub_const :RedisClient, nil do
client.stub :with_tracing, with_tracing_validator do
client.call_pipelined_with_tracing([]) { yield_value }
end
end
end

def client
if Gem::Version.new(Redis::VERSION).segments[0] < 4
:client
Expand Down