sentry-raven 3.0.0 → 3.1.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (192) hide show
  1. checksums.yaml +4 -4
  2. data/.craft.yml +15 -0
  3. data/.github/ISSUE_TEMPLATE/bug_report.md +32 -0
  4. data/.github/pull_request_template.md +16 -0
  5. data/.github/workflows/test.yml +92 -0
  6. data/.github/workflows/zeus_upload.yml +32 -0
  7. data/.gitignore +3 -0
  8. data/.rubocop.yml +50 -12
  9. data/.scripts/bump-version.sh +9 -0
  10. data/{changelog.md → CHANGELOG.md} +163 -1
  11. data/CONTRIBUTING.md +71 -0
  12. data/Gemfile +20 -25
  13. data/README.md +32 -16
  14. data/lib/raven/backtrace.rb +9 -5
  15. data/lib/raven/base.rb +6 -2
  16. data/lib/raven/breadcrumbs.rb +1 -1
  17. data/lib/raven/breadcrumbs/{activesupport.rb → active_support_logger.rb} +9 -3
  18. data/lib/raven/breadcrumbs/logger.rb +2 -92
  19. data/lib/raven/breadcrumbs/sentry_logger.rb +73 -0
  20. data/lib/raven/cli.rb +10 -21
  21. data/lib/raven/client.rb +9 -4
  22. data/lib/raven/configuration.rb +96 -10
  23. data/lib/raven/context.rb +13 -8
  24. data/lib/raven/core_ext/object/deep_dup.rb +57 -0
  25. data/lib/raven/core_ext/object/duplicable.rb +153 -0
  26. data/lib/raven/event.rb +27 -15
  27. data/lib/raven/helpers/deprecation_helper.rb +17 -0
  28. data/lib/raven/instance.rb +14 -4
  29. data/lib/raven/integrations/delayed_job.rb +15 -15
  30. data/lib/raven/integrations/rack-timeout.rb +7 -4
  31. data/lib/raven/integrations/rack.rb +18 -4
  32. data/lib/raven/integrations/rails.rb +13 -3
  33. data/lib/raven/integrations/rails/active_job.rb +6 -4
  34. data/lib/raven/integrations/rails/backtrace_cleaner.rb +29 -0
  35. data/lib/raven/integrations/rails/overrides/debug_exceptions_catcher.rb +2 -2
  36. data/lib/raven/integrations/sidekiq.rb +4 -78
  37. data/lib/raven/integrations/sidekiq/cleanup_middleware.rb +13 -0
  38. data/lib/raven/integrations/sidekiq/error_handler.rb +38 -0
  39. data/lib/raven/interface.rb +2 -2
  40. data/lib/raven/interfaces/stack_trace.rb +1 -1
  41. data/lib/raven/linecache.rb +5 -2
  42. data/lib/raven/logger.rb +3 -2
  43. data/lib/raven/processor/cookies.rb +16 -6
  44. data/lib/raven/processor/post_data.rb +2 -0
  45. data/lib/raven/processor/removecircularreferences.rb +3 -1
  46. data/lib/raven/processor/sanitizedata.rb +65 -17
  47. data/lib/raven/processor/utf8conversion.rb +2 -0
  48. data/lib/raven/transports.rb +4 -0
  49. data/lib/raven/transports/http.rb +5 -7
  50. data/lib/raven/utils/context_filter.rb +42 -0
  51. data/lib/raven/utils/exception_cause_chain.rb +1 -0
  52. data/lib/raven/utils/real_ip.rb +1 -1
  53. data/lib/raven/version.rb +2 -2
  54. data/lib/sentry-raven-without-integrations.rb +6 -1
  55. data/lib/sentry_raven_without_integrations.rb +1 -0
  56. data/sentry-ruby/.gitignore +11 -0
  57. data/sentry-ruby/.rspec +3 -0
  58. data/sentry-ruby/.travis.yml +6 -0
  59. data/sentry-ruby/CODE_OF_CONDUCT.md +74 -0
  60. data/sentry-ruby/Gemfile +9 -0
  61. data/sentry-ruby/LICENSE.txt +21 -0
  62. data/sentry-ruby/README.md +44 -0
  63. data/sentry-ruby/Rakefile +6 -0
  64. data/sentry-ruby/bin/console +14 -0
  65. data/sentry-ruby/bin/setup +8 -0
  66. data/sentry-ruby/examples/rails-6.0/.browserslistrc +1 -0
  67. data/sentry-ruby/examples/rails-6.0/.gitignore +35 -0
  68. data/sentry-ruby/examples/rails-6.0/Gemfile +58 -0
  69. data/sentry-ruby/examples/rails-6.0/README.md +23 -0
  70. data/sentry-ruby/examples/rails-6.0/Rakefile +6 -0
  71. data/sentry-ruby/examples/rails-6.0/app/assets/config/manifest.js +2 -0
  72. data/sentry-ruby/examples/rails-6.0/app/assets/images/.keep +0 -0
  73. data/sentry-ruby/examples/rails-6.0/app/assets/stylesheets/application.css +15 -0
  74. data/sentry-ruby/examples/rails-6.0/app/channels/application_cable/channel.rb +4 -0
  75. data/sentry-ruby/examples/rails-6.0/app/channels/application_cable/connection.rb +4 -0
  76. data/sentry-ruby/examples/rails-6.0/app/controllers/application_controller.rb +2 -0
  77. data/sentry-ruby/examples/rails-6.0/app/controllers/concerns/.keep +0 -0
  78. data/sentry-ruby/examples/rails-6.0/app/controllers/welcome_controller.rb +23 -0
  79. data/sentry-ruby/examples/rails-6.0/app/helpers/application_helper.rb +2 -0
  80. data/sentry-ruby/examples/rails-6.0/app/javascript/channels/consumer.js +6 -0
  81. data/sentry-ruby/examples/rails-6.0/app/javascript/channels/index.js +5 -0
  82. data/sentry-ruby/examples/rails-6.0/app/javascript/packs/application.js +17 -0
  83. data/sentry-ruby/examples/rails-6.0/app/jobs/application_job.rb +7 -0
  84. data/sentry-ruby/examples/rails-6.0/app/mailers/application_mailer.rb +4 -0
  85. data/sentry-ruby/examples/rails-6.0/app/models/application_record.rb +3 -0
  86. data/sentry-ruby/examples/rails-6.0/app/models/concerns/.keep +0 -0
  87. data/sentry-ruby/examples/rails-6.0/app/views/layouts/application.html.erb +15 -0
  88. data/sentry-ruby/examples/rails-6.0/app/views/layouts/mailer.html.erb +13 -0
  89. data/sentry-ruby/examples/rails-6.0/app/views/layouts/mailer.text.erb +1 -0
  90. data/sentry-ruby/examples/rails-6.0/app/views/welcome/report_demo.html.erb +22 -0
  91. data/sentry-ruby/examples/rails-6.0/app/views/welcome/view_error.html.erb +1 -0
  92. data/sentry-ruby/examples/rails-6.0/app/workers/error_worker.rb +7 -0
  93. data/sentry-ruby/examples/rails-6.0/babel.config.js +72 -0
  94. data/sentry-ruby/examples/rails-6.0/bin/bundle +114 -0
  95. data/sentry-ruby/examples/rails-6.0/bin/rails +9 -0
  96. data/sentry-ruby/examples/rails-6.0/bin/rake +9 -0
  97. data/sentry-ruby/examples/rails-6.0/bin/setup +36 -0
  98. data/sentry-ruby/examples/rails-6.0/bin/spring +17 -0
  99. data/sentry-ruby/examples/rails-6.0/bin/webpack +18 -0
  100. data/sentry-ruby/examples/rails-6.0/bin/webpack-dev-server +18 -0
  101. data/sentry-ruby/examples/rails-6.0/bin/yarn +11 -0
  102. data/sentry-ruby/examples/rails-6.0/config.ru +5 -0
  103. data/sentry-ruby/examples/rails-6.0/config/application.rb +28 -0
  104. data/sentry-ruby/examples/rails-6.0/config/boot.rb +4 -0
  105. data/sentry-ruby/examples/rails-6.0/config/cable.yml +10 -0
  106. data/sentry-ruby/examples/rails-6.0/config/credentials.yml.enc +1 -0
  107. data/sentry-ruby/examples/rails-6.0/config/database.yml +25 -0
  108. data/sentry-ruby/examples/rails-6.0/config/environment.rb +5 -0
  109. data/sentry-ruby/examples/rails-6.0/config/environments/development.rb +62 -0
  110. data/sentry-ruby/examples/rails-6.0/config/environments/production.rb +112 -0
  111. data/sentry-ruby/examples/rails-6.0/config/environments/test.rb +48 -0
  112. data/sentry-ruby/examples/rails-6.0/config/initializers/application_controller_renderer.rb +8 -0
  113. data/sentry-ruby/examples/rails-6.0/config/initializers/assets.rb +14 -0
  114. data/sentry-ruby/examples/rails-6.0/config/initializers/backtrace_silencers.rb +7 -0
  115. data/sentry-ruby/examples/rails-6.0/config/initializers/content_security_policy.rb +30 -0
  116. data/sentry-ruby/examples/rails-6.0/config/initializers/cookies_serializer.rb +5 -0
  117. data/sentry-ruby/examples/rails-6.0/config/initializers/filter_parameter_logging.rb +4 -0
  118. data/sentry-ruby/examples/rails-6.0/config/initializers/inflections.rb +16 -0
  119. data/sentry-ruby/examples/rails-6.0/config/initializers/mime_types.rb +4 -0
  120. data/sentry-ruby/examples/rails-6.0/config/initializers/wrap_parameters.rb +14 -0
  121. data/sentry-ruby/examples/rails-6.0/config/locales/en.yml +33 -0
  122. data/sentry-ruby/examples/rails-6.0/config/puma.rb +38 -0
  123. data/sentry-ruby/examples/rails-6.0/config/routes.rb +10 -0
  124. data/sentry-ruby/examples/rails-6.0/config/spring.rb +6 -0
  125. data/sentry-ruby/examples/rails-6.0/config/storage.yml +34 -0
  126. data/sentry-ruby/examples/rails-6.0/config/webpack/development.js +5 -0
  127. data/sentry-ruby/examples/rails-6.0/config/webpack/environment.js +3 -0
  128. data/sentry-ruby/examples/rails-6.0/config/webpack/production.js +5 -0
  129. data/sentry-ruby/examples/rails-6.0/config/webpack/test.js +5 -0
  130. data/sentry-ruby/examples/rails-6.0/config/webpacker.yml +96 -0
  131. data/sentry-ruby/examples/rails-6.0/db/seeds.rb +7 -0
  132. data/sentry-ruby/examples/rails-6.0/lib/assets/.keep +0 -0
  133. data/sentry-ruby/examples/rails-6.0/lib/tasks/.keep +0 -0
  134. data/sentry-ruby/examples/rails-6.0/package.json +15 -0
  135. data/sentry-ruby/examples/rails-6.0/postcss.config.js +12 -0
  136. data/sentry-ruby/examples/rails-6.0/public/404.html +67 -0
  137. data/sentry-ruby/examples/rails-6.0/public/422.html +67 -0
  138. data/sentry-ruby/examples/rails-6.0/public/500.html +66 -0
  139. data/sentry-ruby/examples/rails-6.0/public/apple-touch-icon-precomposed.png +0 -0
  140. data/sentry-ruby/examples/rails-6.0/public/apple-touch-icon.png +0 -0
  141. data/sentry-ruby/examples/rails-6.0/public/favicon.ico +0 -0
  142. data/sentry-ruby/examples/rails-6.0/public/robots.txt +1 -0
  143. data/sentry-ruby/examples/rails-6.0/storage/.keep +0 -0
  144. data/sentry-ruby/examples/rails-6.0/test/application_system_test_case.rb +5 -0
  145. data/sentry-ruby/examples/rails-6.0/test/channels/application_cable/connection_test.rb +11 -0
  146. data/sentry-ruby/examples/rails-6.0/test/controllers/.keep +0 -0
  147. data/sentry-ruby/examples/rails-6.0/test/fixtures/.keep +0 -0
  148. data/sentry-ruby/examples/rails-6.0/test/fixtures/files/.keep +0 -0
  149. data/sentry-ruby/examples/rails-6.0/test/helpers/.keep +0 -0
  150. data/sentry-ruby/examples/rails-6.0/test/integration/.keep +0 -0
  151. data/sentry-ruby/examples/rails-6.0/test/mailers/.keep +0 -0
  152. data/sentry-ruby/examples/rails-6.0/test/models/.keep +0 -0
  153. data/sentry-ruby/examples/rails-6.0/test/system/.keep +0 -0
  154. data/sentry-ruby/examples/rails-6.0/test/test_helper.rb +13 -0
  155. data/sentry-ruby/examples/rails-6.0/vendor/.keep +0 -0
  156. data/sentry-ruby/examples/rails-6.0/yarn.lock +7508 -0
  157. data/sentry-ruby/lib/sentry.rb +16 -0
  158. data/sentry-ruby/lib/sentry/backtrace.rb +128 -0
  159. data/sentry-ruby/lib/sentry/client.rb +162 -0
  160. data/sentry-ruby/lib/sentry/client/state.rb +40 -0
  161. data/sentry-ruby/lib/sentry/configuration.rb +533 -0
  162. data/sentry-ruby/lib/sentry/event.rb +209 -0
  163. data/sentry-ruby/lib/sentry/interface.rb +31 -0
  164. data/sentry-ruby/lib/sentry/interfaces/exception.rb +15 -0
  165. data/sentry-ruby/lib/sentry/interfaces/http.rb +16 -0
  166. data/sentry-ruby/lib/sentry/interfaces/message.rb +18 -0
  167. data/sentry-ruby/lib/sentry/interfaces/single_exception.rb +14 -0
  168. data/sentry-ruby/lib/sentry/interfaces/stack_trace.rb +69 -0
  169. data/sentry-ruby/lib/sentry/linecache.rb +44 -0
  170. data/sentry-ruby/lib/sentry/logger.rb +20 -0
  171. data/sentry-ruby/lib/sentry/transports.rb +19 -0
  172. data/sentry-ruby/lib/sentry/transports/dummy.rb +16 -0
  173. data/sentry-ruby/lib/sentry/transports/http.rb +66 -0
  174. data/sentry-ruby/lib/sentry/transports/stdout.rb +20 -0
  175. data/sentry-ruby/lib/sentry/utils/deep_merge.rb +22 -0
  176. data/sentry-ruby/lib/sentry/utils/exception_cause_chain.rb +20 -0
  177. data/sentry-ruby/lib/sentry/version.rb +3 -0
  178. data/sentry-ruby/sentry-ruby.gemspec +26 -0
  179. data/sentry-ruby/spec/sentry/backtrace_spec.rb +38 -0
  180. data/sentry-ruby/spec/sentry/client_spec.rb +443 -0
  181. data/sentry-ruby/spec/sentry/configuration_spec.rb +400 -0
  182. data/sentry-ruby/spec/sentry/event_spec.rb +238 -0
  183. data/sentry-ruby/spec/sentry/interface_spec.rb +38 -0
  184. data/sentry-ruby/spec/sentry/interfaces/stack_trace_spec.rb +11 -0
  185. data/sentry-ruby/spec/sentry/linecache_spec.rb +40 -0
  186. data/sentry-ruby/spec/sentry/transports/http_spec.rb +57 -0
  187. data/sentry-ruby/spec/sentry/transports/stdout_spec.rb +11 -0
  188. data/sentry-ruby/spec/sentry_spec.rb +9 -0
  189. data/sentry-ruby/spec/spec_helper.rb +49 -0
  190. data/sentry-ruby/spec/support/linecache.txt +6 -0
  191. metadata +155 -5
  192. data/.travis.yml +0 -43
