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

instrument Rails custom events #1659

Merged
merged 15 commits into from
Dec 2, 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
6 changes: 5 additions & 1 deletion CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,12 +2,16 @@

## v8.14.0

Version 8.14.0 of the agent restores desired Capistrano-based changelog lookup functionalty when a deployment is performed.
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.

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

The New Relic Ruby agent offers [a Capistrano recipe for recording app deployments](https://docs.newrelic.com/docs/apm/agents/ruby-agent/features/record-deployments-ruby-agent/#capistrano3). The recipe code was significantly cleaned up with [PR#1498](https://github.com/newrelic/newrelic-ruby-agent/pull/1498) which inadvertently changed the way the recipe handles the changelog for a deployment. Community member [@arthurwozniak](https://github.com/arthurwozniak) spotted and corrected this change in order to restore the desired changelog lookup functionality while retaining all of the previous cleanup. Thank you very much for your contribution, [@arthurwozniak](https://github.com/arthurwozniak)! [PR#1653](https://github.com/newrelic/newrelic-ruby-agent/pull/1653)

* **Support for Rails ActiveSupport::Notifications for custom events**

When the new `custom_events_instrumentation_topics` configuration parameter is be set equal to a comma delimited list of custom event topics to subscribe to, the agent will now subscribe to each of the topics 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)
fallwith marked this conversation as resolved.
Show resolved Hide resolved


## v8.13.1

