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.
Files changed (476) hide show
  1. checksums.yaml +4 -4
  2. data/.gitignore +15 -1
  3. data/.rubocop.yml +1919 -0
  4. data/.rubocop_todo.yml +100 -0
  5. data/.simplecov +15 -0
  6. data/.snyk +11 -0
  7. data/.yardopts +2 -0
  8. data/Brewfile +12 -0
  9. data/CHANGELOG.md +4056 -2339
  10. data/CONTRIBUTING.md +132 -19
  11. data/DOCKER.md +167 -0
  12. data/Dockerfile +10 -0
  13. data/Gemfile +5 -2
  14. data/Guardfile +22 -4
  15. data/LICENSE +202 -38
  16. data/README.md +87 -87
  17. data/Rakefile +27 -27
  18. data/THIRD_PARTY_NOTICES.md +28 -0
  19. data/Thorfile +5 -0
  20. data/bin/newrelic +3 -2
  21. data/bin/newrelic_cmd +1 -0
  22. data/bin/nrdebug +77 -54
  23. data/config.dot +5 -5
  24. data/docker-compose.yml +107 -0
  25. data/init.rb +5 -7
  26. data/install.rb +3 -3
  27. data/lefthook.yml +9 -0
  28. data/lib/new_relic/agent/adaptive_sampler.rb +14 -10
  29. data/lib/new_relic/agent/agent.rb +125 -969
  30. data/lib/new_relic/agent/agent_helpers/connect.rb +227 -0
  31. data/lib/new_relic/agent/agent_helpers/harvest.rb +153 -0
  32. data/lib/new_relic/agent/agent_helpers/shutdown.rb +72 -0
  33. data/lib/new_relic/agent/agent_helpers/special_startup.rb +74 -0
  34. data/lib/new_relic/agent/agent_helpers/start_worker_thread.rb +167 -0
  35. data/lib/new_relic/agent/agent_helpers/startup.rb +202 -0
  36. data/lib/new_relic/agent/agent_helpers/transmit.rb +76 -0
  37. data/lib/new_relic/agent/agent_logger.rb +26 -18
  38. data/lib/new_relic/agent/attribute_filter.rb +69 -52
  39. data/lib/new_relic/agent/attribute_processing.rb +8 -8
  40. data/lib/new_relic/agent/attributes.rb +153 -0
  41. data/lib/new_relic/agent/audit_logger.rb +19 -4
  42. data/lib/new_relic/agent/autostart.rb +34 -28
  43. data/lib/new_relic/agent/chained_call.rb +2 -2
  44. data/lib/new_relic/agent/commands/agent_command.rb +4 -4
  45. data/lib/new_relic/agent/commands/agent_command_router.rb +15 -33
  46. data/lib/new_relic/agent/commands/thread_profiler_session.rb +13 -11
  47. data/lib/new_relic/agent/configuration/default_source.rb +1480 -1053
  48. data/lib/new_relic/agent/configuration/dotted_hash.rb +7 -6
  49. data/lib/new_relic/agent/configuration/environment_source.rb +15 -11
  50. data/lib/new_relic/agent/configuration/event_harvest_config.rb +68 -0
  51. data/lib/new_relic/agent/configuration/high_security_source.rb +9 -9
  52. data/lib/new_relic/agent/configuration/manager.rb +96 -79
  53. data/lib/new_relic/agent/configuration/manual_source.rb +2 -2
  54. data/lib/new_relic/agent/configuration/mask_defaults.rb +4 -4
  55. data/lib/new_relic/agent/configuration/security_policy_source.rb +83 -86
  56. data/lib/new_relic/agent/configuration/server_source.rb +49 -12
  57. data/lib/new_relic/agent/configuration/yaml_source.rb +42 -13
  58. data/lib/new_relic/agent/configuration.rb +2 -2
  59. data/lib/new_relic/agent/connect/request_builder.rb +61 -0
  60. data/lib/new_relic/agent/connect/response_handler.rb +58 -0
  61. data/lib/new_relic/agent/custom_event_aggregator.rb +15 -15
  62. data/lib/new_relic/agent/database/explain_plan_helpers.rb +5 -6
  63. data/lib/new_relic/agent/database/obfuscation_helpers.rb +16 -15
  64. data/lib/new_relic/agent/database/obfuscator.rb +3 -3
  65. data/lib/new_relic/agent/database/postgres_explain_obfuscator.rb +4 -4
  66. data/lib/new_relic/agent/database.rb +44 -53
  67. data/lib/new_relic/agent/database_adapter.rb +35 -0
  68. data/lib/new_relic/agent/datastores/metric_helper.rb +18 -20
  69. data/lib/new_relic/agent/datastores/mongo/event_formatter.rb +9 -8
  70. data/lib/new_relic/agent/datastores/mongo/metric_translator.rb +7 -11
  71. data/lib/new_relic/agent/datastores/mongo.rb +7 -12
  72. data/lib/new_relic/agent/datastores/nosql_obfuscator.rb +41 -0
  73. data/lib/new_relic/agent/datastores/redis.rb +6 -12
  74. data/lib/new_relic/agent/datastores.rb +19 -23
  75. data/lib/new_relic/agent/deprecator.rb +2 -2
  76. data/lib/new_relic/agent/{cross_app_payload.rb → distributed_tracing/cross_app_payload.rb} +13 -12
  77. data/lib/new_relic/agent/{cross_app_tracing.rb → distributed_tracing/cross_app_tracing.rb} +87 -66
  78. data/lib/new_relic/agent/distributed_tracing/distributed_trace_attributes.rb +84 -0
  79. data/lib/new_relic/agent/distributed_tracing/distributed_trace_metrics.rb +75 -0
  80. data/lib/new_relic/agent/distributed_tracing/distributed_trace_payload.rb +163 -0
  81. data/lib/new_relic/agent/distributed_tracing/distributed_trace_transport_type.rb +38 -0
  82. data/lib/new_relic/agent/distributed_tracing/trace_context.rb +245 -0
  83. data/lib/new_relic/agent/distributed_tracing/trace_context_payload.rb +127 -0
  84. data/lib/new_relic/agent/distributed_tracing.rb +113 -32
  85. data/lib/new_relic/agent/encoding_normalizer.rb +5 -3
  86. data/lib/new_relic/agent/error_collector.rb +99 -63
  87. data/lib/new_relic/agent/error_event_aggregator.rb +10 -8
  88. data/lib/new_relic/agent/error_filter.rb +174 -0
  89. data/lib/new_relic/agent/error_trace_aggregator.rb +6 -4
  90. data/lib/new_relic/agent/event_aggregator.rb +43 -48
  91. data/lib/new_relic/agent/event_buffer.rb +8 -9
  92. data/lib/new_relic/agent/event_listener.rb +2 -3
  93. data/lib/new_relic/agent/event_loop.rb +27 -25
  94. data/lib/new_relic/agent/external.rb +20 -51
  95. data/lib/new_relic/agent/guid_generator.rb +30 -0
  96. data/lib/new_relic/agent/harvester.rb +5 -6
  97. data/lib/new_relic/agent/heap.rb +8 -10
  98. data/lib/new_relic/agent/hostname.rb +26 -5
  99. data/lib/new_relic/agent/http_clients/abstract.rb +81 -0
  100. data/lib/new_relic/agent/http_clients/curb_wrappers.rb +26 -26
  101. data/lib/new_relic/agent/http_clients/excon_wrappers.rb +31 -17
  102. data/lib/new_relic/agent/http_clients/http_rb_wrappers.rb +18 -23
  103. data/lib/new_relic/agent/http_clients/httpclient_wrappers.rb +12 -15
  104. data/lib/new_relic/agent/http_clients/net_http_wrappers.rb +24 -8
  105. data/lib/new_relic/agent/http_clients/typhoeus_wrappers.rb +9 -12
  106. data/lib/new_relic/agent/http_clients/uri_util.rb +12 -13
  107. data/lib/new_relic/agent/instrumentation/action_cable_subscriber.rb +22 -52
  108. data/lib/new_relic/agent/instrumentation/action_controller_other_subscriber.rb +39 -0
  109. data/lib/new_relic/agent/instrumentation/action_controller_subscriber.rb +59 -72
  110. data/lib/new_relic/agent/instrumentation/action_dispatch.rb +31 -0
  111. data/lib/new_relic/agent/instrumentation/action_dispatch_subscriber.rb +64 -0
  112. data/lib/new_relic/agent/instrumentation/action_mailbox.rb +30 -0
  113. data/lib/new_relic/agent/instrumentation/action_mailbox_subscriber.rb +33 -0
  114. data/lib/new_relic/agent/instrumentation/action_mailer.rb +30 -0
  115. data/lib/new_relic/agent/instrumentation/action_mailer_subscriber.rb +85 -0
  116. data/lib/new_relic/agent/instrumentation/action_view_subscriber.rb +86 -62
  117. data/lib/new_relic/agent/instrumentation/active_job.rb +38 -19
  118. data/lib/new_relic/agent/instrumentation/active_job_subscriber.rb +41 -0
  119. data/lib/new_relic/agent/instrumentation/active_merchant.rb +21 -7
  120. data/lib/new_relic/agent/instrumentation/active_record.rb +95 -46
  121. data/lib/new_relic/agent/instrumentation/active_record_helper.rb +82 -61
  122. data/lib/new_relic/agent/instrumentation/active_record_notifications.rb +155 -0
  123. data/lib/new_relic/agent/instrumentation/active_record_prepend.rb +36 -12
  124. data/lib/new_relic/agent/instrumentation/active_record_subscriber.rb +69 -64
  125. data/lib/new_relic/agent/instrumentation/active_storage.rb +8 -4
  126. data/lib/new_relic/agent/instrumentation/active_storage_subscriber.rb +11 -32
  127. data/lib/new_relic/agent/instrumentation/active_support.rb +27 -0
  128. data/lib/new_relic/agent/instrumentation/active_support_logger/chain.rb +23 -0
  129. data/lib/new_relic/agent/instrumentation/active_support_logger/instrumentation.rb +20 -0
  130. data/lib/new_relic/agent/instrumentation/active_support_logger/prepend.rb +12 -0
  131. data/lib/new_relic/agent/instrumentation/active_support_logger.rb +24 -0
  132. data/lib/new_relic/agent/instrumentation/active_support_subscriber.rb +41 -0
  133. data/lib/new_relic/agent/instrumentation/bunny/chain.rb +45 -0
  134. data/lib/new_relic/agent/instrumentation/bunny/instrumentation.rb +150 -0
  135. data/lib/new_relic/agent/instrumentation/bunny/prepend.rb +35 -0
  136. data/lib/new_relic/agent/instrumentation/bunny.rb +14 -134
  137. data/lib/new_relic/agent/instrumentation/concurrent_ruby/chain.rb +36 -0
  138. data/lib/new_relic/agent/instrumentation/concurrent_ruby/instrumentation.rb +21 -0
  139. data/lib/new_relic/agent/instrumentation/concurrent_ruby/prepend.rb +27 -0
  140. data/lib/new_relic/agent/instrumentation/concurrent_ruby.rb +31 -0
  141. data/lib/new_relic/agent/instrumentation/controller_instrumentation.rb +77 -61
  142. data/lib/new_relic/agent/instrumentation/curb/chain.rb +91 -0
  143. data/lib/new_relic/agent/instrumentation/curb/instrumentation.rb +221 -0
  144. data/lib/new_relic/agent/instrumentation/curb/prepend.rb +61 -0
  145. data/lib/new_relic/agent/instrumentation/curb.rb +15 -187
  146. data/lib/new_relic/agent/instrumentation/custom_events.rb +12 -0
  147. data/lib/new_relic/agent/instrumentation/custom_events_subscriber.rb +37 -0
  148. data/lib/new_relic/agent/instrumentation/delayed_job/chain.rb +35 -0
  149. data/lib/new_relic/agent/instrumentation/delayed_job/instrumentation.rb +48 -0
  150. data/lib/new_relic/agent/instrumentation/delayed_job/prepend.rb +33 -0
  151. data/lib/new_relic/agent/instrumentation/delayed_job_instrumentation.rb +30 -52
  152. data/lib/new_relic/agent/instrumentation/elasticsearch/chain.rb +29 -0
  153. data/lib/new_relic/agent/instrumentation/elasticsearch/instrumentation.rb +66 -0
  154. data/lib/new_relic/agent/instrumentation/elasticsearch/prepend.rb +13 -0
  155. data/lib/new_relic/agent/instrumentation/elasticsearch.rb +31 -0
  156. data/lib/new_relic/agent/instrumentation/excon/middleware.rb +8 -7
  157. data/lib/new_relic/agent/instrumentation/excon.rb +29 -31
  158. data/lib/new_relic/agent/instrumentation/fiber/chain.rb +20 -0
  159. data/lib/new_relic/agent/instrumentation/fiber/instrumentation.rb +24 -0
  160. data/lib/new_relic/agent/instrumentation/fiber/prepend.rb +18 -0
  161. data/lib/new_relic/agent/instrumentation/fiber.rb +25 -0
  162. data/lib/new_relic/agent/instrumentation/grape/chain.rb +24 -0
  163. data/lib/new_relic/agent/instrumentation/grape/instrumentation.rb +100 -0
  164. data/lib/new_relic/agent/instrumentation/grape/prepend.rb +17 -0
  165. data/lib/new_relic/agent/instrumentation/grape.rb +16 -121
  166. data/lib/new_relic/agent/instrumentation/grpc/client/chain.rb +97 -0
  167. data/lib/new_relic/agent/instrumentation/grpc/client/instrumentation.rb +89 -0
  168. data/lib/new_relic/agent/instrumentation/grpc/client/prepend.rb +111 -0
  169. data/lib/new_relic/agent/instrumentation/grpc/client/request_wrapper.rb +30 -0
  170. data/lib/new_relic/agent/instrumentation/grpc/helper.rb +32 -0
  171. data/lib/new_relic/agent/instrumentation/grpc/server/chain.rb +69 -0
  172. data/lib/new_relic/agent/instrumentation/grpc/server/instrumentation.rb +134 -0
  173. data/lib/new_relic/agent/instrumentation/grpc/server/rpc_desc_prepend.rb +35 -0
  174. data/lib/new_relic/agent/instrumentation/grpc/server/rpc_server_prepend.rb +26 -0
  175. data/lib/new_relic/agent/instrumentation/grpc_client.rb +23 -0
  176. data/lib/new_relic/agent/instrumentation/grpc_server.rb +25 -0
  177. data/lib/new_relic/agent/instrumentation/httpclient/chain.rb +24 -0
  178. data/lib/new_relic/agent/instrumentation/httpclient/instrumentation.rb +37 -0
  179. data/lib/new_relic/agent/instrumentation/httpclient/prepend.rb +15 -0
  180. data/lib/new_relic/agent/instrumentation/httpclient.rb +12 -32
  181. data/lib/new_relic/agent/instrumentation/httprb/chain.rb +22 -0
  182. data/lib/new_relic/agent/instrumentation/httprb/instrumentation.rb +30 -0
  183. data/lib/new_relic/agent/instrumentation/httprb/prepend.rb +15 -0
  184. data/lib/new_relic/agent/instrumentation/httprb.rb +29 -0
  185. data/lib/new_relic/agent/instrumentation/ignore_actions.rb +5 -6
  186. data/lib/new_relic/agent/instrumentation/logger/chain.rb +21 -0
  187. data/lib/new_relic/agent/instrumentation/logger/instrumentation.rb +66 -0
  188. data/lib/new_relic/agent/instrumentation/logger/prepend.rb +13 -0
  189. data/lib/new_relic/agent/instrumentation/logger.rb +26 -0
  190. data/lib/new_relic/agent/instrumentation/memcache/chain.rb +15 -0
  191. data/lib/new_relic/agent/instrumentation/memcache/dalli.rb +58 -125
  192. data/lib/new_relic/agent/instrumentation/memcache/helper.rb +59 -0
  193. data/lib/new_relic/agent/instrumentation/memcache/instrumentation.rb +90 -0
  194. data/lib/new_relic/agent/instrumentation/memcache/prepend.rb +101 -0
  195. data/lib/new_relic/agent/instrumentation/memcache.rb +57 -71
  196. data/lib/new_relic/agent/instrumentation/middleware_proxy.rb +15 -14
  197. data/lib/new_relic/agent/instrumentation/middleware_tracing.rb +21 -14
  198. data/lib/new_relic/agent/instrumentation/mongo.rb +7 -132
  199. data/lib/new_relic/agent/instrumentation/mongodb_command_subscriber.rb +49 -13
  200. data/lib/new_relic/agent/instrumentation/net_http/chain.rb +24 -0
  201. data/lib/new_relic/agent/instrumentation/net_http/instrumentation.rb +40 -0
  202. data/lib/new_relic/agent/instrumentation/net_http/prepend.rb +21 -0
  203. data/lib/new_relic/agent/instrumentation/net_http.rb +44 -0
  204. data/lib/new_relic/agent/instrumentation/notifications_subscriber.rb +142 -0
  205. data/lib/new_relic/agent/instrumentation/padrino/chain.rb +38 -0
  206. data/lib/new_relic/agent/instrumentation/padrino/instrumentation.rb +28 -0
  207. data/lib/new_relic/agent/instrumentation/padrino/prepend.rb +20 -0
  208. data/lib/new_relic/agent/instrumentation/padrino.rb +22 -58
  209. data/lib/new_relic/agent/instrumentation/passenger_instrumentation.rb +7 -7
  210. data/lib/new_relic/agent/instrumentation/queue_time.rb +9 -10
  211. data/lib/new_relic/agent/instrumentation/rack/chain.rb +66 -0
  212. data/lib/new_relic/agent/instrumentation/rack/helpers.rb +33 -0
  213. data/lib/new_relic/agent/instrumentation/rack/instrumentation.rb +75 -0
  214. data/lib/new_relic/agent/instrumentation/rack/prepend.rb +43 -0
  215. data/lib/new_relic/agent/instrumentation/rack.rb +33 -141
  216. data/lib/new_relic/agent/instrumentation/rails3/action_controller.rb +19 -55
  217. data/lib/new_relic/agent/instrumentation/rails_middleware.rb +5 -5
  218. data/lib/new_relic/agent/instrumentation/rails_notifications/action_cable.rb +36 -0
  219. data/lib/new_relic/agent/instrumentation/rails_notifications/action_controller.rb +45 -0
  220. data/lib/new_relic/agent/instrumentation/rails_notifications/action_view.rb +30 -0
  221. data/lib/new_relic/agent/instrumentation/rails_notifications/custom_events.rb +30 -0
  222. data/lib/new_relic/agent/instrumentation/rake/chain.rb +20 -0
  223. data/lib/new_relic/agent/instrumentation/rake/instrumentation.rb +142 -0
  224. data/lib/new_relic/agent/instrumentation/rake/prepend.rb +14 -0
  225. data/lib/new_relic/agent/instrumentation/rake.rb +18 -159
  226. data/lib/new_relic/agent/instrumentation/redis/chain.rb +45 -0
  227. data/lib/new_relic/agent/instrumentation/redis/constants.rb +17 -0
  228. data/lib/new_relic/agent/instrumentation/redis/instrumentation.rb +93 -0
  229. data/lib/new_relic/agent/instrumentation/redis/middleware.rb +16 -0
  230. data/lib/new_relic/agent/instrumentation/redis/prepend.rb +29 -0
  231. data/lib/new_relic/agent/instrumentation/redis.rb +20 -103
  232. data/lib/new_relic/agent/instrumentation/resque/chain.rb +21 -0
  233. data/lib/new_relic/agent/instrumentation/resque/helper.rb +19 -0
  234. data/lib/new_relic/agent/instrumentation/resque/instrumentation.rb +34 -0
  235. data/lib/new_relic/agent/instrumentation/resque/prepend.rb +15 -0
  236. data/lib/new_relic/agent/instrumentation/resque.rb +33 -41
  237. data/lib/new_relic/agent/instrumentation/sequel.rb +17 -20
  238. data/lib/new_relic/agent/instrumentation/sequel_helper.rb +3 -3
  239. data/lib/new_relic/agent/instrumentation/sidekiq/client.rb +20 -0
  240. data/lib/new_relic/agent/instrumentation/sidekiq/extensions/delayed_class.rb +30 -0
  241. data/lib/new_relic/agent/instrumentation/sidekiq/server.rb +37 -0
  242. data/lib/new_relic/agent/instrumentation/sidekiq.rb +29 -46
  243. data/lib/new_relic/agent/instrumentation/sinatra/chain.rb +55 -0
  244. data/lib/new_relic/agent/instrumentation/sinatra/ignorer.rb +31 -37
  245. data/lib/new_relic/agent/instrumentation/sinatra/instrumentation.rb +125 -0
  246. data/lib/new_relic/agent/instrumentation/sinatra/prepend.rb +33 -0
  247. data/lib/new_relic/agent/instrumentation/sinatra/transaction_namer.rb +3 -3
  248. data/lib/new_relic/agent/instrumentation/sinatra.rb +35 -165
  249. data/lib/new_relic/agent/instrumentation/thread/chain.rb +24 -0
  250. data/lib/new_relic/agent/instrumentation/thread/instrumentation.rb +28 -0
  251. data/lib/new_relic/agent/instrumentation/thread/prepend.rb +22 -0
  252. data/lib/new_relic/agent/instrumentation/thread.rb +20 -0
  253. data/lib/new_relic/agent/instrumentation/tilt/chain.rb +24 -0
  254. data/lib/new_relic/agent/instrumentation/tilt/instrumentation.rb +41 -0
  255. data/lib/new_relic/agent/instrumentation/tilt/prepend.rb +13 -0
  256. data/lib/new_relic/agent/instrumentation/tilt.rb +25 -0
  257. data/lib/new_relic/agent/instrumentation/typhoeus/chain.rb +22 -0
  258. data/lib/new_relic/agent/instrumentation/typhoeus/instrumentation.rb +80 -0
  259. data/lib/new_relic/agent/instrumentation/typhoeus/prepend.rb +14 -0
  260. data/lib/new_relic/agent/instrumentation/typhoeus.rb +14 -76
  261. data/lib/new_relic/agent/instrumentation.rb +2 -2
  262. data/lib/new_relic/agent/internal_agent_error.rb +3 -3
  263. data/lib/new_relic/agent/javascript_instrumentor.rb +51 -45
  264. data/lib/new_relic/agent/linking_metadata.rb +44 -0
  265. data/lib/new_relic/agent/local_log_decorator.rb +37 -0
  266. data/lib/new_relic/agent/log_event_aggregator.rb +235 -0
  267. data/lib/new_relic/agent/log_once.rb +2 -2
  268. data/lib/new_relic/agent/log_priority.rb +20 -0
  269. data/lib/new_relic/agent/logging.rb +142 -0
  270. data/lib/new_relic/agent/memory_logger.rb +3 -3
  271. data/lib/new_relic/agent/messaging.rb +81 -164
  272. data/lib/new_relic/agent/method_tracer.rb +152 -145
  273. data/lib/new_relic/agent/method_tracer_helpers.rb +90 -13
  274. data/lib/new_relic/agent/monitors/cross_app_monitor.rb +117 -0
  275. data/lib/new_relic/agent/monitors/distributed_tracing_monitor.rb +28 -0
  276. data/lib/new_relic/agent/{inbound_request_monitor.rb → monitors/inbound_request_monitor.rb} +5 -6
  277. data/lib/new_relic/agent/{synthetics_monitor.rb → monitors/synthetics_monitor.rb} +9 -15
  278. data/lib/new_relic/agent/monitors.rb +26 -0
  279. data/lib/new_relic/agent/new_relic_service/encoders.rb +7 -7
  280. data/lib/new_relic/agent/new_relic_service/json_marshaller.rb +6 -7
  281. data/lib/new_relic/agent/new_relic_service/marshaller.rb +8 -29
  282. data/lib/new_relic/agent/new_relic_service/security_policy_settings.rb +5 -5
  283. data/lib/new_relic/agent/new_relic_service.rb +282 -166
  284. data/lib/new_relic/agent/noticeable_error.rb +19 -0
  285. data/lib/new_relic/agent/null_logger.rb +8 -4
  286. data/lib/new_relic/agent/obfuscator.rb +9 -11
  287. data/lib/new_relic/agent/parameter_filtering.rb +35 -8
  288. data/lib/new_relic/agent/payload_metric_mapping.rb +10 -11
  289. data/lib/new_relic/agent/pipe_channel_manager.rb +28 -18
  290. data/lib/new_relic/agent/pipe_service.rb +9 -6
  291. data/lib/new_relic/agent/prepend_supportability.rb +3 -3
  292. data/lib/new_relic/agent/priority_sampled_buffer.rb +16 -14
  293. data/lib/new_relic/agent/range_extensions.rb +9 -29
  294. data/lib/new_relic/agent/rules_engine/replacement_rule.rb +12 -12
  295. data/lib/new_relic/agent/rules_engine/segment_terms_rule.rb +13 -14
  296. data/lib/new_relic/agent/rules_engine.rb +6 -5
  297. data/lib/new_relic/agent/sampler.rb +4 -5
  298. data/lib/new_relic/agent/sampler_collection.rb +4 -5
  299. data/lib/new_relic/agent/samplers/cpu_sampler.rb +4 -3
  300. data/lib/new_relic/agent/samplers/delayed_job_sampler.rb +14 -11
  301. data/lib/new_relic/agent/samplers/memory_sampler.rb +26 -15
  302. data/lib/new_relic/agent/samplers/object_sampler.rb +2 -2
  303. data/lib/new_relic/agent/samplers/vm_sampler.rb +22 -20
  304. data/lib/new_relic/agent/span_event_aggregator.rb +14 -16
  305. data/lib/new_relic/agent/span_event_primitive.rb +118 -58
  306. data/lib/new_relic/agent/sql_sampler.rb +25 -25
  307. data/lib/new_relic/agent/stats.rb +79 -42
  308. data/lib/new_relic/agent/stats_engine/gc_profiler.rb +11 -13
  309. data/lib/new_relic/agent/stats_engine/stats_hash.rb +13 -14
  310. data/lib/new_relic/agent/stats_engine.rb +11 -11
  311. data/lib/new_relic/agent/synthetics_event_aggregator.rb +8 -9
  312. data/lib/new_relic/agent/system_info.rb +100 -66
  313. data/lib/new_relic/agent/threading/agent_thread.rb +20 -16
  314. data/lib/new_relic/agent/threading/backtrace_node.rb +13 -14
  315. data/lib/new_relic/agent/threading/backtrace_service.rb +18 -18
  316. data/lib/new_relic/agent/threading/thread_profile.rb +31 -45
  317. data/lib/new_relic/agent/timestamp_sampled_buffer.rb +2 -2
  318. data/lib/new_relic/agent/tracer.rb +513 -0
  319. data/lib/new_relic/agent/transaction/abstract_segment.rb +131 -41
  320. data/lib/new_relic/agent/transaction/datastore_segment.rb +22 -18
  321. data/lib/new_relic/agent/transaction/distributed_tracer.rb +184 -0
  322. data/lib/new_relic/agent/transaction/distributed_tracing.rb +72 -163
  323. data/lib/new_relic/agent/transaction/external_request_segment.rb +66 -63
  324. data/lib/new_relic/agent/transaction/message_broker_segment.rb +34 -46
  325. data/lib/new_relic/agent/transaction/request_attributes.rb +36 -36
  326. data/lib/new_relic/agent/transaction/segment.rb +46 -10
  327. data/lib/new_relic/agent/transaction/slowest_sample_buffer.rb +2 -4
  328. data/lib/new_relic/agent/transaction/synthetics_sample_buffer.rb +2 -2
  329. data/lib/new_relic/agent/transaction/trace.rb +21 -24
  330. data/lib/new_relic/agent/transaction/trace_builder.rb +11 -12
  331. data/lib/new_relic/agent/transaction/trace_context.rb +168 -0
  332. data/lib/new_relic/agent/transaction/trace_node.rb +31 -28
  333. data/lib/new_relic/agent/transaction/tracing.rb +15 -111
  334. data/lib/new_relic/agent/transaction/transaction_sample_buffer.rb +6 -6
  335. data/lib/new_relic/agent/transaction.rb +252 -259
  336. data/lib/new_relic/agent/transaction_error_primitive.rb +34 -37
  337. data/lib/new_relic/agent/transaction_event_aggregator.rb +13 -13
  338. data/lib/new_relic/agent/transaction_event_primitive.rb +44 -56
  339. data/lib/new_relic/agent/transaction_event_recorder.rb +17 -16
  340. data/lib/new_relic/agent/transaction_metrics.rb +11 -10
  341. data/lib/new_relic/agent/transaction_sampler.rb +7 -12
  342. data/lib/new_relic/agent/transaction_time_aggregator.rb +41 -26
  343. data/lib/new_relic/agent/utilization/aws.rb +34 -4
  344. data/lib/new_relic/agent/utilization/azure.rb +4 -4
  345. data/lib/new_relic/agent/utilization/gcp.rb +8 -8
  346. data/lib/new_relic/agent/utilization/pcf.rb +6 -5
  347. data/lib/new_relic/agent/utilization/vendor.rb +44 -29
  348. data/lib/new_relic/agent/utilization_data.rb +43 -6
  349. data/lib/new_relic/agent/vm/jruby_vm.rb +2 -2
  350. data/lib/new_relic/agent/vm/monotonic_gc_profiler.rb +3 -3
  351. data/lib/new_relic/agent/vm/mri_vm.rb +46 -19
  352. data/lib/new_relic/agent/vm/snapshot.rb +6 -6
  353. data/lib/new_relic/agent/vm.rb +2 -2
  354. data/lib/new_relic/agent/worker_loop.rb +11 -13
  355. data/lib/new_relic/agent.rb +151 -79
  356. data/lib/new_relic/cli/command.rb +21 -23
  357. data/lib/new_relic/cli/commands/deployments.rb +94 -45
  358. data/lib/new_relic/cli/commands/install.rb +24 -26
  359. data/lib/new_relic/coerce.rb +42 -15
  360. data/lib/new_relic/collection_helper.rb +51 -49
  361. data/lib/new_relic/constants.rb +39 -0
  362. data/lib/new_relic/control/class_methods.rb +11 -5
  363. data/lib/new_relic/control/frameworks/external.rb +3 -3
  364. data/lib/new_relic/control/frameworks/rails.rb +24 -18
  365. data/lib/new_relic/control/frameworks/rails3.rb +4 -5
  366. data/lib/new_relic/control/frameworks/rails4.rb +2 -2
  367. data/lib/new_relic/control/frameworks/rails_notifications.rb +14 -0
  368. data/lib/new_relic/control/frameworks/ruby.rb +4 -4
  369. data/lib/new_relic/control/frameworks/sinatra.rb +8 -2
  370. data/lib/new_relic/control/frameworks.rb +2 -2
  371. data/lib/new_relic/control/instance_methods.rb +33 -42
  372. data/lib/new_relic/control/instrumentation.rb +40 -12
  373. data/lib/new_relic/control/private_instance_methods.rb +48 -0
  374. data/lib/new_relic/control/server_methods.rb +4 -5
  375. data/lib/new_relic/control.rb +2 -3
  376. data/lib/new_relic/delayed_job_injection.rb +2 -2
  377. data/lib/new_relic/dependency_detection.rb +129 -18
  378. data/lib/new_relic/environment_report.rb +41 -35
  379. data/lib/new_relic/helper.rb +49 -8
  380. data/lib/new_relic/language_support.rb +30 -6
  381. data/lib/new_relic/latest_changes.rb +9 -8
  382. data/lib/new_relic/local_environment.rb +23 -27
  383. data/lib/new_relic/metric_data.rb +32 -27
  384. data/lib/new_relic/metric_spec.rb +9 -7
  385. data/lib/new_relic/noticed_error.rb +46 -33
  386. data/lib/new_relic/rack/agent_hooks.rb +2 -2
  387. data/lib/new_relic/rack/agent_middleware.rb +7 -5
  388. data/lib/new_relic/rack/browser_monitoring.rb +134 -117
  389. data/lib/new_relic/rack.rb +2 -2
  390. data/lib/new_relic/recipes/capistrano3.rb +4 -62
  391. data/lib/new_relic/recipes/capistrano_legacy.rb +24 -27
  392. data/lib/new_relic/recipes/helpers/send_deployment.rb +70 -0
  393. data/lib/new_relic/recipes.rb +2 -2
  394. data/lib/new_relic/supportability_helper.rb +21 -7
  395. data/lib/new_relic/traced_thread.rb +39 -0
  396. data/lib/new_relic/version.rb +7 -18
  397. data/lib/newrelic_rpm.rb +20 -33
  398. data/lib/sequel/extensions/{newrelic_instrumentation.rb → new_relic_instrumentation.rb} +16 -19
  399. data/lib/sequel/plugins/{newrelic_instrumentation.rb → new_relic_instrumentation.rb} +9 -15
  400. data/lib/tasks/all.rb +4 -4
  401. data/lib/tasks/config.rake +22 -118
  402. data/lib/tasks/coverage_report.rake +28 -0
  403. data/lib/tasks/helpers/config.html.erb +21 -0
  404. data/lib/tasks/helpers/format.rb +123 -0
  405. data/lib/tasks/helpers/matches.rb +12 -0
  406. data/lib/tasks/helpers/prompt.rb +24 -0
  407. data/lib/tasks/helpers/removers.rb +33 -0
  408. data/lib/tasks/install.rake +4 -0
  409. data/lib/tasks/instrumentation_generator/README.md +63 -0
  410. data/lib/tasks/instrumentation_generator/TODO.md +33 -0
  411. data/lib/tasks/instrumentation_generator/instrumentation.thor +121 -0
  412. data/lib/tasks/instrumentation_generator/templates/Envfile.tt +9 -0
  413. data/lib/tasks/instrumentation_generator/templates/chain.tt +22 -0
  414. data/lib/tasks/instrumentation_generator/templates/chain_method.tt +8 -0
  415. data/lib/tasks/instrumentation_generator/templates/dependency_detection.tt +29 -0
  416. data/lib/tasks/instrumentation_generator/templates/instrumentation.tt +13 -0
  417. data/lib/tasks/instrumentation_generator/templates/instrumentation_method.tt +3 -0
  418. data/lib/tasks/instrumentation_generator/templates/newrelic.yml.tt +19 -0
  419. data/lib/tasks/instrumentation_generator/templates/prepend.tt +13 -0
  420. data/lib/tasks/instrumentation_generator/templates/prepend_method.tt +3 -0
  421. data/lib/tasks/instrumentation_generator/templates/test.tt +15 -0
  422. data/lib/tasks/multiverse.rake +4 -0
  423. data/lib/tasks/multiverse.rb +12 -5
  424. data/lib/tasks/newrelic.rb +2 -2
  425. data/lib/tasks/tests.rake +14 -14
  426. data/newrelic.yml +672 -3
  427. data/newrelic_rpm.gemspec +40 -31
  428. data/recipes/newrelic.rb +3 -3
  429. data/test/agent_helper.rb +419 -98
  430. metadata +238 -127
  431. data/.travis.yml +0 -228
  432. data/bin/mongrel_rpm +0 -33
  433. data/cert/cacert.pem +0 -1177
  434. data/lib/new_relic/agent/commands/xray_session.rb +0 -55
  435. data/lib/new_relic/agent/commands/xray_session_collection.rb +0 -161
  436. data/lib/new_relic/agent/cross_app_monitor.rb +0 -110
  437. data/lib/new_relic/agent/datastores/mongo/obfuscator.rb +0 -44
  438. data/lib/new_relic/agent/datastores/mongo/statement_formatter.rb +0 -53
  439. data/lib/new_relic/agent/distributed_trace_monitor.rb +0 -41
  440. data/lib/new_relic/agent/distributed_trace_payload.rb +0 -246
  441. data/lib/new_relic/agent/http_clients/abstract_request.rb +0 -31
  442. data/lib/new_relic/agent/instrumentation/active_record_4.rb +0 -42
  443. data/lib/new_relic/agent/instrumentation/active_record_5.rb +0 -41
  444. data/lib/new_relic/agent/instrumentation/acts_as_solr.rb +0 -74
  445. data/lib/new_relic/agent/instrumentation/authlogic.rb +0 -25
  446. data/lib/new_relic/agent/instrumentation/data_mapper.rb +0 -202
  447. data/lib/new_relic/agent/instrumentation/evented_subscriber.rb +0 -104
  448. data/lib/new_relic/agent/instrumentation/excon/connection.rb +0 -46
  449. data/lib/new_relic/agent/instrumentation/http.rb +0 -46
  450. data/lib/new_relic/agent/instrumentation/merb/controller.rb +0 -44
  451. data/lib/new_relic/agent/instrumentation/merb/errors.rb +0 -33
  452. data/lib/new_relic/agent/instrumentation/net.rb +0 -50
  453. data/lib/new_relic/agent/instrumentation/rails/action_controller.rb +0 -125
  454. data/lib/new_relic/agent/instrumentation/rails/action_web_service.rb +0 -46
  455. data/lib/new_relic/agent/instrumentation/rails4/action_controller.rb +0 -32
  456. data/lib/new_relic/agent/instrumentation/rails4/action_view.rb +0 -27
  457. data/lib/new_relic/agent/instrumentation/rails5/action_cable.rb +0 -36
  458. data/lib/new_relic/agent/instrumentation/rails5/action_controller.rb +0 -33
  459. data/lib/new_relic/agent/instrumentation/rails5/action_view.rb +0 -27
  460. data/lib/new_relic/agent/instrumentation/rainbows_instrumentation.rb +0 -26
  461. data/lib/new_relic/agent/instrumentation/sunspot.rb +0 -33
  462. data/lib/new_relic/agent/supported_versions.rb +0 -275
  463. data/lib/new_relic/agent/transaction/attributes.rb +0 -154
  464. data/lib/new_relic/agent/transaction/xray_sample_buffer.rb +0 -64
  465. data/lib/new_relic/agent/transaction_state.rb +0 -186
  466. data/lib/new_relic/build.rb +0 -2
  467. data/lib/new_relic/control/frameworks/merb.rb +0 -29
  468. data/lib/new_relic/control/frameworks/rails5.rb +0 -14
  469. data/lib/new_relic/metrics.rb +0 -13
  470. data/lib/tasks/config.html.erb +0 -32
  471. data/lib/tasks/versions.html.erb +0 -28
  472. data/lib/tasks/versions.postface.html +0 -8
  473. data/lib/tasks/versions.preface.html +0 -9
  474. data/lib/tasks/versions.rake +0 -65
  475. data/lib/tasks/versions.txt.erb +0 -14
  476. /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/rpm/blob/master/LICENSE for complete details.
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( array=[] )
17
+ def initialize(array = [])
12
18
  @array = array