@@ -10,10 +10,13 @@ module RackTimeoutExtensions
10
10
  # Only rack-timeout 0.3.0+ provides the request environment, but we can't
11
11
  # gate this based on a gem version constant because rack-timeout does
12
12
  # not provide one.
13
- { :fingerprint => ["{{ default }}", env["REQUEST_URI"]] } if defined?(env)
13
+ if defined?(env)
14
+ { :fingerprint => ["{{ default }}", env["REQUEST_URI"]] }
15
+ else
16
+ {}
17
+ end
14
18
  end
15
19
  end
16
20
 
17
- # Include is private in Ruby 1.9
18
- Rack::Timeout::Error.__send__(:include, RackTimeoutExtensions)
19
- Rack::Timeout::RequestTimeoutException.__send__(:include, RackTimeoutExtensions)
21
+ Rack::Timeout::Error.include(RackTimeoutExtensions)
22
+ Rack::Timeout::RequestTimeoutException.include(RackTimeoutExtensions)
@@ -67,6 +67,8 @@ module Raven
67
67
  end
68
68
 
69
69
  module RackInterface
70
+ REQUEST_ID_HEADERS = %w(action_dispatch.request_id HTTP_X_REQUEST_ID).freeze
71
+
70
72
  def from_rack(env_hash)