Expand Down
19 changes: 19 additions & 0 deletions lib/new_relic/agent/configuration/default_source.rb
Original file line number Diff line number Diff line change
Expand Up @@ -422,6 +422,17 @@ def self.enforce_fallback(allowed_values: nil, fallback: nil)
:allowed_from_server => false,
:description => 'Path to <b>newrelic.yml</b>. If undefined, the agent checks the following directories (in order): <b>config/newrelic.yml</b>, <b>newrelic.yml</b>, <b>$HOME/.newrelic/newrelic.yml</b> and <b>$HOME/newrelic.yml</b>.'
},
:'custom_events_instrumentation_topics' => {
:default => [],
:public => true,
:type => Array,
:allowed_from_server => true,
fallwith marked this conversation as resolved.
Show resolved Hide resolved
:dynamic_name => true,
fallwith marked this conversation as resolved.
Show resolved Hide resolved
:description => <<-DESCRIPTION
A comma separated list of ActiveSupport custom event topics to subscribe to and
fallwith marked this conversation as resolved.
Show resolved Hide resolved
provide instrumentation for. Ex: "my.custom.event,another.event,a.third.event"
fallwith marked this conversation as resolved.
Show resolved Hide resolved
DESCRIPTION
},
:'exclude_newrelic_header' => {
:default => false,
:public => true,
Expand Down Expand Up @@ -1187,6 +1198,14 @@ def self.enforce_fallback(allowed_values: nil, fallback: nil)
:allowed_from_server => false,
:description => deprecated_description(:'instrumentation.curb', 'If `true`, disables instrumentation for the curb gem.')
},
:disable_custom_events_instrumentation => {
:default => false,
:public => true,
:type => Boolean,
:dynamic_name => true,
fallwith marked this conversation as resolved.
Show resolved Hide resolved
:allowed_from_server => false,
:description => 'If `true`, disables ActiveSupport custom events instrumentation.'
},
:disable_database_instrumentation => {
:default => false,
:public => true,
Expand Down
37 changes: 37 additions & 0 deletions lib/new_relic/agent/instrumentation/custom_events_subscriber.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
# This file is distributed under New Relic's license terms.
# See https://github.com/newrelic/newrelic-ruby-agent/blob/main/LICENSE for complete details.
# frozen_string_literal: true

require 'new_relic/agent/instrumentation/notifications_subscriber'

# Listen for ActiveSupport::Notifications events for custom events
module NewRelic::Agent::Instrumentation
class CustomEventsSubscriber < NotificationsSubscriber
def start(name, id, _payload) # THREAD_LOCAL_ACCESS
return unless state.is_execution_traced?

finishable = NewRelic::Agent::Tracer.start_transaction_or_segment(name: transaction_name(name),
category: :custom_events)
push_segment(id, finishable)
rescue => e
log_notification_error(e, name, 'start')
end

def finish(name, id, payload) # THREAD_LOCAL_ACCESS
return unless state.is_execution_traced?

NewRelic::Agent.notice_error(payload[:exception_object]) if payload.key?(:exception_object)

finishable = pop_segment(id)
finishable.finish if finishable
rescue => e
log_notification_error(e, name, 'finish')
end

private

def transaction_name(name)
"Controller/ActiveSupport/CustomEvents/#{name}"
fallwith marked this conversation as resolved.
Show resolved Hide resolved
end
end
end
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
# This file is distributed under New Relic's license terms.
# See https://github.com/newrelic/newrelic-ruby-agent/blob/main/LICENSE for complete details.
# frozen_string_literal: true

require 'new_relic/agent/instrumentation/custom_events_subscriber'
require 'new_relic/agent/prepend_supportability'

DependencyDetection.defer do
@name = :custom_event_notifications

depends_on do
defined?(::Rails::VERSION::MAJOR) &&
::Rails::VERSION::MAJOR.to_i >= 5 &&
defined?(::ActiveSupport::Notifications) &&
defined?(::ActiveSupport::IsolatedExecutionState)
end

depends_on do
!::NewRelic::Agent.config[:disable_custom_events_instrumentation] &&
!::NewRelic::Agent.config[:custom_events_instrumentation_topics].empty? &&
!::NewRelic::Agent::Instrumentation::CustomEventsSubscriber.subscribed?
end

executes do
::NewRelic::Agent.logger.info('Installing notifications based ActiveSupport custom events instrumentation')
end

executes do
::NewRelic::Agent.config[:custom_events_instrumentation_topics].each do |topic|
::ActiveSupport::Notifications.subscribe(topic, NewRelic::Agent::Instrumentation::CustomEventsSubscriber.new)
end
end
end
7 changes: 7 additions & 0 deletions newrelic.yml
Original file line number Diff line number Diff line change
Expand Up @@ -130,6 +130,10 @@ common: &default_settings
# and $HOME/newrelic.yml.
# config_path: newrelic.yml

# A comma separated list of ActiveSupport custom event topics to subscribe to and
# provide instrumentation for. Ex: "my.custom.event,another.event,a.third.event"
# custom_events_instrumentation_topics: ""

# If true, enables cross application tracing. Cross application tracing is now
# deprecated, and disabled by default. Distributed tracing is replacing cross
# application tracing as the default means of tracing between services.
Expand Down Expand Up @@ -173,6 +177,9 @@ common: &default_settings
# If true, the agent won't sample the CPU usage of the host process.
# disable_cpu_sampler: false

# If true, disables ActiveSupport custom events instrumentation.
# disable_custom_events_instrumentation: false

# If true, disables DataMapper instrumentation.
# disable_data_mapper: false

Expand Down
6 changes: 6 additions & 0 deletions test/helpers/misc.rb
Original file line number Diff line number Diff line change
Expand Up @@ -118,3 +118,9 @@ def wait_until_not_nil(give_up_after = 3, &block)
current_tries += 1
end
end

def skip_unless_minitest5_or_above
fallwith marked this conversation as resolved.
Show resolved Hide resolved
return if defined?(MiniTest::VERSION) && MiniTest::VERSION > '5'

skip 'This test requires MiniTest v5+'
end
Original file line number Diff line number Diff line change
@@ -0,0 +1,97 @@
# This file is distributed under New Relic's license terms.
# See https://github.com/newrelic/newrelic-ruby-agent/blob/main/LICENSE for complete details.
# frozen_string_literal: true

require_relative '../../../../test_helper'
require 'new_relic/agent/instrumentation/custom_events_subscriber'

module NewRelic::Agent::Instrumentation
class CustomEventsSubscriberTest < Minitest::Test
TOPIC = 'abre los ojos'
fallwith marked this conversation as resolved.
Show resolved Hide resolved
ID = 1138
SUBSCRIBER = NewRelic::Agent::Instrumentation::CustomEventsSubscriber.new

def test_start
in_transaction do |txn|
time = Time.now.to_f
SUBSCRIBER.start(TOPIC, ID, {})
segment = txn.segments.last

assert_in_delta time, segment.start_time
assert_equal "Nested/Controller/ActiveSupport/CustomEvents/#{TOPIC}", segment.name
end
end

def test_start_when_not_traced
SUBSCRIBER.state.stub :is_execution_traced?, false do
in_transaction do |txn|
SUBSCRIBER.start(TOPIC, ID, {})

assert_empty txn.segments
end
end
end

def test_start_with_exception_raised
logger = MiniTest::Mock.new

NewRelic::Agent.stub :logger, logger do
logger.expect :error, nil, [/Error during .* callback/]
logger.expect :log_exception, nil, [:error, ArgumentError]

in_transaction do |txn|
NewRelic::Agent::Tracer.stub :start_transaction_or_segment, -> { raise 'kaboom' } do
SUBSCRIBER.start(TOPIC, ID, {})
end

assert_equal 1, txn.segments.size
end
end
logger.verify
end

def test_finish
in_transaction do |txn|
started_segment = NewRelic::Agent::Tracer.start_transaction_or_segment(name: TOPIC, category: :testing)
SUBSCRIBER.push_segment(ID, started_segment)

time = Time.now.to_f
SUBSCRIBER.finish(TOPIC, ID, {})
segment = txn.segments.last

assert_in_delta time, segment.end_time
assert_predicate(segment, :finished?)
end
end

def test_finish_with_exception_payload
skip_unless_minitest5_or_above

noticed = false
exception_object = StandardError.new
NewRelic::Agent.stub :notice_error, ->(_) { noticed = true }, [exception_object] do
SUBSCRIBER.finish(TOPIC, ID, {exception_object: exception_object})
end

assert noticed
end

def test_finish_with_exception_raised
logger = MiniTest::Mock.new

NewRelic::Agent.stub :logger, logger do
logger.expect :error, nil, [/Error during .* callback/]
logger.expect :log_exception, nil, [:error, RuntimeError]

in_transaction do |txn|
SUBSCRIBER.state.stub :is_execution_traced?, -> { raise 'kaboom' } do
SUBSCRIBER.finish(TOPIC, ID, {})
end

assert_equal 1, txn.segments.size
end
end
logger.verify
end
end
end
12 changes: 4 additions & 8 deletions test/new_relic/recipes/helpers/send_deployment_test.rb
Original file line number Diff line number Diff line change
Expand Up @@ -24,12 +24,8 @@ def tester
@tester ||= Tester.new
end

def minitest5?
defined?(MiniTest::VERSION) && MiniTest::VERSION > '5'
end

def test_fetch_changelog_initial_fetch_succeeds_and_using_scm_then_use_fetched_value
skip 'Stubbing uses MiniTest v5 syntax' unless minitest5?
skip_unless_minitest5_or_above

tester.stub :has_scm?, true do
tester.stub :fetch, Tester::NR_CHANGELOG, [:nr_changelog] do
Expand All @@ -39,7 +35,7 @@ def test_fetch_changelog_initial_fetch_succeeds_and_using_scm_then_use_fetched_v
end

def test_fetch_changelog_initial_fetch_succeeds_and_not_using_scm_then_use_fetched_value
skip 'Stubbing uses MiniTest v5 syntax' unless minitest5?
skip_unless_minitest5_or_above

tester.stub :has_scm?, false do
tester.stub :fetch, Tester::NR_CHANGELOG, [:nr_changelog] do
Expand All @@ -49,7 +45,7 @@ def test_fetch_changelog_initial_fetch_succeeds_and_not_using_scm_then_use_fetch
end

def test_fetch_changelog_initial_fetch_fails_and_using_scm_then_perform_lookup
skip 'Stubbing uses MiniTest v5 syntax' unless minitest5?
skip_unless_minitest5_or_above

tester.stub :has_scm?, true do
tester.stub :fetch, nil, [:nr_changelog] do
Expand All @@ -59,7 +55,7 @@ def test_fetch_changelog_initial_fetch_fails_and_using_scm_then_perform_lookup
end

def test_fetch_changelog_initial_fetch_fails_and_not_using_scm_then_return_nil
skip 'Stubbing uses MiniTest v5 syntax' unless minitest5?
skip_unless_minitest5_or_above

tester.stub :has_scm?, false do
tester.stub :fetch, nil, [:nr_changelog] do
Expand Down