13
19
  end
14
20
  attr_reader :array
15
21
 
16
- def write( message )
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 assert_between(floor, ceiling, value, message="expected #{floor} <= #{value} <= #{ceiling}")
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
- assert \
42
- errors.find {|e| e.exception_class_name == error_class.name} != nil, \
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?( assert_block )
67
- def assert_block(*msgs)
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
- assert_block( msg ) { collection.include?(member) }
87
+
88
+ assert_includes collection, member, msg
76
89
  end
77
90
  end
78
91
 
79
- unless defined?( assert_not_includes )
80
- def assert_not_includes( collection, member, msg=nil )
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
- assert !collection.include?(member), msg
95
+
96
+ refute_includes collection, member, msg
83
97
  end
84
98
  end
85
99
 
86
- unless defined?( assert_empty )
87
- def assert_empty(collection, msg=nil)
88
- assert collection.empty?, msg
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
- assert(haystack.include?(needle), "Expected log to contain '#{needle}'")
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 then NewRelic::MetricSpec.new(*specish)
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
- { expectations => {} }
178
+ {expectations => {}}
155
179
  else
156
180
  expectations
157
181
  end
158
182
  end
159
183
 
160
184
  def dump_stats(stats)
161
- str = " Call count: #{stats.call_count}\n"
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
- assert_equal(expected_value, actual_value,
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
- # whitelists broad swathes of metric territory (e.g. 'Supportability/').
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 = expected.keys.map { |s| metric_spec_from_specish(s) }
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 =~ /GC\/Transaction/ }
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
- assert_equal([], found_but_not_expected, "Found unexpected metrics: #{format_metric_spec_list(found_but_not_expected)}")
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 regex
323
+ matching_metrics << metric if metric.match(regex)
273
324
  end