71
73
  req = ::Rack::Request.new(env_hash)
72
74
 
@@ -82,6 +84,15 @@ module Raven
82
84
 
83
85
  private
84
86
 
87
+ # Request ID based on ActionDispatch::RequestId
88
+ def read_request_id_from(env_hash)
89
+ REQUEST_ID_HEADERS.each do |key|
90
+ request_id = env_hash[key]
91
+ return request_id if request_id
92
+ end
93
+ nil
94
+ end
95
+
85
96
  # See Sentry server default limits at
86
97
  # https://github.com/getsentry/sentry/blob/master/src/sentry/conf/server.py
87
98
  def read_data_from(request)
@@ -92,8 +103,8 @@ module Raven
92
103
  request.body.rewind
93
104
  data
94
105
  end
95
- rescue IOError => ex
96
- ex.message
106
+ rescue IOError => e
107
+ e.message
97
108
  end
98
109
 
99
110
  def format_headers_for_sentry(env_hash)
@@ -101,6 +112,7 @@ module Raven
101
112
  begin
102
113
  key = key.to_s # rack env can contain symbols
103
114
  value = value.to_s
115
+ next memo['X-Request-Id'] ||= read_request_id_from(env_hash) if REQUEST_ID_HEADERS.include?(key)
104
116
  next unless key.upcase == key # Non-upper case stuff isn't either
