newrelic_rpm 5.7.0.350 → 9.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/.gitignore +15 -1
- data/.rubocop.yml +1919 -0
- data/.rubocop_todo.yml +100 -0
- data/.simplecov +15 -0
- data/.snyk +11 -0
- data/.yardopts +2 -0
- data/Brewfile +12 -0
- data/CHANGELOG.md +4056 -2339
- data/CONTRIBUTING.md +132 -19
- data/DOCKER.md +167 -0
- data/Dockerfile +10 -0
- data/Gemfile +5 -2
- data/Guardfile +22 -4
- data/LICENSE +202 -38
- data/README.md +87 -87
- data/Rakefile +27 -27
- data/THIRD_PARTY_NOTICES.md +28 -0
- data/Thorfile +5 -0
- data/bin/newrelic +3 -2
- data/bin/newrelic_cmd +1 -0
- data/bin/nrdebug +77 -54
- data/config.dot +5 -5
- data/docker-compose.yml +107 -0
- data/init.rb +5 -7
- data/install.rb +3 -3
- data/lefthook.yml +9 -0
- data/lib/new_relic/agent/adaptive_sampler.rb +14 -10
- data/lib/new_relic/agent/agent.rb +125 -969
- data/lib/new_relic/agent/agent_helpers/connect.rb +227 -0
- data/lib/new_relic/agent/agent_helpers/harvest.rb +153 -0
- data/lib/new_relic/agent/agent_helpers/shutdown.rb +72 -0
- data/lib/new_relic/agent/agent_helpers/special_startup.rb +74 -0
- data/lib/new_relic/agent/agent_helpers/start_worker_thread.rb +167 -0
- data/lib/new_relic/agent/agent_helpers/startup.rb +202 -0
- data/lib/new_relic/agent/agent_helpers/transmit.rb +76 -0
- data/lib/new_relic/agent/agent_logger.rb +26 -18
- data/lib/new_relic/agent/attribute_filter.rb +69 -52
- data/lib/new_relic/agent/attribute_processing.rb +8 -8
- data/lib/new_relic/agent/attributes.rb +153 -0
- data/lib/new_relic/agent/audit_logger.rb +19 -4
- data/lib/new_relic/agent/autostart.rb +34 -28
- data/lib/new_relic/agent/chained_call.rb +2 -2
- data/lib/new_relic/agent/commands/agent_command.rb +4 -4
- data/lib/new_relic/agent/commands/agent_command_router.rb +15 -33
- data/lib/new_relic/agent/commands/thread_profiler_session.rb +13 -11
- data/lib/new_relic/agent/configuration/default_source.rb +1480 -1053
- data/lib/new_relic/agent/configuration/dotted_hash.rb +7 -6
- data/lib/new_relic/agent/configuration/environment_source.rb +15 -11
- data/lib/new_relic/agent/configuration/event_harvest_config.rb +68 -0
- data/lib/new_relic/agent/configuration/high_security_source.rb +9 -9
- data/lib/new_relic/agent/configuration/manager.rb +96 -79
- data/lib/new_relic/agent/configuration/manual_source.rb +2 -2
- data/lib/new_relic/agent/configuration/mask_defaults.rb +4 -4
- data/lib/new_relic/agent/configuration/security_policy_source.rb +83 -86
- data/lib/new_relic/agent/configuration/server_source.rb +49 -12
- data/lib/new_relic/agent/configuration/yaml_source.rb +42 -13
- data/lib/new_relic/agent/configuration.rb +2 -2
- data/lib/new_relic/agent/connect/request_builder.rb +61 -0
- data/lib/new_relic/agent/connect/response_handler.rb +58 -0
- data/lib/new_relic/agent/custom_event_aggregator.rb +15 -15
- data/lib/new_relic/agent/database/explain_plan_helpers.rb +5 -6
- data/lib/new_relic/agent/database/obfuscation_helpers.rb +16 -15
- data/lib/new_relic/agent/database/obfuscator.rb +3 -3
- data/lib/new_relic/agent/database/postgres_explain_obfuscator.rb +4 -4
- data/lib/new_relic/agent/database.rb +44 -53
- data/lib/new_relic/agent/database_adapter.rb +35 -0
- data/lib/new_relic/agent/datastores/metric_helper.rb +18 -20
- data/lib/new_relic/agent/datastores/mongo/event_formatter.rb +9 -8
- data/lib/new_relic/agent/datastores/mongo/metric_translator.rb +7 -11
- data/lib/new_relic/agent/datastores/mongo.rb +7 -12
- data/lib/new_relic/agent/datastores/nosql_obfuscator.rb +41 -0
- data/lib/new_relic/agent/datastores/redis.rb +6 -12
- data/lib/new_relic/agent/datastores.rb +19 -23
- data/lib/new_relic/agent/deprecator.rb +2 -2
- data/lib/new_relic/agent/{cross_app_payload.rb → distributed_tracing/cross_app_payload.rb} +13 -12
- data/lib/new_relic/agent/{cross_app_tracing.rb → distributed_tracing/cross_app_tracing.rb} +87 -66
- data/lib/new_relic/agent/distributed_tracing/distributed_trace_attributes.rb +84 -0
- data/lib/new_relic/agent/distributed_tracing/distributed_trace_metrics.rb +75 -0
- data/lib/new_relic/agent/distributed_tracing/distributed_trace_payload.rb +163 -0
- data/lib/new_relic/agent/distributed_tracing/distributed_trace_transport_type.rb +38 -0
- data/lib/new_relic/agent/distributed_tracing/trace_context.rb +245 -0
- data/lib/new_relic/agent/distributed_tracing/trace_context_payload.rb +127 -0
- data/lib/new_relic/agent/distributed_tracing.rb +113 -32
- data/lib/new_relic/agent/encoding_normalizer.rb +5 -3
- data/lib/new_relic/agent/error_collector.rb +99 -63
- data/lib/new_relic/agent/error_event_aggregator.rb +10 -8
- data/lib/new_relic/agent/error_filter.rb +174 -0
- data/lib/new_relic/agent/error_trace_aggregator.rb +6 -4
- data/lib/new_relic/agent/event_aggregator.rb +43 -48
- data/lib/new_relic/agent/event_buffer.rb +8 -9
- data/lib/new_relic/agent/event_listener.rb +2 -3
- data/lib/new_relic/agent/event_loop.rb +27 -25
- data/lib/new_relic/agent/external.rb +20 -51
- data/lib/new_relic/agent/guid_generator.rb +30 -0
- data/lib/new_relic/agent/harvester.rb +5 -6
- data/lib/new_relic/agent/heap.rb +8 -10
- data/lib/new_relic/agent/hostname.rb +26 -5
- data/lib/new_relic/agent/http_clients/abstract.rb +81 -0
- data/lib/new_relic/agent/http_clients/curb_wrappers.rb +26 -26
- data/lib/new_relic/agent/http_clients/excon_wrappers.rb +31 -17
- data/lib/new_relic/agent/http_clients/http_rb_wrappers.rb +18 -23
- data/lib/new_relic/agent/http_clients/httpclient_wrappers.rb +12 -15
- data/lib/new_relic/agent/http_clients/net_http_wrappers.rb +24 -8
- data/lib/new_relic/agent/http_clients/typhoeus_wrappers.rb +9 -12
- data/lib/new_relic/agent/http_clients/uri_util.rb +12 -13
- data/lib/new_relic/agent/instrumentation/action_cable_subscriber.rb +22 -52
- data/lib/new_relic/agent/instrumentation/action_controller_other_subscriber.rb +39 -0
- data/lib/new_relic/agent/instrumentation/action_controller_subscriber.rb +59 -72
- data/lib/new_relic/agent/instrumentation/action_dispatch.rb +31 -0
- data/lib/new_relic/agent/instrumentation/action_dispatch_subscriber.rb +64 -0
- data/lib/new_relic/agent/instrumentation/action_mailbox.rb +30 -0
- data/lib/new_relic/agent/instrumentation/action_mailbox_subscriber.rb +33 -0
- data/lib/new_relic/agent/instrumentation/action_mailer.rb +30 -0
- data/lib/new_relic/agent/instrumentation/action_mailer_subscriber.rb +85 -0
- data/lib/new_relic/agent/instrumentation/action_view_subscriber.rb +86 -62
- data/lib/new_relic/agent/instrumentation/active_job.rb +38 -19
- data/lib/new_relic/agent/instrumentation/active_job_subscriber.rb +41 -0
- data/lib/new_relic/agent/instrumentation/active_merchant.rb +21 -7
- data/lib/new_relic/agent/instrumentation/active_record.rb +95 -46
- data/lib/new_relic/agent/instrumentation/active_record_helper.rb +82 -61
- data/lib/new_relic/agent/instrumentation/active_record_notifications.rb +155 -0
- data/lib/new_relic/agent/instrumentation/active_record_prepend.rb +36 -12
- data/lib/new_relic/agent/instrumentation/active_record_subscriber.rb +69 -64
- data/lib/new_relic/agent/instrumentation/active_storage.rb +8 -4
- data/lib/new_relic/agent/instrumentation/active_storage_subscriber.rb +11 -32
- data/lib/new_relic/agent/instrumentation/active_support.rb +27 -0
- data/lib/new_relic/agent/instrumentation/active_support_logger/chain.rb +23 -0
- data/lib/new_relic/agent/instrumentation/active_support_logger/instrumentation.rb +20 -0
- data/lib/new_relic/agent/instrumentation/active_support_logger/prepend.rb +12 -0
- data/lib/new_relic/agent/instrumentation/active_support_logger.rb +24 -0
- data/lib/new_relic/agent/instrumentation/active_support_subscriber.rb +41 -0
- data/lib/new_relic/agent/instrumentation/bunny/chain.rb +45 -0
- data/lib/new_relic/agent/instrumentation/bunny/instrumentation.rb +150 -0
- data/lib/new_relic/agent/instrumentation/bunny/prepend.rb +35 -0
- data/lib/new_relic/agent/instrumentation/bunny.rb +14 -134
- data/lib/new_relic/agent/instrumentation/concurrent_ruby/chain.rb +36 -0
- data/lib/new_relic/agent/instrumentation/concurrent_ruby/instrumentation.rb +21 -0
- data/lib/new_relic/agent/instrumentation/concurrent_ruby/prepend.rb +27 -0
- data/lib/new_relic/agent/instrumentation/concurrent_ruby.rb +31 -0
- data/lib/new_relic/agent/instrumentation/controller_instrumentation.rb +77 -61
- data/lib/new_relic/agent/instrumentation/curb/chain.rb +91 -0
- data/lib/new_relic/agent/instrumentation/curb/instrumentation.rb +221 -0
- data/lib/new_relic/agent/instrumentation/curb/prepend.rb +61 -0
- data/lib/new_relic/agent/instrumentation/curb.rb +15 -187
- data/lib/new_relic/agent/instrumentation/custom_events.rb +12 -0
- data/lib/new_relic/agent/instrumentation/custom_events_subscriber.rb +37 -0
- data/lib/new_relic/agent/instrumentation/delayed_job/chain.rb +35 -0
- data/lib/new_relic/agent/instrumentation/delayed_job/instrumentation.rb +48 -0
- data/lib/new_relic/agent/instrumentation/delayed_job/prepend.rb +33 -0
- data/lib/new_relic/agent/instrumentation/delayed_job_instrumentation.rb +30 -52
- data/lib/new_relic/agent/instrumentation/elasticsearch/chain.rb +29 -0
- data/lib/new_relic/agent/instrumentation/elasticsearch/instrumentation.rb +66 -0
- data/lib/new_relic/agent/instrumentation/elasticsearch/prepend.rb +13 -0
- data/lib/new_relic/agent/instrumentation/elasticsearch.rb +31 -0
- data/lib/new_relic/agent/instrumentation/excon/middleware.rb +8 -7
- data/lib/new_relic/agent/instrumentation/excon.rb +29 -31
- data/lib/new_relic/agent/instrumentation/fiber/chain.rb +20 -0
- data/lib/new_relic/agent/instrumentation/fiber/instrumentation.rb +24 -0
- data/lib/new_relic/agent/instrumentation/fiber/prepend.rb +18 -0
- data/lib/new_relic/agent/instrumentation/fiber.rb +25 -0
- data/lib/new_relic/agent/instrumentation/grape/chain.rb +24 -0
- data/lib/new_relic/agent/instrumentation/grape/instrumentation.rb +100 -0
- data/lib/new_relic/agent/instrumentation/grape/prepend.rb +17 -0
- data/lib/new_relic/agent/instrumentation/grape.rb +16 -121
- data/lib/new_relic/agent/instrumentation/grpc/client/chain.rb +97 -0
- data/lib/new_relic/agent/instrumentation/grpc/client/instrumentation.rb +89 -0
- data/lib/new_relic/agent/instrumentation/grpc/client/prepend.rb +111 -0
- data/lib/new_relic/agent/instrumentation/grpc/client/request_wrapper.rb +30 -0
- data/lib/new_relic/agent/instrumentation/grpc/helper.rb +32 -0
- data/lib/new_relic/agent/instrumentation/grpc/server/chain.rb +69 -0
- data/lib/new_relic/agent/instrumentation/grpc/server/instrumentation.rb +134 -0
- data/lib/new_relic/agent/instrumentation/grpc/server/rpc_desc_prepend.rb +35 -0
- data/lib/new_relic/agent/instrumentation/grpc/server/rpc_server_prepend.rb +26 -0
- data/lib/new_relic/agent/instrumentation/grpc_client.rb +23 -0
- data/lib/new_relic/agent/instrumentation/grpc_server.rb +25 -0
- data/lib/new_relic/agent/instrumentation/httpclient/chain.rb +24 -0
- data/lib/new_relic/agent/instrumentation/httpclient/instrumentation.rb +37 -0
- data/lib/new_relic/agent/instrumentation/httpclient/prepend.rb +15 -0
- data/lib/new_relic/agent/instrumentation/httpclient.rb +12 -32
- data/lib/new_relic/agent/instrumentation/httprb/chain.rb +22 -0
- data/lib/new_relic/agent/instrumentation/httprb/instrumentation.rb +30 -0
- data/lib/new_relic/agent/instrumentation/httprb/prepend.rb +15 -0
- data/lib/new_relic/agent/instrumentation/httprb.rb +29 -0
- data/lib/new_relic/agent/instrumentation/ignore_actions.rb +5 -6
- data/lib/new_relic/agent/instrumentation/logger/chain.rb +21 -0
- data/lib/new_relic/agent/instrumentation/logger/instrumentation.rb +66 -0
- data/lib/new_relic/agent/instrumentation/logger/prepend.rb +13 -0
- data/lib/new_relic/agent/instrumentation/logger.rb +26 -0
- data/lib/new_relic/agent/instrumentation/memcache/chain.rb +15 -0
- data/lib/new_relic/agent/instrumentation/memcache/dalli.rb +58 -125
- data/lib/new_relic/agent/instrumentation/memcache/helper.rb +59 -0
- data/lib/new_relic/agent/instrumentation/memcache/instrumentation.rb +90 -0
- data/lib/new_relic/agent/instrumentation/memcache/prepend.rb +101 -0
- data/lib/new_relic/agent/instrumentation/memcache.rb +57 -71
- data/lib/new_relic/agent/instrumentation/middleware_proxy.rb +15 -14
- data/lib/new_relic/agent/instrumentation/middleware_tracing.rb +21 -14
- data/lib/new_relic/agent/instrumentation/mongo.rb +7 -132
- data/lib/new_relic/agent/instrumentation/mongodb_command_subscriber.rb +49 -13
- data/lib/new_relic/agent/instrumentation/net_http/chain.rb +24 -0
- data/lib/new_relic/agent/instrumentation/net_http/instrumentation.rb +40 -0
- data/lib/new_relic/agent/instrumentation/net_http/prepend.rb +21 -0
- data/lib/new_relic/agent/instrumentation/net_http.rb +44 -0
- data/lib/new_relic/agent/instrumentation/notifications_subscriber.rb +142 -0
- data/lib/new_relic/agent/instrumentation/padrino/chain.rb +38 -0
- data/lib/new_relic/agent/instrumentation/padrino/instrumentation.rb +28 -0
- data/lib/new_relic/agent/instrumentation/padrino/prepend.rb +20 -0
- data/lib/new_relic/agent/instrumentation/padrino.rb +22 -58
- data/lib/new_relic/agent/instrumentation/passenger_instrumentation.rb +7 -7
- data/lib/new_relic/agent/instrumentation/queue_time.rb +9 -10
- data/lib/new_relic/agent/instrumentation/rack/chain.rb +66 -0
- data/lib/new_relic/agent/instrumentation/rack/helpers.rb +33 -0
- data/lib/new_relic/agent/instrumentation/rack/instrumentation.rb +75 -0
- data/lib/new_relic/agent/instrumentation/rack/prepend.rb +43 -0
- data/lib/new_relic/agent/instrumentation/rack.rb +33 -141
- data/lib/new_relic/agent/instrumentation/rails3/action_controller.rb +19 -55
- data/lib/new_relic/agent/instrumentation/rails_middleware.rb +5 -5
- data/lib/new_relic/agent/instrumentation/rails_notifications/action_cable.rb +36 -0
- data/lib/new_relic/agent/instrumentation/rails_notifications/action_controller.rb +45 -0
- data/lib/new_relic/agent/instrumentation/rails_notifications/action_view.rb +30 -0
- data/lib/new_relic/agent/instrumentation/rails_notifications/custom_events.rb +30 -0
- data/lib/new_relic/agent/instrumentation/rake/chain.rb +20 -0
- data/lib/new_relic/agent/instrumentation/rake/instrumentation.rb +142 -0
- data/lib/new_relic/agent/instrumentation/rake/prepend.rb +14 -0
- data/lib/new_relic/agent/instrumentation/rake.rb +18 -159
- data/lib/new_relic/agent/instrumentation/redis/chain.rb +45 -0
- data/lib/new_relic/agent/instrumentation/redis/constants.rb +17 -0
- data/lib/new_relic/agent/instrumentation/redis/instrumentation.rb +93 -0
- data/lib/new_relic/agent/instrumentation/redis/middleware.rb +16 -0
- data/lib/new_relic/agent/instrumentation/redis/prepend.rb +29 -0
- data/lib/new_relic/agent/instrumentation/redis.rb +20 -103
- data/lib/new_relic/agent/instrumentation/resque/chain.rb +21 -0
- data/lib/new_relic/agent/instrumentation/resque/helper.rb +19 -0
- data/lib/new_relic/agent/instrumentation/resque/instrumentation.rb +34 -0
- data/lib/new_relic/agent/instrumentation/resque/prepend.rb +15 -0
- data/lib/new_relic/agent/instrumentation/resque.rb +33 -41
- data/lib/new_relic/agent/instrumentation/sequel.rb +17 -20
- data/lib/new_relic/agent/instrumentation/sequel_helper.rb +3 -3
- data/lib/new_relic/agent/instrumentation/sidekiq/client.rb +20 -0
- data/lib/new_relic/agent/instrumentation/sidekiq/extensions/delayed_class.rb +30 -0
- data/lib/new_relic/agent/instrumentation/sidekiq/server.rb +37 -0
- data/lib/new_relic/agent/instrumentation/sidekiq.rb +29 -46
- data/lib/new_relic/agent/instrumentation/sinatra/chain.rb +55 -0
- data/lib/new_relic/agent/instrumentation/sinatra/ignorer.rb +31 -37
- data/lib/new_relic/agent/instrumentation/sinatra/instrumentation.rb +125 -0
- data/lib/new_relic/agent/instrumentation/sinatra/prepend.rb +33 -0
- data/lib/new_relic/agent/instrumentation/sinatra/transaction_namer.rb +3 -3
- data/lib/new_relic/agent/instrumentation/sinatra.rb +35 -165
- data/lib/new_relic/agent/instrumentation/thread/chain.rb +24 -0
- data/lib/new_relic/agent/instrumentation/thread/instrumentation.rb +28 -0
- data/lib/new_relic/agent/instrumentation/thread/prepend.rb +22 -0
- data/lib/new_relic/agent/instrumentation/thread.rb +20 -0
- data/lib/new_relic/agent/instrumentation/tilt/chain.rb +24 -0
- data/lib/new_relic/agent/instrumentation/tilt/instrumentation.rb +41 -0
- data/lib/new_relic/agent/instrumentation/tilt/prepend.rb +13 -0
- data/lib/new_relic/agent/instrumentation/tilt.rb +25 -0
- data/lib/new_relic/agent/instrumentation/typhoeus/chain.rb +22 -0
- data/lib/new_relic/agent/instrumentation/typhoeus/instrumentation.rb +80 -0
- data/lib/new_relic/agent/instrumentation/typhoeus/prepend.rb +14 -0
- data/lib/new_relic/agent/instrumentation/typhoeus.rb +14 -76
- data/lib/new_relic/agent/instrumentation.rb +2 -2
- data/lib/new_relic/agent/internal_agent_error.rb +3 -3
- data/lib/new_relic/agent/javascript_instrumentor.rb +51 -45
- data/lib/new_relic/agent/linking_metadata.rb +44 -0
- data/lib/new_relic/agent/local_log_decorator.rb +37 -0
- data/lib/new_relic/agent/log_event_aggregator.rb +235 -0
- data/lib/new_relic/agent/log_once.rb +2 -2
- data/lib/new_relic/agent/log_priority.rb +20 -0
- data/lib/new_relic/agent/logging.rb +142 -0
- data/lib/new_relic/agent/memory_logger.rb +3 -3
- data/lib/new_relic/agent/messaging.rb +81 -164
- data/lib/new_relic/agent/method_tracer.rb +152 -145
- data/lib/new_relic/agent/method_tracer_helpers.rb +90 -13
- data/lib/new_relic/agent/monitors/cross_app_monitor.rb +117 -0
- data/lib/new_relic/agent/monitors/distributed_tracing_monitor.rb +28 -0
- data/lib/new_relic/agent/{inbound_request_monitor.rb → monitors/inbound_request_monitor.rb} +5 -6
- data/lib/new_relic/agent/{synthetics_monitor.rb → monitors/synthetics_monitor.rb} +9 -15
- data/lib/new_relic/agent/monitors.rb +26 -0
- data/lib/new_relic/agent/new_relic_service/encoders.rb +7 -7
- data/lib/new_relic/agent/new_relic_service/json_marshaller.rb +6 -7
- data/lib/new_relic/agent/new_relic_service/marshaller.rb +8 -29
- data/lib/new_relic/agent/new_relic_service/security_policy_settings.rb +5 -5
- data/lib/new_relic/agent/new_relic_service.rb +282 -166
- data/lib/new_relic/agent/noticeable_error.rb +19 -0
- data/lib/new_relic/agent/null_logger.rb +8 -4
- data/lib/new_relic/agent/obfuscator.rb +9 -11
- data/lib/new_relic/agent/parameter_filtering.rb +35 -8
- data/lib/new_relic/agent/payload_metric_mapping.rb +10 -11
- data/lib/new_relic/agent/pipe_channel_manager.rb +28 -18
- data/lib/new_relic/agent/pipe_service.rb +9 -6
- data/lib/new_relic/agent/prepend_supportability.rb +3 -3
- data/lib/new_relic/agent/priority_sampled_buffer.rb +16 -14
- data/lib/new_relic/agent/range_extensions.rb +9 -29
- data/lib/new_relic/agent/rules_engine/replacement_rule.rb +12 -12
- data/lib/new_relic/agent/rules_engine/segment_terms_rule.rb +13 -14
- data/lib/new_relic/agent/rules_engine.rb +6 -5
- data/lib/new_relic/agent/sampler.rb +4 -5
- data/lib/new_relic/agent/sampler_collection.rb +4 -5
- data/lib/new_relic/agent/samplers/cpu_sampler.rb +4 -3
- data/lib/new_relic/agent/samplers/delayed_job_sampler.rb +14 -11
- data/lib/new_relic/agent/samplers/memory_sampler.rb +26 -15
- data/lib/new_relic/agent/samplers/object_sampler.rb +2 -2
- data/lib/new_relic/agent/samplers/vm_sampler.rb +22 -20
- data/lib/new_relic/agent/span_event_aggregator.rb +14 -16
- data/lib/new_relic/agent/span_event_primitive.rb +118 -58
- data/lib/new_relic/agent/sql_sampler.rb +25 -25
- data/lib/new_relic/agent/stats.rb +79 -42
- data/lib/new_relic/agent/stats_engine/gc_profiler.rb +11 -13
- data/lib/new_relic/agent/stats_engine/stats_hash.rb +13 -14
- data/lib/new_relic/agent/stats_engine.rb +11 -11
- data/lib/new_relic/agent/synthetics_event_aggregator.rb +8 -9
- data/lib/new_relic/agent/system_info.rb +100 -66
- data/lib/new_relic/agent/threading/agent_thread.rb +20 -16
- data/lib/new_relic/agent/threading/backtrace_node.rb +13 -14
- data/lib/new_relic/agent/threading/backtrace_service.rb +18 -18
- data/lib/new_relic/agent/threading/thread_profile.rb +31 -45
- data/lib/new_relic/agent/timestamp_sampled_buffer.rb +2 -2
- data/lib/new_relic/agent/tracer.rb +513 -0
- data/lib/new_relic/agent/transaction/abstract_segment.rb +131 -41
- data/lib/new_relic/agent/transaction/datastore_segment.rb +22 -18
- data/lib/new_relic/agent/transaction/distributed_tracer.rb +184 -0
- data/lib/new_relic/agent/transaction/distributed_tracing.rb +72 -163
- data/lib/new_relic/agent/transaction/external_request_segment.rb +66 -63
- data/lib/new_relic/agent/transaction/message_broker_segment.rb +34 -46
- data/lib/new_relic/agent/transaction/request_attributes.rb +36 -36
- data/lib/new_relic/agent/transaction/segment.rb +46 -10
- data/lib/new_relic/agent/transaction/slowest_sample_buffer.rb +2 -4
- data/lib/new_relic/agent/transaction/synthetics_sample_buffer.rb +2 -2
- data/lib/new_relic/agent/transaction/trace.rb +21 -24
- data/lib/new_relic/agent/transaction/trace_builder.rb +11 -12
- data/lib/new_relic/agent/transaction/trace_context.rb +168 -0
- data/lib/new_relic/agent/transaction/trace_node.rb +31 -28
- data/lib/new_relic/agent/transaction/tracing.rb +15 -111
- data/lib/new_relic/agent/transaction/transaction_sample_buffer.rb +6 -6
- data/lib/new_relic/agent/transaction.rb +252 -259
- data/lib/new_relic/agent/transaction_error_primitive.rb +34 -37
- data/lib/new_relic/agent/transaction_event_aggregator.rb +13 -13
- data/lib/new_relic/agent/transaction_event_primitive.rb +44 -56
- data/lib/new_relic/agent/transaction_event_recorder.rb +17 -16
- data/lib/new_relic/agent/transaction_metrics.rb +11 -10
- data/lib/new_relic/agent/transaction_sampler.rb +7 -12
- data/lib/new_relic/agent/transaction_time_aggregator.rb +41 -26
- data/lib/new_relic/agent/utilization/aws.rb +34 -4
- data/lib/new_relic/agent/utilization/azure.rb +4 -4
- data/lib/new_relic/agent/utilization/gcp.rb +8 -8
- data/lib/new_relic/agent/utilization/pcf.rb +6 -5
- data/lib/new_relic/agent/utilization/vendor.rb +44 -29
- data/lib/new_relic/agent/utilization_data.rb +43 -6
- data/lib/new_relic/agent/vm/jruby_vm.rb +2 -2
- data/lib/new_relic/agent/vm/monotonic_gc_profiler.rb +3 -3
- data/lib/new_relic/agent/vm/mri_vm.rb +46 -19
- data/lib/new_relic/agent/vm/snapshot.rb +6 -6
- data/lib/new_relic/agent/vm.rb +2 -2
- data/lib/new_relic/agent/worker_loop.rb +11 -13
- data/lib/new_relic/agent.rb +151 -79
- data/lib/new_relic/cli/command.rb +21 -23
- data/lib/new_relic/cli/commands/deployments.rb +94 -45
- data/lib/new_relic/cli/commands/install.rb +24 -26
- data/lib/new_relic/coerce.rb +42 -15
- data/lib/new_relic/collection_helper.rb +51 -49
- data/lib/new_relic/constants.rb +39 -0
- data/lib/new_relic/control/class_methods.rb +11 -5
- data/lib/new_relic/control/frameworks/external.rb +3 -3
- data/lib/new_relic/control/frameworks/rails.rb +24 -18
- data/lib/new_relic/control/frameworks/rails3.rb +4 -5
- data/lib/new_relic/control/frameworks/rails4.rb +2 -2
- data/lib/new_relic/control/frameworks/rails_notifications.rb +14 -0
- data/lib/new_relic/control/frameworks/ruby.rb +4 -4
- data/lib/new_relic/control/frameworks/sinatra.rb +8 -2
- data/lib/new_relic/control/frameworks.rb +2 -2
- data/lib/new_relic/control/instance_methods.rb +33 -42
- data/lib/new_relic/control/instrumentation.rb +40 -12
- data/lib/new_relic/control/private_instance_methods.rb +48 -0
- data/lib/new_relic/control/server_methods.rb +4 -5
- data/lib/new_relic/control.rb +2 -3
- data/lib/new_relic/delayed_job_injection.rb +2 -2
- data/lib/new_relic/dependency_detection.rb +129 -18
- data/lib/new_relic/environment_report.rb +41 -35
- data/lib/new_relic/helper.rb +49 -8
- data/lib/new_relic/language_support.rb +30 -6
- data/lib/new_relic/latest_changes.rb +9 -8
- data/lib/new_relic/local_environment.rb +23 -27
- data/lib/new_relic/metric_data.rb +32 -27
- data/lib/new_relic/metric_spec.rb +9 -7
- data/lib/new_relic/noticed_error.rb +46 -33
- data/lib/new_relic/rack/agent_hooks.rb +2 -2
- data/lib/new_relic/rack/agent_middleware.rb +7 -5
- data/lib/new_relic/rack/browser_monitoring.rb +134 -117
- data/lib/new_relic/rack.rb +2 -2
- data/lib/new_relic/recipes/capistrano3.rb +4 -62
- data/lib/new_relic/recipes/capistrano_legacy.rb +24 -27
- data/lib/new_relic/recipes/helpers/send_deployment.rb +70 -0
- data/lib/new_relic/recipes.rb +2 -2
- data/lib/new_relic/supportability_helper.rb +21 -7
- data/lib/new_relic/traced_thread.rb +39 -0
- data/lib/new_relic/version.rb +7 -18
- data/lib/newrelic_rpm.rb +20 -33
- data/lib/sequel/extensions/{newrelic_instrumentation.rb → new_relic_instrumentation.rb} +16 -19
- data/lib/sequel/plugins/{newrelic_instrumentation.rb → new_relic_instrumentation.rb} +9 -15
- data/lib/tasks/all.rb +4 -4
- data/lib/tasks/config.rake +22 -118
- data/lib/tasks/coverage_report.rake +28 -0
- data/lib/tasks/helpers/config.html.erb +21 -0
- data/lib/tasks/helpers/format.rb +123 -0
- data/lib/tasks/helpers/matches.rb +12 -0
- data/lib/tasks/helpers/prompt.rb +24 -0
- data/lib/tasks/helpers/removers.rb +33 -0
- data/lib/tasks/install.rake +4 -0
- data/lib/tasks/instrumentation_generator/README.md +63 -0
- data/lib/tasks/instrumentation_generator/TODO.md +33 -0
- data/lib/tasks/instrumentation_generator/instrumentation.thor +121 -0
- data/lib/tasks/instrumentation_generator/templates/Envfile.tt +9 -0
- data/lib/tasks/instrumentation_generator/templates/chain.tt +22 -0
- data/lib/tasks/instrumentation_generator/templates/chain_method.tt +8 -0
- data/lib/tasks/instrumentation_generator/templates/dependency_detection.tt +29 -0
- data/lib/tasks/instrumentation_generator/templates/instrumentation.tt +13 -0
- data/lib/tasks/instrumentation_generator/templates/instrumentation_method.tt +3 -0
- data/lib/tasks/instrumentation_generator/templates/newrelic.yml.tt +19 -0
- data/lib/tasks/instrumentation_generator/templates/prepend.tt +13 -0
- data/lib/tasks/instrumentation_generator/templates/prepend_method.tt +3 -0
- data/lib/tasks/instrumentation_generator/templates/test.tt +15 -0
- data/lib/tasks/multiverse.rake +4 -0
- data/lib/tasks/multiverse.rb +12 -5
- data/lib/tasks/newrelic.rb +2 -2
- data/lib/tasks/tests.rake +14 -14
- data/newrelic.yml +672 -3
- data/newrelic_rpm.gemspec +40 -31
- data/recipes/newrelic.rb +3 -3
- data/test/agent_helper.rb +419 -98
- metadata +238 -127
- data/.travis.yml +0 -228
- data/bin/mongrel_rpm +0 -33
- data/cert/cacert.pem +0 -1177
- data/lib/new_relic/agent/commands/xray_session.rb +0 -55
- data/lib/new_relic/agent/commands/xray_session_collection.rb +0 -161
- data/lib/new_relic/agent/cross_app_monitor.rb +0 -110
- data/lib/new_relic/agent/datastores/mongo/obfuscator.rb +0 -44
- data/lib/new_relic/agent/datastores/mongo/statement_formatter.rb +0 -53
- data/lib/new_relic/agent/distributed_trace_monitor.rb +0 -41
- data/lib/new_relic/agent/distributed_trace_payload.rb +0 -246
- data/lib/new_relic/agent/http_clients/abstract_request.rb +0 -31
- data/lib/new_relic/agent/instrumentation/active_record_4.rb +0 -42
- data/lib/new_relic/agent/instrumentation/active_record_5.rb +0 -41
- data/lib/new_relic/agent/instrumentation/acts_as_solr.rb +0 -74
- data/lib/new_relic/agent/instrumentation/authlogic.rb +0 -25
- data/lib/new_relic/agent/instrumentation/data_mapper.rb +0 -202
- data/lib/new_relic/agent/instrumentation/evented_subscriber.rb +0 -104
- data/lib/new_relic/agent/instrumentation/excon/connection.rb +0 -46
- data/lib/new_relic/agent/instrumentation/http.rb +0 -46
- data/lib/new_relic/agent/instrumentation/merb/controller.rb +0 -44
- data/lib/new_relic/agent/instrumentation/merb/errors.rb +0 -33
- data/lib/new_relic/agent/instrumentation/net.rb +0 -50
- data/lib/new_relic/agent/instrumentation/rails/action_controller.rb +0 -125
- data/lib/new_relic/agent/instrumentation/rails/action_web_service.rb +0 -46
- data/lib/new_relic/agent/instrumentation/rails4/action_controller.rb +0 -32
- data/lib/new_relic/agent/instrumentation/rails4/action_view.rb +0 -27
- data/lib/new_relic/agent/instrumentation/rails5/action_cable.rb +0 -36
- data/lib/new_relic/agent/instrumentation/rails5/action_controller.rb +0 -33
- data/lib/new_relic/agent/instrumentation/rails5/action_view.rb +0 -27
- data/lib/new_relic/agent/instrumentation/rainbows_instrumentation.rb +0 -26
- data/lib/new_relic/agent/instrumentation/sunspot.rb +0 -33
- data/lib/new_relic/agent/supported_versions.rb +0 -275
- data/lib/new_relic/agent/transaction/attributes.rb +0 -154
- data/lib/new_relic/agent/transaction/xray_sample_buffer.rb +0 -64
- data/lib/new_relic/agent/transaction_state.rb +0 -186
- data/lib/new_relic/build.rb +0 -2
- data/lib/new_relic/control/frameworks/merb.rb +0 -29
- data/lib/new_relic/control/frameworks/rails5.rb +0 -14
- data/lib/new_relic/metrics.rb +0 -13
- data/lib/tasks/config.html.erb +0 -32
- data/lib/tasks/versions.html.erb +0 -28
- data/lib/tasks/versions.postface.html +0 -8
- data/lib/tasks/versions.preface.html +0 -9
- data/lib/tasks/versions.rake +0 -65
- data/lib/tasks/versions.txt.erb +0 -14
- /data/lib/tasks/{config.text.erb → helpers/config.text.erb} +0 -0
data/test/agent_helper.rb
CHANGED
@@ -1,26 +1,36 @@
|
|
1
|
-
# encoding: utf-8
|
2
1
|
# This file is distributed under New Relic's license terms.
|
3
|
-
# See https://github.com/newrelic/
|
2
|
+
# See https://github.com/newrelic/newrelic-ruby-agent/blob/main/LICENSE for complete details.
|
3
|
+
# frozen_string_literal: true
|
4
4
|
|
5
5
|
# These helpers should not have any gem dependencies except on newrelic_rpm
|
6
6
|
# itself, and should be usable from within any multiverse suite.
|
7
7
|
|
8
8
|
require 'json'
|
9
|
+
require 'net/http'
|
10
|
+
begin
|
11
|
+
require 'net/http/status'
|
12
|
+
rescue LoadError
|
13
|
+
# NOP -- Net::HTTP::STATUS_CODES was introduced in Ruby 2.5
|
14
|
+
end
|
9
15
|
|
10
16
|
class ArrayLogDevice
|
11
|
-
def initialize(
|
17
|
+
def initialize(array = [])
|
12
18
|
@array = array
|
13
19
|
end
|
14
20
|
attr_reader :array
|
15
21
|
|
16
|
-
def write(
|
22
|
+
def write(message)
|
17
23
|
@array << message
|
18
24
|
end
|
19
25
|
|
20
26
|
def close; end
|
21
27
|
end
|
22
28
|
|
23
|
-
def
|
29
|
+
def fake_guid(length = 16)
|
30
|
+
NewRelic::Agent::GuidGenerator.generate_guid(length)
|
31
|
+
end
|
32
|
+
|
33
|
+
def assert_between(floor, ceiling, value, message = "expected #{floor} <= #{value} <= #{ceiling}")
|
24
34
|
assert((floor <= value && value <= ceiling), message)
|
25
35
|
end
|
26
36
|
|
@@ -38,8 +48,8 @@ end
|
|
38
48
|
|
39
49
|
def assert_has_traced_error(error_class)
|
40
50
|
errors = harvest_error_traces!
|
41
|
-
|
42
|
-
|
51
|
+
|
52
|
+
refute_nil errors.find { |e| e.exception_class_name == error_class.name }, \
|
43
53
|
"Didn't find error of class #{error_class}"
|
44
54
|
end
|
45
55
|
|
@@ -55,6 +65,14 @@ def last_transaction_event
|
|
55
65
|
harvest_transaction_events!.last.last
|
56
66
|
end
|
57
67
|
|
68
|
+
def harvest_span_events!
|
69
|
+
NewRelic::Agent.instance.span_event_aggregator.harvest!
|
70
|
+
end
|
71
|
+
|
72
|
+
def last_span_event
|
73
|
+
harvest_span_events!.last.last
|
74
|
+
end
|
75
|
+
|
58
76
|
def harvest_error_events!
|
59
77
|
NewRelic::Agent.instance.error_collector.error_event_aggregator.harvest!
|
60
78
|
end
|
@@ -63,29 +81,25 @@ def last_error_event
|
|
63
81
|
harvest_error_events!.last.last
|
64
82
|
end
|
65
83
|
|
66
|
-
unless defined?
|
67
|
-
def
|
68
|
-
assert yield, *msgs
|
69
|
-
end
|
70
|
-
end
|
71
|
-
|
72
|
-
unless defined?( assert_includes )
|
73
|
-
def assert_includes( collection, member, msg=nil )
|
84
|
+
unless defined? assert_includes
|
85
|
+
def assert_includes(collection, member, msg = nil)
|
74
86
|
msg = "Expected #{collection.inspect} to include #{member.inspect}"
|
75
|
-
|
87
|
+
|
88
|
+
assert_includes collection, member, msg
|
76
89
|
end
|
77
90
|
end
|
78
91
|
|
79
|
-
unless defined?
|
80
|
-
def assert_not_includes(
|
92
|
+
unless defined? assert_not_includes
|
93
|
+
def assert_not_includes(collection, member, msg = nil)
|
81
94
|
msg = "Expected #{collection.inspect} not to include #{member.inspect}"
|
82
|
-
|
95
|
+
|
96
|
+
refute_includes collection, member, msg
|
83
97
|
end
|
84
98
|
end
|
85
99
|
|
86
|
-
unless defined?
|
87
|
-
def assert_empty(collection, msg=nil)
|
88
|
-
|
100
|
+
unless defined? assert_empty
|
101
|
+
def assert_empty(collection, msg = nil)
|
102
|
+
assert_empty collection, msg
|
89
103
|
end
|
90
104
|
end
|
91
105
|
|
@@ -94,6 +108,12 @@ def assert_equal_unordered(left, right)
|
|
94
108
|
left.each { |element| assert_includes(right, element) }
|
95
109
|
end
|
96
110
|
|
111
|
+
def assert_log_contains(log, message)
|
112
|
+
lines = log.array
|
113
|
+
|
114
|
+
assert (lines.any? { |line| line.match(message) })
|
115
|
+
end
|
116
|
+
|
97
117
|
def assert_audit_log_contains(audit_log_contents, needle)
|
98
118
|
# Original request bodies dumped to the log have symbol keys, but once
|
99
119
|
# they go through a dump/load, they're strings again, so we strip
|
@@ -101,7 +121,8 @@ def assert_audit_log_contains(audit_log_contents, needle)
|
|
101
121
|
regex = /[:"]/
|
102
122
|
needle = needle.gsub(regex, '')
|
103
123
|
haystack = audit_log_contents.gsub(regex, '')
|
104
|
-
|
124
|
+
|
125
|
+
assert_includes(haystack, needle, "Expected log to contain '#{needle}'")
|
105
126
|
end
|
106
127
|
|
107
128
|
# Because we don't generate a strictly machine-readable representation of
|
@@ -115,15 +136,17 @@ end
|
|
115
136
|
def assert_audit_log_contains_object(audit_log_contents, o, format = :json)
|
116
137
|
case o
|
117
138
|
when Hash
|
118
|
-
o.each do |k,v|
|
139
|
+
o.each do |k, v|
|
119
140
|
assert_audit_log_contains_object(audit_log_contents, v, format)
|
120
141
|
assert_audit_log_contains_object(audit_log_contents, k, format)
|
121
142
|
end
|
122
143
|
when Array
|
144
|
+
|
123
145
|
o.each do |el|
|
124
146
|
assert_audit_log_contains_object(audit_log_contents, el, format)
|
125
147
|
end
|
126
148
|
when NilClass
|
149
|
+
|
127
150
|
assert_audit_log_contains(audit_log_contents, format == :json ? "null" : "nil")
|
128
151
|
else
|
129
152
|
assert_audit_log_contains(audit_log_contents, o.inspect)
|
@@ -131,14 +154,15 @@ def assert_audit_log_contains_object(audit_log_contents, o, format = :json)
|
|
131
154
|
end
|
132
155
|
|
133
156
|
def compare_metrics(expected, actual)
|
134
|
-
actual.delete_if {|a| a.include?('GC/Transaction/') }
|
157
|
+
actual.delete_if { |a| a.include?('GC/Transaction/') }
|
158
|
+
|
135
159
|
assert_equal(expected.to_a.sort, actual.to_a.sort, "extra: #{(actual - expected).to_a.inspect}; missing: #{(expected - actual).to_a.inspect}")
|
136
160
|
end
|
137
161
|
|
138
162
|
def metric_spec_from_specish(specish)
|
139
163
|
spec = case specish
|
140
164
|
when String then NewRelic::MetricSpec.new(specish)
|
141
|
-
when Array
|
165
|
+
when Array then NewRelic::MetricSpec.new(*specish)
|
142
166
|
end
|
143
167
|
spec
|
144
168
|
end
|
@@ -148,17 +172,17 @@ def _normalize_metric_expectations(expectations)
|
|
148
172
|
when Array
|
149
173
|
hash = {}
|
150
174
|
# Just assert that the metric is present, nothing about the attributes
|
151
|
-
expectations.each { |k| hash[k] = {
|
175
|
+
expectations.each { |k| hash[k] = {} }
|
152
176
|
hash
|
153
177
|
when String
|
154
|
-
{
|
178
|
+
{expectations => {}}
|
155
179
|
else
|
156
180
|
expectations
|
157
181
|
end
|
158
182
|
end
|
159
183
|
|
160
184
|
def dump_stats(stats)
|
161
|
-
str =
|
185
|
+
str = String.new(" Call count: #{stats.call_count}\n")
|
162
186
|
str << " Total call time: #{stats.total_call_time}\n"
|
163
187
|
str << " Total exclusive time: #{stats.total_exclusive_time}\n"
|
164
188
|
str << " Min call time: #{stats.min_call_time}\n"
|
@@ -173,16 +197,31 @@ end
|
|
173
197
|
def assert_stats_has_values(stats, expected_spec, expected_attrs)
|
174
198
|
expected_attrs.each do |attr, expected_value|
|
175
199
|
actual_value = stats.send(attr)
|
200
|
+
|
201
|
+
msg = "Expected #{attr} for #{expected_spec} to be #{'~' unless attr == :call_count}#{expected_value}, " \
|
202
|
+
"got #{actual_value}.\nActual stats:\n#{dump_stats(stats)}"
|
203
|
+
|
176
204
|
if attr == :call_count
|
177
|
-
|
178
|
-
"Expected #{attr} for #{expected_spec} to be #{expected_value}, got #{actual_value}.\nActual stats:\n#{dump_stats(stats)}")
|
205
|
+
assert_stats_has_values_with_call_count(expected_value, actual_value, msg)
|
179
206
|
else
|
180
|
-
assert_in_delta(expected_value, actual_value, 0.0001,
|
181
|
-
"Expected #{attr} for #{expected_spec} to be ~#{expected_value}, got #{actual_value}.\nActual stats:\n#{dump_stats(stats)}")
|
207
|
+
assert_in_delta(expected_value, actual_value, 0.0001, msg)
|
182
208
|
end
|
183
209
|
end
|
184
210
|
end
|
185
211
|
|
212
|
+
def assert_stats_has_values_with_call_count(expected_value, actual_value, msg)
|
213
|
+
# >, <, >=, <= comparisons
|
214
|
+
if expected_value.to_s =~ /([<>]=?)\s*(\d+)/
|
215
|
+
operator = Regexp.last_match(1).to_sym
|
216
|
+
count = Regexp.last_match(2).to_i
|
217
|
+
|
218
|
+
assert_operator(actual_value, operator, count, msg)
|
219
|
+
# == comparison
|
220
|
+
else
|
221
|
+
assert_equal(expected_value, actual_value, msg)
|
222
|
+
end
|
223
|
+
end
|
224
|
+
|
186
225
|
def assert_metrics_recorded(expected)
|
187
226
|
expected = _normalize_metric_expectations(expected)
|
188
227
|
expected.each do |specish, expected_attrs|
|
@@ -199,6 +238,7 @@ def assert_metrics_recorded(expected)
|
|
199
238
|
|
200
239
|
assert(actual_stats, msg)
|
201
240
|
end
|
241
|
+
|
202
242
|
assert_stats_has_values(actual_stats, expected_spec, expected_attrs)
|
203
243
|
end
|
204
244
|
end
|
@@ -212,10 +252,11 @@ end
|
|
212
252
|
#
|
213
253
|
# If you want to *allow* unexpected metrics matching certain patterns, use
|
214
254
|
# the :ignore_filter option. This will allow you to specify a Regex that
|
215
|
-
#
|
255
|
+
# allowlists broad swathes of metric territory (e.g. 'Supportability/').
|
216
256
|
#
|
217
|
-
def assert_metrics_recorded_exclusive(expected, options={})
|
257
|
+
def assert_metrics_recorded_exclusive(expected, options = {})
|
218
258
|
expected = _normalize_metric_expectations(expected)
|
259
|
+
|
219
260
|
assert_metrics_recorded(expected)
|
220
261
|
|
221
262
|
recorded_metrics = NewRelic::Agent.instance.stats_engine.to_h.keys
|
@@ -227,14 +268,23 @@ def assert_metrics_recorded_exclusive(expected, options={})
|
|
227
268
|
recorded_metrics.reject! { |m| m.name.match(options[:ignore_filter]) }
|
228
269
|
end
|
229
270
|
|
230
|
-
expected_metrics
|
271
|
+
expected_metrics = expected.keys.map { |s| metric_spec_from_specish(s) }
|
231
272
|
|
232
273
|
unexpected_metrics = recorded_metrics - expected_metrics
|
233
|
-
unexpected_metrics.reject! { |m| m.name
|
274
|
+
unexpected_metrics.reject! { |m| m.name.include?('GC/Transaction') }
|
234
275
|
|
235
276
|
assert_equal(0, unexpected_metrics.size, "Found unexpected metrics: #{format_metric_spec_list(unexpected_metrics)}")
|
236
277
|
end
|
237
278
|
|
279
|
+
def assert_newrelic_metadata_present(metadata)
|
280
|
+
assert metadata.key?('newrelic')
|
281
|
+
refute_nil metadata['newrelic']
|
282
|
+
end
|
283
|
+
|
284
|
+
def assert_distributed_tracing_payload_created_for_transaction(transaction)
|
285
|
+
assert transaction.distributed_tracer.instance_variable_get(:@distributed_trace_payload_created)
|
286
|
+
end
|
287
|
+
|
238
288
|
# The clear_metrics! method prevents metrics from "leaking" between tests by resetting
|
239
289
|
# the @stats_hash instance variable in the current instance of NewRelic::Agent::StatsEngine.
|
240
290
|
|
@@ -261,7 +311,8 @@ def assert_metrics_not_recorded(not_expected)
|
|
261
311
|
found_but_not_expected << spec
|
262
312
|
end
|
263
313
|
end
|
264
|
-
|
314
|
+
|
315
|
+
assert_empty(found_but_not_expected, "Found unexpected metrics: #{format_metric_spec_list(found_but_not_expected)}")
|
265
316
|
end
|
266
317
|
|
267
318
|
alias :refute_metrics_recorded :assert_metrics_not_recorded
|
@@ -269,13 +320,12 @@ alias :refute_metrics_recorded :assert_metrics_not_recorded
|
|
269
320
|
def assert_no_metrics_match(regex)
|
270
321
|
matching_metrics = []
|
271
322
|
NewRelic::Agent.instance.stats_engine.to_h.keys.map(&:to_s).each do |metric|
|
272
|
-
matching_metrics << metric if metric.match
|
323
|
+
matching_metrics << metric if metric.match(regex)
|
273
324
|
end
|
274
325
|
|
275
|
-
|
276
|
-
[],
|
326
|
+
assert_empty(
|
277
327
|
matching_metrics,
|
278
|
-
"Found unexpected metrics:\n" +
|
328
|
+
"Found unexpected metrics:\n" + matching_metrics.map { |m| " '#{m}'" }.join("\n") + "\n\n"
|
279
329
|
)
|
280
330
|
end
|
281
331
|
|
@@ -290,21 +340,23 @@ end
|
|
290
340
|
|
291
341
|
def assert_truthy(expected, msg = nil)
|
292
342
|
msg ||= "Expected #{expected.inspect} to be truthy"
|
293
|
-
|
343
|
+
|
344
|
+
refute !expected, msg
|
294
345
|
end
|
295
346
|
|
296
347
|
def assert_falsy(expected, msg = nil)
|
297
348
|
msg ||= "Expected #{expected.inspect} to be falsy"
|
298
|
-
|
349
|
+
|
350
|
+
refute expected, msg
|
299
351
|
end
|
300
352
|
|
301
|
-
unless defined?
|
353
|
+
unless defined? assert_false
|
302
354
|
def assert_false(expected)
|
303
|
-
|
355
|
+
refute expected
|
304
356
|
end
|
305
357
|
end
|
306
358
|
|
307
|
-
unless defined?
|
359
|
+
unless defined? refute
|
308
360
|
alias refute assert_false
|
309
361
|
end
|
310
362
|
|
@@ -327,39 +379,85 @@ end
|
|
327
379
|
# in_transaction('foobar', :category => :controller) { ... }
|
328
380
|
#
|
329
381
|
def in_transaction(*args, &blk)
|
330
|
-
opts
|
382
|
+
opts = args.last && args.last.is_a?(Hash) ? args.pop : {}
|
331
383
|
category = (opts && opts.delete(:category)) || :other
|
332
384
|
|
333
385
|
# At least one test passes `:transaction_name => nil`, so handle it gently
|
334
|
-
name = opts.key?(:transaction_name) ? opts.delete(:transaction_name) :
|
335
|
-
args.first || 'dummy'
|
386
|
+
name = opts.key?(:transaction_name) ? opts.delete(:transaction_name) : args.first || 'dummy'
|
336
387
|
|
337
|
-
state = NewRelic::Agent::
|
388
|
+
state = NewRelic::Agent::Tracer.state
|
338
389
|
txn = nil
|
339
390
|
|
340
|
-
NewRelic::Agent::
|
391
|
+
NewRelic::Agent::Tracer.in_transaction(name: name, category: category, options: opts) do
|
341
392
|
txn = state.current_transaction
|
342
|
-
yield
|
393
|
+
yield(state.current_transaction)
|
343
394
|
end
|
344
395
|
|
345
396
|
txn
|
346
397
|
end
|
347
398
|
|
399
|
+
# Temporarily disables default transformer so tests with invalid inputs can be tried
|
400
|
+
def with_disabled_defaults_transformer(key)
|
401
|
+
begin
|
402
|
+
transformer = NewRelic::Agent::Configuration::DEFAULTS[key][:transform]
|
403
|
+
NewRelic::Agent::Configuration::DEFAULTS[key][:transform] = nil
|
404
|
+
yield
|
405
|
+
ensure
|
406
|
+
NewRelic::Agent::Configuration::DEFAULTS[key][:transform] = transformer
|
407
|
+
end
|
408
|
+
end
|
409
|
+
|
410
|
+
# Convenience wrapper to stand up a transaction and provide a segment within
|
411
|
+
# that transaction to work with. The same arguments as provided to in_transaction
|
412
|
+
# may be supplied.
|
413
|
+
def with_segment(*args, &blk)
|
414
|
+
segment = nil
|
415
|
+
txn = in_transaction(*args) do |t|
|
416
|
+
segment = t.current_segment
|
417
|
+
yield(segment, t)
|
418
|
+
end
|
419
|
+
[segment, txn]
|
420
|
+
end
|
421
|
+
|
422
|
+
# building error attributes on segments are deferred until it's time
|
423
|
+
# to publish/harvest them as spans, so for testing, we'll explicitly
|
424
|
+
# build 'em as appropriate so we can test 'em
|
425
|
+
def build_deferred_error_attributes(segment)
|
426
|
+
return unless segment.noticed_error
|
427
|
+
return if segment.noticed_error_attributes.frozen?
|
428
|
+
|
429
|
+
segment.noticed_error.build_error_attributes
|
430
|
+
end
|
431
|
+
|
432
|
+
def capture_segment_with_error
|
433
|
+
begin
|
434
|
+
segment_with_error = nil
|
435
|
+
with_segment do |segment|
|
436
|
+
segment_with_error = segment
|
437
|
+
raise "oops!"
|
438
|
+
end
|
439
|
+
rescue Exception => exception
|
440
|
+
assert segment_with_error, "expected to have a segment_with_error"
|
441
|
+
build_deferred_error_attributes(segment_with_error)
|
442
|
+
return segment_with_error, exception
|
443
|
+
end
|
444
|
+
end
|
445
|
+
|
348
446
|
def stub_transaction_guid(guid)
|
349
447
|
NewRelic::Agent::Transaction.tl_current.instance_variable_set(:@guid, guid)
|
350
448
|
end
|
351
449
|
|
352
450
|
# Convenience wrapper around in_transaction that sets the category so that it
|
353
451
|
# looks like we are in a web transaction
|
354
|
-
def in_web_transaction(name='dummy')
|
452
|
+
def in_web_transaction(name = 'dummy')
|
355
453
|
in_transaction(name, :category => :controller, :request => stub(:path => '/')) do |txn|
|
356
|
-
yield
|
454
|
+
yield(txn)
|
357
455
|
end
|
358
456
|
end
|
359
457
|
|
360
|
-
def in_background_transaction(name='silly')
|
458
|
+
def in_background_transaction(name = 'silly')
|
361
459
|
in_transaction(name, :category => :task) do |txn|
|
362
|
-
yield
|
460
|
+
yield(txn)
|
363
461
|
end
|
364
462
|
end
|
365
463
|
|
@@ -371,6 +469,7 @@ end
|
|
371
469
|
|
372
470
|
def last_transaction_trace
|
373
471
|
return unless last_sample = NewRelic::Agent.agent.transaction_sampler.last_sample
|
472
|
+
|
374
473
|
NewRelic::Agent::Transaction::TraceBuilder.build_trace(last_sample)
|
375
474
|
end
|
376
475
|
|
@@ -392,7 +491,7 @@ def last_sql_trace
|
|
392
491
|
NewRelic::Agent.agent.sql_sampler.sql_traces.values.last
|
393
492
|
end
|
394
493
|
|
395
|
-
def find_last_transaction_node(transaction_sample=nil)
|
494
|
+
def find_last_transaction_node(transaction_sample = nil)
|
396
495
|
if transaction_sample
|
397
496
|
root_node = transaction_sample.root_node
|
398
497
|
else
|
@@ -400,7 +499,7 @@ def find_last_transaction_node(transaction_sample=nil)
|
|
400
499
|
end
|
401
500
|
|
402
501
|
last_node = nil
|
403
|
-
root_node.each_node {|s| last_node = s }
|
502
|
+
root_node.each_node { |s| last_node = s }
|
404
503
|
|
405
504
|
return last_node
|
406
505
|
end
|
@@ -417,7 +516,7 @@ end
|
|
417
516
|
|
418
517
|
def find_node_with_name_matching(transaction_sample, regex)
|
419
518
|
transaction_sample.root_node.each_node do |node|
|
420
|
-
if node.metric_name.match
|
519
|
+
if node.metric_name.match(regex)
|
421
520
|
return node
|
422
521
|
end
|
423
522
|
end
|
@@ -431,7 +530,7 @@ def find_all_nodes_with_name_matching(transaction_sample, regexes)
|
|
431
530
|
|
432
531
|
transaction_sample.root_node.each_node do |node|
|
433
532
|
regexes.each do |regex|
|
434
|
-
if node.metric_name.match
|
533
|
+
if node.metric_name.match(regex)
|
435
534
|
matching_nodes << node
|
436
535
|
end
|
437
536
|
end
|
@@ -440,7 +539,7 @@ def find_all_nodes_with_name_matching(transaction_sample, regexes)
|
|
440
539
|
matching_nodes
|
441
540
|
end
|
442
541
|
|
443
|
-
def with_config(config_hash, at_start=true)
|
542
|
+
def with_config(config_hash, at_start = true)
|
444
543
|
config = NewRelic::Agent::Configuration::DottedHash.new(config_hash, true)
|
445
544
|
NewRelic::Agent.config.add_config_for_testing(config, at_start)
|
446
545
|
NewRelic::Agent.instance.refresh_attribute_filter
|
@@ -452,6 +551,13 @@ def with_config(config_hash, at_start=true)
|
|
452
551
|
end
|
453
552
|
end
|
454
553
|
|
554
|
+
def with_server_source(config_hash, at_start = true)
|
555
|
+
with_config(config_hash, at_start) do
|
556
|
+
NewRelic::Agent.config.notify_server_source_added
|
557
|
+
yield
|
558
|
+
end
|
559
|
+
end
|
560
|
+
|
455
561
|
def with_config_low_priority(config_hash)
|
456
562
|
with_config(config_hash, false) do
|
457
563
|
yield
|
@@ -484,7 +590,7 @@ unless Time.respond_to?(:__original_now)
|
|
484
590
|
end
|
485
591
|
end
|
486
592
|
|
487
|
-
def nr_freeze_time(now=Time.now)
|
593
|
+
def nr_freeze_time(now = Time.now)
|
488
594
|
Time.__frozen_now = now
|
489
595
|
end
|
490
596
|
|
@@ -496,7 +602,32 @@ def advance_time(seconds)
|
|
496
602
|
Time.__frozen_now = Time.now + seconds
|
497
603
|
end
|
498
604
|
|
499
|
-
|
605
|
+
unless Process.respond_to?(:__original_clock_gettime)
|
606
|
+
Process.instance_eval do
|
607
|
+
class << self
|
608
|
+
attr_accessor :__frozen_clock_gettime
|
609
|
+
alias_method :__original_clock_gettime, :clock_gettime
|
610
|
+
|
611
|
+
def clock_gettime(clock_id, unit = :float_second)
|
612
|
+
__frozen_clock_gettime || __original_clock_gettime(clock_id, unit)
|
613
|
+
end
|
614
|
+
end
|
615
|
+
end
|
616
|
+
end
|
617
|
+
|
618
|
+
def advance_process_time(seconds, clock_id = Process::CLOCK_REALTIME)
|
619
|
+
Process.__frozen_clock_gettime = Process.clock_gettime(clock_id) + seconds
|
620
|
+
end
|
621
|
+
|
622
|
+
def nr_freeze_process_time(now = Process.clock_gettime(Process::CLOCK_REALTIME))
|
623
|
+
Process.__frozen_clock_gettime = now
|
624
|
+
end
|
625
|
+
|
626
|
+
def nr_unfreeze_process_time
|
627
|
+
Process.__frozen_clock_gettime = nil
|
628
|
+
end
|
629
|
+
|
630
|
+
def with_constant_defined(constant_symbol, implementation = Module.new)
|
500
631
|
const_path = constant_path(constant_symbol.to_s)
|
501
632
|
|
502
633
|
if const_path
|
@@ -516,24 +647,32 @@ def with_constant_defined(constant_symbol, implementation=Module.new)
|
|
516
647
|
end
|
517
648
|
end
|
518
649
|
|
519
|
-
def constant_path(name, opts={})
|
650
|
+
def constant_path(name, opts = {})
|
520
651
|
allow_partial = opts[:allow_partial]
|
521
652
|
path = [Object]
|
522
653
|
parts = name.gsub(/^::/, '').split('::')
|
523
654
|
parts.each do |part|
|
524
|
-
if !path.last.
|
655
|
+
if !path.last.constants.include?(part.to_sym)
|
525
656
|
return allow_partial ? path : nil
|
526
657
|
end
|
658
|
+
|
527
659
|
path << path.last.const_get(part)
|
528
660
|
end
|
529
661
|
path
|
530
662
|
end
|
531
663
|
|
664
|
+
def get_parent(constant_name)
|
665
|
+
parent_name = constant_name.gsub(/::[^:]*$/, '')
|
666
|
+
const_path = constant_path(parent_name)
|
667
|
+
const_path ? const_path[-1] : nil
|
668
|
+
end
|
669
|
+
|
532
670
|
def undefine_constant(constant_symbol)
|
533
|
-
|
534
|
-
|
535
|
-
|
536
|
-
|
671
|
+
const_str = constant_symbol.to_s
|
672
|
+
parent = get_parent(const_str)
|
673
|
+
const_name = const_str.gsub(/.*::/, '')
|
674
|
+
return yield unless parent && parent.constants.include?(const_name.to_sym)
|
675
|
+
|
537
676
|
removed_constant = parent.send(:remove_const, const_name)
|
538
677
|
yield
|
539
678
|
ensure
|
@@ -544,7 +683,7 @@ def with_debug_logging
|
|
544
683
|
orig_logger = NewRelic::Agent.logger
|
545
684
|
$stderr.puts '', '---', ''
|
546
685
|
NewRelic::Agent.logger =
|
547
|
-
NewRelic::Agent::AgentLogger.new('', Logger.new($stderr)
|
686
|
+
NewRelic::Agent::AgentLogger.new('', Logger.new($stderr))
|
548
687
|
|
549
688
|
with_config(:log_level => 'debug') do
|
550
689
|
yield
|
@@ -554,17 +693,17 @@ ensure
|
|
554
693
|
end
|
555
694
|
|
556
695
|
def create_agent_command(args = {})
|
557
|
-
NewRelic::Agent::Commands::AgentCommand.new([-1, {
|
696
|
+
NewRelic::Agent::Commands::AgentCommand.new([-1, {"name" => "command_name", "arguments" => args}])
|
558
697
|
end
|
559
698
|
|
560
|
-
def wait_for_backtrace_service_poll(opts={})
|
699
|
+
def wait_for_backtrace_service_poll(opts = {})
|
561
700
|
defaults = {
|
562
701
|
:timeout => 10.0,
|
563
702
|
:service => NewRelic::Agent.agent.agent_command_router.backtrace_service,
|
564
703
|
:iterations => 1
|
565
704
|
}
|
566
705
|
opts = defaults.merge(opts)
|
567
|
-
deadline =
|
706
|
+
deadline = Process.clock_gettime(Process::CLOCK_REALTIME) + opts[:timeout]
|
568
707
|
|
569
708
|
service = opts[:service]
|
570
709
|
worker_loop = service.worker_loop
|
@@ -572,20 +711,20 @@ def wait_for_backtrace_service_poll(opts={})
|
|
572
711
|
|
573
712
|
until worker_loop.iterations > opts[:iterations]
|
574
713
|
sleep(0.01)
|
575
|
-
if
|
714
|
+
if Process.clock_gettime(Process::CLOCK_REALTIME) > deadline
|
576
715
|
raise "Timed out waiting #{opts[:timeout]} s for backtrace service poll\n" +
|
577
|
-
|
578
|
-
|
579
|
-
|
580
|
-
|
581
|
-
|
716
|
+
"Worker loop ran for #{opts[:service].worker_loop.iterations} iterations\n\n" +
|
717
|
+
Thread.list.map { |t|
|
718
|
+
"#{t.to_s}: newrelic_label: #{t[:newrelic_label].inspect}\n\n" +
|
719
|
+
(t.backtrace || []).join("\n\t")
|
720
|
+
}.join("\n\n")
|
582
721
|
end
|
583
722
|
end
|
584
723
|
end
|
585
724
|
|
586
|
-
def with_array_logger(level
|
725
|
+
def with_array_logger(level = :info)
|
587
726
|
orig_logger = NewRelic::Agent.logger
|
588
|
-
config = {
|
727
|
+
config = {:log_level => level}
|
589
728
|
logdev = ArrayLogDevice.new
|
590
729
|
override_logger = Logger.new(logdev)
|
591
730
|
|
@@ -599,17 +738,87 @@ ensure
|
|
599
738
|
NewRelic::Agent.logger = orig_logger
|
600
739
|
end
|
601
740
|
|
602
|
-
|
603
|
-
|
604
|
-
|
605
|
-
|
606
|
-
|
741
|
+
# The EnvUpdater was introduced due to random fails in JRuby environment
|
742
|
+
# whereby attempting to set ENV[key] = some_value randomly failed.
|
743
|
+
# It is conjectured that this is thread related, but may also be
|
744
|
+
# a core bug in the JVM implementation of Ruby. Root cause was not
|
745
|
+
# discovered, but it was found that a combination of retrying and using
|
746
|
+
# mutex lock around the update operation was the only consistently working
|
747
|
+
# solution as the error continued to surface without the mutex and
|
748
|
+
# retry alone wasn't enough, either.
|
749
|
+
#
|
750
|
+
# JRUBY: oraclejdk8 + jruby-9.2.6.0
|
751
|
+
#
|
752
|
+
# NOTE: Singleton pattern to ensure one mutex lock for all threads
|
753
|
+
class EnvUpdater
|
754
|
+
MAX_RETRIES = 5
|
755
|
+
|
756
|
+
def initialize
|
757
|
+
@mutex = Mutex.new
|
607
758
|
end
|
608
|
-
|
759
|
+
|
760
|
+
# Will attempt the given block up to MAX_RETRIES before
|
761
|
+
# surfacing the exception down the chain.
|
762
|
+
def with_retry(retry_limit = MAX_RETRIES)
|
763
|
+
retries ||= 0
|
764
|
+
sleep(retries)
|
609
765
|
yield
|
610
|
-
|
611
|
-
|
766
|
+
rescue
|
767
|
+
(retries += 1) < retry_limit ? retry : raise
|
612
768
|
end
|
769
|
+
|
770
|
+
# Locks and updates the ENV
|
771
|
+
def safe_update(env)
|
772
|
+
with_retry do
|
773
|
+
@mutex.synchronize do
|
774
|
+
env.each { |key, val| ENV[key] = val.to_s }
|
775
|
+
end
|
776
|
+
end
|
777
|
+
end
|
778
|
+
|
779
|
+
# Locks and restores the ENV
|
780
|
+
def safe_restore(old_env)
|
781
|
+
with_retry do
|
782
|
+
@mutex.synchronize do
|
783
|
+
old_env.each { |key, val| val ? ENV[key] = val : ENV.delete(key) }
|
784
|
+
end
|
785
|
+
end
|
786
|
+
end
|
787
|
+
|
788
|
+
# Singleton pattern implemented via @@instance
|
789
|
+
def self.instance
|
790
|
+
@@instance ||= EnvUpdater.new
|
791
|
+
end
|
792
|
+
|
793
|
+
def self.safe_update(env)
|
794
|
+
instance.safe_update(env)
|
795
|
+
end
|
796
|
+
|
797
|
+
def self.safe_restore(old_env)
|
798
|
+
instance.safe_restore(old_env)
|
799
|
+
end
|
800
|
+
|
801
|
+
# Effectively saves current ENV settings for given env's key/values,
|
802
|
+
# runs given block, then restores ENV to original state before returning.
|
803
|
+
def self.inject(env, &block)
|
804
|
+
old_env = {}
|
805
|
+
env.each { |key, val| old_env[key] = ENV[key] }
|
806
|
+
begin
|
807
|
+
safe_update(env)
|
808
|
+
yield
|
809
|
+
ensure
|
810
|
+
safe_restore(old_env)
|
811
|
+
end
|
812
|
+
end
|
813
|
+
|
814
|
+
# must call instance here to ensure only one @mutex for all threads.
|
815
|
+
instance
|
816
|
+
end
|
817
|
+
|
818
|
+
# Changes ENV settings to given and runs given block and restores ENV
|
819
|
+
# to original values before returning.
|
820
|
+
def with_environment(env, &block)
|
821
|
+
EnvUpdater.inject(env) { yield }
|
613
822
|
end
|
614
823
|
|
615
824
|
def with_argv(argv)
|
@@ -635,7 +844,7 @@ ensure
|
|
635
844
|
end
|
636
845
|
|
637
846
|
def json_dump_and_encode(object)
|
638
|
-
Base64.encode64(
|
847
|
+
Base64.encode64(JSON.dump(object))
|
639
848
|
end
|
640
849
|
|
641
850
|
def get_last_analytics_event
|
@@ -661,15 +870,15 @@ def load_cross_agent_test(name)
|
|
661
870
|
test_file_path = File.join(cross_agent_tests_dir, "#{name}.json")
|
662
871
|
data = File.read(test_file_path)
|
663
872
|
data.gsub!('callCount', 'call_count')
|
664
|
-
data =
|
665
|
-
data.each { |testcase| testcase['testname'].
|
873
|
+
data = JSON.load(data)
|
874
|
+
data.each { |testcase| testcase['testname'].tr!(' ', '_') if String === testcase['testname'] }
|
666
875
|
data
|
667
876
|
end
|
668
877
|
|
669
878
|
def each_cross_agent_test(options)
|
670
879
|
options = {:dir => nil, :pattern => "*"}.update(options)
|
671
|
-
path = File.join
|
672
|
-
Dir.glob(path).each { |file| yield
|
880
|
+
path = File.join([cross_agent_tests_dir, options[:dir], options[:pattern]].compact)
|
881
|
+
Dir.glob(path).each { |file| yield(file) }
|
673
882
|
end
|
674
883
|
|
675
884
|
def assert_event_attributes(event, test_name, expected_attributes, non_expected_attributes)
|
@@ -682,7 +891,7 @@ def assert_event_attributes(event, test_name, expected_attributes, non_expected_
|
|
682
891
|
incorrect_attributes << name unless actual_value == expected_value
|
683
892
|
end
|
684
893
|
|
685
|
-
msg = "Found missing or incorrect attribute values in #{test_name}:\n"
|
894
|
+
msg = String.new("Found missing or incorrect attribute values in #{test_name}:\n")
|
686
895
|
|
687
896
|
incorrect_attributes.each do |name|
|
688
897
|
msg << " #{name}: expected = #{expected_attributes[name].inspect}, actual = #{event_attrs[name].inspect}\n"
|
@@ -693,13 +902,125 @@ def assert_event_attributes(event, test_name, expected_attributes, non_expected_
|
|
693
902
|
event_attrs.each do |name, actual_value|
|
694
903
|
msg << " #{name}: #{actual_value.inspect}\n"
|
695
904
|
end
|
696
|
-
|
905
|
+
|
906
|
+
assert_empty(incorrect_attributes, msg)
|
697
907
|
|
698
908
|
non_expected_attributes.each do |name|
|
699
|
-
|
909
|
+
refute event_attrs[name], "Found value '#{event_attrs[name]}' for attribute '#{name}', but expected nothing in #{test_name}"
|
700
910
|
end
|
701
911
|
end
|
702
912
|
|
703
913
|
def attributes_for(sample, type)
|
704
914
|
sample.attributes.instance_variable_get("@#{type}_attributes")
|
705
915
|
end
|
916
|
+
|
917
|
+
def uncache_trusted_account_key
|
918
|
+
NewRelic::Agent::Transaction::TraceContext::AccountHelpers.instance_variable_set(:@trace_state_entry_key, nil)
|
919
|
+
end
|
920
|
+
|
921
|
+
def reset_buffers_and_caches
|
922
|
+
NewRelic::Agent.drop_buffered_data
|
923
|
+
uncache_trusted_account_key
|
924
|
+
end
|
925
|
+
|
926
|
+
def message_for_status_code(code)
|
927
|
+
# Net::HTTP::STATUS_CODES was introduced in Ruby 2.5
|
928
|
+
if defined?(Net::HTTP::STATUS_CODES)
|
929
|
+
return Net::HTTP::STATUS_CODES[code]
|
930
|
+
end
|
931
|
+
|
932
|
+
case code
|
933
|
+
when 200 then "OK"
|
934
|
+
when 404 then "Not Found"
|
935
|
+
when 403 then "Forbidden"
|
936
|
+
else "Unknown"
|
937
|
+
end
|
938
|
+
end
|
939
|
+
|
940
|
+
# wraps the given headers in a Net::HTTPResponse which has accompanying
|
941
|
+
# http status code associated with it.
|
942
|
+
# a "status_code" may be passed in the headers to alter the HTTP Status Code
|
943
|
+
# that is wrapped in the response.
|
944
|
+
def mock_http_response(headers, wrap_it = true)
|
945
|
+
status_code = (headers.delete("status_code") || 200).to_i
|
946
|
+
net_http_resp = Net::HTTPResponse.new(1.0, status_code, message_for_status_code(status_code))
|
947
|
+
headers.each do |key, value|
|
948
|
+
net_http_resp.add_field(key.to_s, value)
|
949
|
+
end
|
950
|
+
return net_http_resp unless wrap_it
|
951
|
+
|
952
|
+
NewRelic::Agent::HTTPClients::NetHTTPResponse.new(net_http_resp)
|
953
|
+
end
|
954
|
+
|
955
|
+
# +expected+ can be a string or regular expression
|
956
|
+
def assert_match_or_equal(expected, value)
|
957
|
+
if expected.is_a?(Regexp)
|
958
|
+
assert_match expected, value
|
959
|
+
else
|
960
|
+
assert_equal expected, value
|
961
|
+
end
|
962
|
+
end
|
963
|
+
|
964
|
+
# selects the last segment with a noticed_error and checks
|
965
|
+
# the expectations against it.
|
966
|
+
def assert_segment_noticed_error(txn, segment_name, error_classes, error_message)
|
967
|
+
error_segment = txn.segments.reverse.detect { |s| s.noticed_error }
|
968
|
+
|
969
|
+
assert error_segment, "Expected at least one segment with a noticed_error"
|
970
|
+
|
971
|
+
assert_match_or_equal segment_name, error_segment.name
|
972
|
+
|
973
|
+
noticed_error = error_segment.noticed_error
|
974
|
+
|
975
|
+
assert_match_or_equal error_classes, noticed_error.exception_class_name
|
976
|
+
assert_match_or_equal error_message, noticed_error.message
|
977
|
+
end
|
978
|
+
|
979
|
+
def assert_transaction_noticed_error(txn, error_classes)
|
980
|
+
refute_empty txn.exceptions, "Expected transaction to notice the error"
|
981
|
+
assert_match_or_equal error_classes, txn.exceptions.keys.first.class.name
|
982
|
+
end
|
983
|
+
|
984
|
+
def refute_transaction_noticed_error(txn, error_class)
|
985
|
+
error_segment = txn.segments.reverse.detect { |s| s.noticed_error }
|
986
|
+
|
987
|
+
assert error_segment, "Expected at least one segment with a noticed_error"
|
988
|
+
assert_empty txn.exceptions, "Expected transaction to NOT notice any segment errors"
|
989
|
+
end
|
990
|
+
|
991
|
+
def refute_raises(*exp)
|
992
|
+
msg = "#{exp.pop}.\n" if String === exp.last
|
993
|
+
|
994
|
+
begin
|
995
|
+
yield
|
996
|
+
rescue MiniTest::Skip => e
|
997
|
+
puts "SKIP REPORTS: #{e.inspect}"
|
998
|
+
return e if exp.include?(MiniTest::Skip)
|
999
|
+
|
1000
|
+
raise e
|
1001
|
+
rescue Exception => e
|
1002
|
+
puts "EXCEPTION RAISED: #{e.inspect}\n#{e.backtrace}"
|
1003
|
+
exp = exp.first if exp.size == 1
|
1004
|
+
|
1005
|
+
flunk(msg || "unexpected exception raised: #{e}")
|
1006
|
+
end
|
1007
|
+
end
|
1008
|
+
|
1009
|
+
def assert_implements(instance, method, *args)
|
1010
|
+
fail_message = "expected #{instance.class}##{method} method to be implemented"
|
1011
|
+
refute_raises NotImplementedError, fail_message do
|
1012
|
+
instance.send(method, *args)
|
1013
|
+
end
|
1014
|
+
end
|
1015
|
+
|
1016
|
+
def defer_testing_to_min_supported_rails(test_file, min_rails_version, supports_jruby = true)
|
1017
|
+
if defined?(Rails) &&
|
1018
|
+
defined?(Rails::VERSION::STRING) &&
|
1019
|
+
(Rails::VERSION::STRING.to_f >= min_rails_version) &&
|
1020
|
+
(supports_jruby || !NewRelic::LanguageSupport.jruby?)
|
1021
|
+
|
1022
|
+
yield
|
1023
|
+
else
|
1024
|
+
puts "Skipping tests in #{File.basename(test_file)} because Rails >= #{min_rails_version} is unavailable" if ENV["VERBOSE_TEST_OUTPUT"]
|
1025
|
+
end
|
1026
|
+
end
|