274
325
 
275
- assert_equal(
276
- [],
326
+ assert_empty(
277
327
  matching_metrics,
278
- "Found unexpected metrics:\n" + matching_metrics.map { |m| " '#{m}'"}.join("\n") + "\n\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
- assert !!expected, msg
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
- assert !expected, msg
349
+
350
+ refute expected, msg
299
351
  end
300
352
 
301
- unless defined?( assert_false )
353
+ unless defined? assert_false
302
354
  def assert_false(expected)
303
- assert_equal false, expected
355
+ refute expected
304
356
  end
305
357
  end
306
358
 
307
- unless defined?(refute)
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 = (args.last && args.last.is_a?(Hash)) ? args.pop : {}
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::TransactionState.tl_get
388
+ state = NewRelic::Agent::Tracer.state
338
389
  txn = nil
339
390
 
340
- NewRelic::Agent::Transaction.wrap(state, name, category, opts) do
391
+ NewRelic::Agent::Tracer.in_transaction(name: name, category: category, options: opts) do
341
392
  txn = state.current_transaction
342
- yield state.current_transaction
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 txn
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 txn
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 regex
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 regex
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
- def with_constant_defined(constant_symbol, implementation=Module.new)
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.const_defined?(part)
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
- const_path = constant_path(constant_symbol.to_s)
534
- return yield unless const_path
535
- parent = const_path[-2]
536
- const_name = constant_symbol.to_s.gsub(/.*::/, '')
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, { "name" => "command_name", "arguments" => args}])
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 = Time.now + opts[:timeout]
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 Time.now > deadline
714
+ if Process.clock_gettime(Process::CLOCK_REALTIME) > deadline
576
715
  raise "Timed out waiting #{opts[:timeout]} s for backtrace service poll\n" +