105
117
 
106
118
  # Rack adds in an incorrect HTTP_VERSION key, which causes downstream
@@ -110,8 +122,8 @@ module Raven
110
122
  # See: https://github.com/rack/rack/blob/028438f/lib/rack/handler/cgi.rb#L29
111
123
  next if key == 'HTTP_VERSION' && value == env_hash['SERVER_PROTOCOL']
112
124
  next if key == 'HTTP_COOKIE' # Cookies don't go here, they go somewhere else
113
-
114
125
  next unless key.start_with?('HTTP_') || %w(CONTENT_TYPE CONTENT_LENGTH).include?(key)
126
+
115
127
  # Rack stores headers as HTTP_WHAT_EVER, we need What-Ever
116
128
  key = key.sub(/^HTTP_/, "")
117
129
  key = key.split('_').map(&:capitalize).join('-')
@@ -127,8 +139,10 @@ module Raven
127
139
  end
128
140
 
129
141
  def format_env_for_sentry(env_hash)
142
+ return env_hash if Raven.configuration.rack_env_whitelist.empty?
143
+
130
144
  env_hash.select do |k, _v|
131
- %w(REMOTE_ADDR SERVER_NAME SERVER_PORT).include? k.to_s
145
+ Raven.configuration.rack_env_whitelist.include? k.to_s
132
146
  end
133
147
  end
134
148
  end
@@ -5,6 +5,8 @@ module Raven
5
5
  require 'raven/integrations/rails/overrides/streaming_reporter'
6
6
  require 'raven/integrations/rails/controller_methods'
7
7
  require 'raven/integrations/rails/controller_transaction'
8
+ require 'raven/integrations/rails/backtrace_cleaner'
9
+ require 'raven/integrations/rack'
8
10
 
9
11
  initializer "raven.use_rack_middleware" do |app|
10
12
  app.config.middleware.insert 0, Raven::Rack
@@ -36,12 +38,20 @@ module Raven
36
38
 
37
39
  config.before_initialize do
38
40
  Raven.configuration.logger = ::Rails.logger
41
+
42
+ backtrace_cleaner = Raven::Rails::BacktraceCleaner.new
43
+
44
+ Raven.configuration.backtrace_cleanup_callback = lambda do |backtrace|
45
+ backtrace_cleaner.clean(backtrace)
46
+ end
39
47
  end
40
48
 
41
49
  config.after_initialize do
42
- if Raven.configuration.rails_activesupport_breadcrumbs
43
- require 'raven/breadcrumbs/activesupport'
44
- Raven::ActiveSupportBreadcrumbs.inject
50
+ if Raven.configuration.breadcrumbs_logger.include?(:active_support_logger) ||
51
+ Raven.configuration.rails_activesupport_breadcrumbs
52
+
53
+ require 'raven/breadcrumbs/active_support_logger'
54
+ Raven::Breadcrumbs::ActiveSupportLogger.inject
45
55
  end