577
- "Worker loop ran for #{opts[:service].worker_loop.iterations} iterations\n\n" +
578
- Thread.list.map { |t|
579
- "#{t.to_s}: newrelic_label: #{t[:newrelic_label].inspect}\n\n" +
580
- (t.backtrace || []).join("\n\t")
581
- }.join("\n\n")
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=:info)
725
+ def with_array_logger(level = :info)
587
726
  orig_logger = NewRelic::Agent.logger
588
- config = { :log_level => level }
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
- def with_environment(env)
603
- old_env = {}
604
- env.each do |key, val|
605
- old_env[key] = ENV[key]
606
- ENV[key] = val.to_s
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
- begin
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
- ensure
611
- old_env.each { |key, old_val| ENV[key] = old_val }
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(::JSON.dump(object))
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 = ::JSON.load(data)
665
- data.each { |testcase| testcase['testname'].gsub! ' ', '_' if String === 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 [cross_agent_tests_dir, options[:dir], options[:pattern]].compact
672
- Dir.glob(path).each { |file| yield file}
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
- assert(incorrect_attributes.empty?, msg)
905
+
906
+ assert_empty(incorrect_attributes, msg)
697
907
 
698
908
  non_expected_attributes.each do |name|
699
- assert_nil(event_attrs[name], "Found value '#{event_attrs[name]}' for attribute '#{name}', but expected nothing in #{test_name}")
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