46
56
 
47
57
  if Raven.configuration.rails_report_rescued_exceptions
@@ -20,10 +20,12 @@ module Raven
20
20
 
21
21
  def capture_and_reraise_with_sentry(job, block)
22
22
  block.call
23
- rescue Exception => exception # rubocop:disable Lint/RescueException
24
- return if rescue_with_handler(exception)
25
- Raven.capture_exception(exception, :extra => raven_context(job))
26
- raise exception
23
+ rescue Exception => e # rubocop:disable Lint/RescueException
24
+ rescue_handler_result = rescue_with_handler(e)
25
+ return rescue_handler_result if rescue_handler_result
26
+
27
+ Raven.capture_exception(e, :extra => raven_context(job))
28
+ raise e
27
29
  ensure
28
30
  Context.clear!
29
31
  BreadcrumbBuffer.clear!
@@ -0,0 +1,29 @@
1
+ require "active_support/backtrace_cleaner"
2
+ require "active_support/core_ext/string/access"
3
+
4
+ module Raven
5
+ class Rails
6
+ class BacktraceCleaner < ActiveSupport::BacktraceCleaner
7
+ APP_DIRS_PATTERN = /\A(?:\.\/)?(?:app|config|lib|test|\(\w*\))/.freeze
8
+ RENDER_TEMPLATE_PATTERN = /:in `.*_\w+_{2,3}\d+_\d+'/.freeze
9
+
10
+ def initialize
11
+ super
12
+ # we don't want any default silencers because they're too aggressive
13
+ remove_silencers!
14
+
15
+ @root = "#{Raven.configuration.project_root}/"
16
+ add_filter do |line|
17
+ line.start_with?(@root) ? line.from(@root.size) : line
18
+ end
19
+ add_filter do |line|
20
+ if line =~ RENDER_TEMPLATE_PATTERN
21
+ line.sub(RENDER_TEMPLATE_PATTERN, "")
22
+ else
23
+ line
24
+ end
25
+ end
26
+ end
27
+ end
28
+ end
29
+ end
@@ -6,7 +6,7 @@ module Raven
6
6
  begin
7
7
  env = env_or_request.respond_to?(:env) ? env_or_request.env : env_or_request
8
8
  Raven::Rack.capture_exception(exception, env)
9
- rescue # rubocop:disable Lint/HandleExceptions
9
+ rescue
10
10
  end
11
11
  super
12
12
  end
@@ -21,7 +21,7 @@ module Raven
21
21
  begin
22
22
  env = env_or_request.respond_to?(:env) ? env_or_request.env : env_or_request
23
23
  Raven::Rack.capture_exception(exception, env)
24
- rescue # rubocop:disable Lint/HandleExceptions
24
+ rescue
25
25
  end
26
26
  render_exception_without_raven(env_or_request, exception)
27
27
  end
@@ -1,87 +1,13 @@
1
1
  require 'time'
2
2
  require 'sidekiq'
3
-
4
- module Raven
5
- class SidekiqCleanupMiddleware
6
- def call(_worker, job, queue)
7
- Raven.context.transaction.push "Sidekiq/#{job['class']}"
8
- Raven.extra_context(:sidekiq => job.merge("queue" => queue))
9
- yield
10
- Context.clear!
11
- BreadcrumbBuffer.clear!
12
- end
13
- end
14
-
15
- class SidekiqErrorHandler
16
- ACTIVEJOB_RESERVED_PREFIX = "_aj_".freeze
17
- HAS_GLOBALID = const_defined?('GlobalID')
18
-
19
- def call(ex, context)
20
- context = filter_context(context)
21
- Raven.context.transaction.push transaction_from_context(context)
22
- Raven.capture_exception(
23
- ex,
24
- :message => ex.message,
25
- :extra => { :sidekiq => context }
26
- )
27
- Context.clear!
28
- BreadcrumbBuffer.clear!
29
- end
30
-
31
- private
32
-
33
- # Once an ActiveJob is queued, ActiveRecord references get serialized into
34
- # some internal reserved keys, such as _aj_globalid.
35
- #
36
- # The problem is, if this job in turn gets queued back into ActiveJob with
37
- # these magic reserved keys, ActiveJob will throw up and error. We want to
38
- # capture these and mutate the keys so we can sanely report it.
39
- def filter_context(context)
40
- case context
41
- when Array
42
- context.map { |arg| filter_context(arg) }
43
- when Hash
44
- Hash[context.map { |key, value| filter_context_hash(key, value) }]
45
- else
46
- format_globalid(context)
47
- end
48
- end
49
-
50
- def filter_context_hash(key, value)
51
- (key = key[3..-1]) if key [0..3] == ACTIVEJOB_RESERVED_PREFIX
52
- [key, filter_context(value)]
53
- end
54
-
55
- # this will change in the future:
56
- # https://github.com/mperham/sidekiq/pull/3161
57
- def transaction_from_context(context)
58
- classname = (context["wrapped"] || context["class"] ||
59
- (context[:job] && (context[:job]["wrapped"] || context[:job]["class"]))
60
- )
61
- if classname
62
- "Sidekiq/#{classname}"
63
- elsif context[:event]
64
- "Sidekiq/#{context[:event]}"
65
- else
66
- "Sidekiq"
67
- end
68
- end
69
-
70
- def format_globalid(context)
71
- if HAS_GLOBALID && context.is_a?(GlobalID)
72
- context.to_s
73
- else
74
- context
75
- end
76
- end
77
- end
78
- end
3
+ require 'raven/integrations/sidekiq/cleanup_middleware'
4
+ require 'raven/integrations/sidekiq/error_handler'
79
5
 
80
6
  if Sidekiq::VERSION > '3'
81
7
  Sidekiq.configure_server do |config|
82
- config.error_handlers << Raven::SidekiqErrorHandler.new
8
+ config.error_handlers << Raven::Sidekiq::ErrorHandler.new
83
9
  config.server_middleware do |chain|
84
- chain.add Raven::SidekiqCleanupMiddleware
10
+ chain.add Raven::Sidekiq::CleanupMiddleware
85
11
  end
86
12
  end
87
13
  end
@@ -0,0 +1,13 @@
1
+ module Raven
2
+ module Sidekiq
3
+ class CleanupMiddleware
4
+ def call(_worker, job, queue)
5
+ Raven.context.transaction.push "Sidekiq/#{job['class']}"
6
+ Raven.extra_context(:sidekiq => job.merge("queue" => queue))
7
+ yield
8
+ Context.clear!
9
+ BreadcrumbBuffer.clear!
10
+ end
11
+ end
12
+ end
13
+ end
@@ -0,0 +1,38 @@
1
+ require 'raven/utils/context_filter'
2
+
3
+ module Raven
4
+ module Sidekiq
5
+ class ErrorHandler
6
+ SIDEKIQ_NAME = "Sidekiq".freeze
7
+
8
+ def call(ex, context)
9
+ context = Utils::ContextFilter.filter_context(context)
10
+ Raven.context.transaction.push transaction_from_context(context)
11
+ Raven.capture_exception(
12
+ ex,
13
+ :message => ex.message,
14
+ :extra => { :sidekiq => context }
15
+ )
16
+ Context.clear!
17
+ BreadcrumbBuffer.clear!
18
+ end
19
+
20
+ private
21
+
22
+ # this will change in the future:
23
+ # https://github.com/mperham/sidekiq/pull/3161
24
+ def transaction_from_context(context)
25
+ classname = (context["wrapped"] || context["class"] ||
26
+ (context[:job] && (context[:job]["wrapped"] || context[:job]["class"]))
27
+ )
28
+ if classname
29
+ "#{SIDEKIQ_NAME}/#{classname}"
30
+ elsif context[:event]
31
+ "#{SIDEKIQ_NAME}/#{context[:event]}"
32
+ else
33
+ SIDEKIQ_NAME
34
+ end
35
+ end
36
+ end
37
+ end
38
+ end
@@ -1,9 +1,9 @@
1
1
  module Raven
2
2
  class Interface
3
3
  def initialize(attributes = nil)
4
- attributes.each do |attr, value|
4
+ attributes&.each do |attr, value|
5
5
  public_send "#{attr}=", value
6
- end if attributes
6
+ end
7
7
 
8
8
  yield self if block_given?
9
9
  end
@@ -58,7 +58,7 @@ module Raven
58
58
  end
59
59
 
60
60
  def project_root
61
- @project_root ||= Raven.configuration.project_root && Raven.configuration.project_root.to_s
61
+ @project_root ||= Raven.configuration.project_root&.to_s
62
62
  end
63
63
 
64
64
  def longest_load_path
@@ -10,6 +10,7 @@ module Raven
10
10
  # line should be the line requested by lineno. See specs for more information.
11
11
  def get_file_context(filename, lineno, context)
12
12
  return nil, nil, nil unless valid_path?(filename)
13
+
13
14
  lines = Array.new(2 * context + 1) do |i|
14
15
  getline(filename, lineno - context + i)
15
16
  end
@@ -26,15 +27,17 @@ module Raven
26
27
  def getlines(path)
27
28
  @cache[path] ||= begin
28
29
  IO.readlines(path)
29
- rescue
30
- nil
30
+ rescue
31
+ nil
31
32
  end
32
33
  end
33
34
 
34
35
  def getline(path, n)
35
36
  return nil if n < 1
37
+
36
38
  lines = getlines(path)
37
39
  return nil if lines.nil?
40
+
38
41
  lines[n - 1]
39
42
  end
40
43
  end
@@ -1,10 +1,11 @@
1
1
  # frozen_string_literal: true
2
+
2
3
  require 'logger'
3
4
 
4
5
  module Raven
5
6
  class Logger < ::Logger
6
- LOG_PREFIX = "** [Raven] ".freeze
7
- PROGNAME = "sentry".freeze
7
+ LOG_PREFIX = "** [Raven] "
8
+ PROGNAME = "sentry"
8
9
 
9
10
  def initialize(*)
10
11
  super
@@ -10,17 +10,27 @@ module Raven
10
10
  private
11
11
 
12
12
  def process_if_symbol_keys(data)
13
- data[:request][:cookies] = STRING_MASK if data[:request][:cookies]
13
+ if cookies = data.dig(:request, :cookies)
14
+ data[:request][:cookies] = generate_masked_cookies(cookies)
15
+ end
14
16
 
15
- return unless data[:request][:headers] && data[:request][:headers]["Cookie"]
16
- data[:request][:headers]["Cookie"] = STRING_MASK
17
+ if cookies_header = data[:request][:headers]["Cookie"]
18
+ data[:request][:headers]["Cookie"] = generate_masked_cookies(cookies_header)
19
+ end
17
20
  end
18
21
 
19
22
  def process_if_string_keys(data)
20
- data["request"]["cookies"] = STRING_MASK if data["request"]["cookies"]
23
+ if cookies = data.dig("request", "cookies")
24
+ data["request"]["cookies"] = generate_masked_cookies(cookies)
25
+ end
21
26
 
22
- return unless data["request"]["headers"] && data["request"]["headers"]["Cookie"]
23
- data["request"]["headers"]["Cookie"] = STRING_MASK
27
+ if cookies_header = data.dig("request", "headers", "Cookie")
28
+ data["request"]["headers"]["Cookie"] = generate_masked_cookies(cookies_header)
29
+ end
30
+ end
31
+
32
+ def generate_masked_cookies(cookies)
33
+ cookies.merge(cookies) { STRING_MASK } if cookies.respond_to?(:merge)
24
34
  end
25
35
  end
26
36
  end
@@ -11,11 +11,13 @@ module Raven
11
11
 
12
12
  def process_if_symbol_keys(data)
13
13
  return unless data[:request][:method] == "POST"
14
+
14
15
  data[:request][:data] = STRING_MASK
15
16
  end
16
17
 
17
18
  def process_if_string_keys(data)
18
19
  return unless data["request"]["method"] == "POST"
20
+
19
21
  data["request"]["data"] = STRING_MASK
20
22
  end
21
23
  end
@@ -1,7 +1,9 @@
1
1
  module Raven
2
2
  class Processor::RemoveCircularReferences < Processor
3
+ ELISION_STRING = "(...)".freeze
3
4
  def process(value, visited = [])
4
- return "(...)" if visited.include?(value.__id__)
5
+ return ELISION_STRING if visited.include?(value.__id__)
6
+
5
7
  visited << value.__id__ if value.is_a?(Array) || value.is_a?(Hash)
6
8
 
7
9
  case value