Commit 6113a16f authored by Bob Van Landuyt's avatar Bob Van Landuyt

Merge branch 'improve-exception-handling' into 'master'

Improve exception handling

Closes #32906

See merge request gitlab-org/gitlab!17819
parents bc085f11 1ee162b6
...@@ -23,7 +23,7 @@ class ApplicationController < ActionController::Base ...@@ -23,7 +23,7 @@ class ApplicationController < ActionController::Base
before_action :validate_user_service_ticket! before_action :validate_user_service_ticket!
before_action :check_password_expiration, if: :html_request? before_action :check_password_expiration, if: :html_request?
before_action :ldap_security_check before_action :ldap_security_check
before_action :sentry_context around_action :sentry_context
before_action :default_headers before_action :default_headers
before_action :add_gon_variables, if: :html_request? before_action :add_gon_variables, if: :html_request?
before_action :configure_permitted_parameters, if: :devise_controller? before_action :configure_permitted_parameters, if: :devise_controller?
...@@ -165,7 +165,7 @@ class ApplicationController < ActionController::Base ...@@ -165,7 +165,7 @@ class ApplicationController < ActionController::Base
end end
def log_exception(exception) def log_exception(exception)
Gitlab::Sentry.track_acceptable_exception(exception) Gitlab::Sentry.track_exception(exception)
backtrace_cleaner = request.env["action_dispatch.backtrace_cleaner"] backtrace_cleaner = request.env["action_dispatch.backtrace_cleaner"]
application_trace = ActionDispatch::ExceptionWrapper.new(backtrace_cleaner, exception).application_trace application_trace = ActionDispatch::ExceptionWrapper.new(backtrace_cleaner, exception).application_trace
...@@ -532,8 +532,8 @@ class ApplicationController < ActionController::Base ...@@ -532,8 +532,8 @@ class ApplicationController < ActionController::Base
@impersonator ||= User.find(session[:impersonator_id]) if session[:impersonator_id] @impersonator ||= User.find(session[:impersonator_id]) if session[:impersonator_id]
end end
def sentry_context def sentry_context(&block)
Gitlab::Sentry.context(current_user) Gitlab::Sentry.with_context(current_user, &block)
end end
def allow_gitaly_ref_name_caching def allow_gitaly_ref_name_caching
......
...@@ -98,13 +98,11 @@ module IssuableActions ...@@ -98,13 +98,11 @@ module IssuableActions
error_message = "Destroy confirmation not provided for #{issuable.human_class_name}" error_message = "Destroy confirmation not provided for #{issuable.human_class_name}"
exception = RuntimeError.new(error_message) exception = RuntimeError.new(error_message)
Gitlab::Sentry.track_acceptable_exception( Gitlab::Sentry.track_exception(
exception, exception,
extra: {
project_path: issuable.project.full_path, project_path: issuable.project.full_path,
issuable_type: issuable.class.name, issuable_type: issuable.class.name,
issuable_id: issuable.id issuable_id: issuable.id
}
) )
index_path = polymorphic_path([parent, issuable.class]) index_path = polymorphic_path([parent, issuable.class])
......
...@@ -42,11 +42,9 @@ module IconsHelper ...@@ -42,11 +42,9 @@ module IconsHelper
end end
def sprite_icon(icon_name, size: nil, css_class: nil) def sprite_icon(icon_name, size: nil, css_class: nil)
if Gitlab::Sentry.should_raise_for_dev?
unless known_sprites.include?(icon_name) unless known_sprites.include?(icon_name)
exception = ArgumentError.new("#{icon_name} is not a known icon in @gitlab-org/gitlab-svg") exception = ArgumentError.new("#{icon_name} is not a known icon in @gitlab-org/gitlab-svg")
raise exception Gitlab::Sentry.track_and_raise_for_dev_exception(exception)
end
end end
css_classes = [] css_classes = []
......
...@@ -57,7 +57,7 @@ module UsersHelper ...@@ -57,7 +57,7 @@ module UsersHelper
unless user.association(:status).loaded? unless user.association(:status).loaded?
exception = RuntimeError.new("Status was not preloaded") exception = RuntimeError.new("Status was not preloaded")
Gitlab::Sentry.track_exception(exception, extra: { user: user.inspect }) Gitlab::Sentry.track_and_raise_for_dev_exception(exception, user: user.inspect)
end end
return unless user.status return unless user.status
......
...@@ -289,7 +289,7 @@ module Ci ...@@ -289,7 +289,7 @@ module Ci
begin begin
build.deployment.drop! build.deployment.drop!
rescue => e rescue => e
Gitlab::Sentry.track_exception(e, extra: { build_id: build.id }) Gitlab::Sentry.track_and_raise_for_dev_exception(e, build_id: build.id)
end end
true true
......
...@@ -27,7 +27,7 @@ module Ci ...@@ -27,7 +27,7 @@ module Ci
create_ref(sha, path) create_ref(sha, path)
rescue => e rescue => e
Gitlab::Sentry Gitlab::Sentry
.track_acceptable_exception(e, extra: { pipeline_id: pipeline.id }) .track_exception(e, pipeline_id: pipeline.id)
end end
def delete def delete
...@@ -38,7 +38,7 @@ module Ci ...@@ -38,7 +38,7 @@ module Ci
# no-op # no-op
rescue => e rescue => e
Gitlab::Sentry Gitlab::Sentry
.track_acceptable_exception(e, extra: { pipeline_id: pipeline.id }) .track_exception(e, pipeline_id: pipeline.id)
end end
def path def path
......
...@@ -71,6 +71,8 @@ module Clusters ...@@ -71,6 +71,8 @@ module Clusters
# `proxy_url` could raise an exception because gitlab can not communicate with the cluster. # `proxy_url` could raise an exception because gitlab can not communicate with the cluster.
# We check for a nil client in downstream use and behaviour is equivalent to an empty state # We check for a nil client in downstream use and behaviour is equivalent to an empty state
log_exception(error, :failed_to_create_elasticsearch_client) log_exception(error, :failed_to_create_elasticsearch_client)
nil
end end
end end
......
...@@ -335,7 +335,7 @@ module Clusters ...@@ -335,7 +335,7 @@ module Clusters
rescue Kubeclient::HttpError => e rescue Kubeclient::HttpError => e
kubeclient_error_status(e.message) kubeclient_error_status(e.message)
rescue => e rescue => e
Gitlab::Sentry.track_acceptable_exception(e, extra: { cluster_id: id }) Gitlab::Sentry.track_exception(e, cluster_id: id)
:unknown_failure :unknown_failure
else else
......
...@@ -76,7 +76,7 @@ module Clusters ...@@ -76,7 +76,7 @@ module Clusters
message: error.message message: error.message
}) })
Gitlab::Sentry.track_acceptable_exception(error, extra: { cluster_id: cluster&.id, application_id: id }) Gitlab::Sentry.track_exception(error, cluster_id: cluster&.id, application_id: id)
end end
end end
end end
......
...@@ -48,11 +48,11 @@ module GroupDescendant ...@@ -48,11 +48,11 @@ module GroupDescendant
extras = { extras = {
parent: parent.inspect, parent: parent.inspect,
child: child.inspect, child: child.inspect,
preloaded: preloaded.map(&:full_path) preloaded: preloaded.map(&:full_path),
issue_url: 'https://gitlab.com/gitlab-org/gitlab-foss/issues/49404'
} }
issue_url = 'https://gitlab.com/gitlab-org/gitlab-foss/issues/49404'
Gitlab::Sentry.track_exception(exception, issue_url: issue_url, extra: extras) Gitlab::Sentry.track_and_raise_for_dev_exception(exception, extras)
end end
if parent.nil? && hierarchy_top.present? if parent.nil? && hierarchy_top.present?
......
...@@ -37,8 +37,10 @@ module Storage ...@@ -37,8 +37,10 @@ module Storage
send_update_instructions send_update_instructions
write_projects_repository_config write_projects_repository_config
rescue => e rescue => e
# Raise if development/test environment, else just notify Sentry Gitlab::Sentry.track_and_raise_for_dev_exception(e,
Gitlab::Sentry.track_exception(e, extra: { full_path_before_last_save: full_path_before_last_save, full_path: full_path, action: 'move_dir' }) full_path_before_last_save: full_path_before_last_save,
full_path: full_path,
action: 'move_dir')
end end
true # false would cancel later callbacks but not rollback true # false would cancel later callbacks but not rollback
......
...@@ -1514,7 +1514,7 @@ class MergeRequest < ApplicationRecord ...@@ -1514,7 +1514,7 @@ class MergeRequest < ApplicationRecord
end end
end end
rescue ActiveRecord::LockWaitTimeout => e rescue ActiveRecord::LockWaitTimeout => e
Gitlab::Sentry.track_acceptable_exception(e) Gitlab::Sentry.track_exception(e)
raise RebaseLockTimeout, REBASE_LOCK_MESSAGE raise RebaseLockTimeout, REBASE_LOCK_MESSAGE
end end
......
...@@ -103,10 +103,8 @@ class Upload < ApplicationRecord ...@@ -103,10 +103,8 @@ class Upload < ApplicationRecord
# Help sysadmins find missing upload files # Help sysadmins find missing upload files
if persisted? && !exist if persisted? && !exist
if Gitlab::Sentry.enabled? exception = RuntimeError.new("Uploaded file does not exist")
Raven.capture_message(_("Upload file does not exist"), extra: self.attributes) Gitlab::Sentry.track_exception(exception, self.attributes)
end
Gitlab::Metrics.counter(:upload_file_does_not_exist_total, _('The number of times an upload record could not find its file')).increment Gitlab::Metrics.counter(:upload_file_does_not_exist_total, _('The number of times an upload record could not find its file')).increment
end end
......
...@@ -23,7 +23,8 @@ module Uploads ...@@ -23,7 +23,8 @@ module Uploads
unless in_uploads?(path) unless in_uploads?(path)
message = "Path '#{path}' is not in uploads dir, skipping" message = "Path '#{path}' is not in uploads dir, skipping"
logger.warn(message) logger.warn(message)
Gitlab::Sentry.track_exception(RuntimeError.new(message), extra: { uploads_dir: storage_dir }) Gitlab::Sentry.track_and_raise_for_dev_exception(
RuntimeError.new(message), uploads_dir: storage_dir)
return return
end end
......
...@@ -47,9 +47,9 @@ module Ci ...@@ -47,9 +47,9 @@ module Ci
job_id: job.id) job_id: job.id)
Gitlab::Sentry Gitlab::Sentry
.track_exception(error, .track_and_raise_for_dev_exception(error,
issue_url: 'https://gitlab.com/gitlab-org/gitlab-foss/issues/51502', issue_url: 'https://gitlab.com/gitlab-org/gitlab-foss/issues/51502',
extra: { job_id: job.id }) job_id: job.id )
end end
end end
end end
...@@ -15,7 +15,7 @@ module Ci ...@@ -15,7 +15,7 @@ module Ci
data: data data: data
} }
rescue => e rescue => e
Gitlab::Sentry.track_acceptable_exception(e, extra: { project_id: project.id }) Gitlab::Sentry.track_exception(e, project_id: project.id)
{ {
status: :error, status: :error,
key: key(base_pipeline, head_pipeline), key: key(base_pipeline, head_pipeline),
......
...@@ -13,7 +13,7 @@ module Ci ...@@ -13,7 +13,7 @@ module Ci
build.enqueue! build.enqueue!
rescue => e rescue => e
Gitlab::Sentry.track_acceptable_exception(e, extra: { build_id: build.id }) Gitlab::Sentry.track_exception(e, build_id: build.id)
build.drop(:unmet_prerequisites) build.drop(:unmet_prerequisites)
end end
......
...@@ -128,13 +128,13 @@ module Ci ...@@ -128,13 +128,13 @@ module Ci
end end
def track_exception_for_build(ex, build) def track_exception_for_build(ex, build)
Gitlab::Sentry.track_acceptable_exception(ex, extra: { Gitlab::Sentry.track_exception(ex,
build_id: build.id, build_id: build.id,
build_name: build.name, build_name: build.name,
build_stage: build.stage, build_stage: build.stage,
pipeline_id: build.pipeline_id, pipeline_id: build.pipeline_id,
project_id: build.project_id project_id: build.project_id
}) )
end end
# rubocop: disable CodeReuse/ActiveRecord # rubocop: disable CodeReuse/ActiveRecord
......
...@@ -21,14 +21,7 @@ module Clusters ...@@ -21,14 +21,7 @@ module Clusters
group_ids: app.cluster.group_ids group_ids: app.cluster.group_ids
} }
logger_meta = meta.merge( Gitlab::Sentry.track_exception(error, meta)
exception: error.class.name,
message: error.message,
backtrace: Gitlab::Profiler.clean_backtrace(error.backtrace)
)
logger.error(logger_meta)
Gitlab::Sentry.track_acceptable_exception(error, extra: meta)
end end
def log_event(event) def log_event(event)
......
...@@ -51,7 +51,7 @@ module Projects ...@@ -51,7 +51,7 @@ module Projects
digests = deleted_tags.values.uniq digests = deleted_tags.values.uniq
# rubocop: disable CodeReuse/ActiveRecord # rubocop: disable CodeReuse/ActiveRecord
Gitlab::Sentry.track_exception(ArgumentError.new('multiple tag digests')) if digests.many? Gitlab::Sentry.track_and_raise_for_dev_exception(ArgumentError.new('multiple tag digests')) if digests.many?
deleted_tags deleted_tags
end end
......
...@@ -25,13 +25,13 @@ module Projects ...@@ -25,13 +25,13 @@ module Projects
success success
rescue Gitlab::UrlBlocker::BlockedUrlError => e rescue Gitlab::UrlBlocker::BlockedUrlError => e
Gitlab::Sentry.track_acceptable_exception(e, extra: { project_path: project.full_path, importer: project.import_type }) Gitlab::Sentry.track_exception(e, project_path: project.full_path, importer: project.import_type)
error(s_("ImportProjects|Error importing repository %{project_safe_import_url} into %{project_full_path} - %{message}") % { project_safe_import_url: project.safe_import_url, project_full_path: project.full_path, message: e.message }) error(s_("ImportProjects|Error importing repository %{project_safe_import_url} into %{project_full_path} - %{message}") % { project_safe_import_url: project.safe_import_url, project_full_path: project.full_path, message: e.message })
rescue => e rescue => e
message = Projects::ImportErrorFilter.filter_message(e.message) message = Projects::ImportErrorFilter.filter_message(e.message)
Gitlab::Sentry.track_acceptable_exception(e, extra: { project_path: project.full_path, importer: project.import_type }) Gitlab::Sentry.track_exception(e, project_path: project.full_path, importer: project.import_type)
error(s_("ImportProjects|Error importing repository %{project_safe_import_url} into %{project_full_path} - %{message}") % { project_safe_import_url: project.safe_import_url, project_full_path: project.full_path, message: message }) error(s_("ImportProjects|Error importing repository %{project_safe_import_url} into %{project_full_path} - %{message}") % { project_safe_import_url: project.safe_import_url, project_full_path: project.full_path, message: message })
end end
......
...@@ -32,7 +32,7 @@ module Prometheus ...@@ -32,7 +32,7 @@ module Prometheus
success(result) success(result)
rescue TypeError, ArgumentError => exception rescue TypeError, ArgumentError => exception
log_error(exception.message) log_error(exception.message)
Gitlab::Sentry.track_acceptable_exception(exception, extra: { Gitlab::Sentry.track_exception(exception, extra: {
template_string: query, template_string: query,
variables: predefined_context variables: predefined_context
}) })
......
...@@ -15,7 +15,7 @@ class DeleteStoredFilesWorker ...@@ -15,7 +15,7 @@ class DeleteStoredFilesWorker
unless klass unless klass
message = "Unknown class '#{class_name}'" message = "Unknown class '#{class_name}'"
logger.error(message) logger.error(message)
Gitlab::Sentry.track_exception(RuntimeError.new(message)) Gitlab::Sentry.track_and_raise_for_dev_exception(RuntimeError.new(message))
return return
end end
......
...@@ -11,7 +11,7 @@ class PagesDomainRemovalCronWorker ...@@ -11,7 +11,7 @@ class PagesDomainRemovalCronWorker
PagesDomain.for_removal.find_each do |domain| PagesDomain.for_removal.find_each do |domain|
domain.destroy! domain.destroy!
rescue => e rescue => e
Raven.capture_exception(e) Gitlab::Sentry.track_exception(e)
end end
end end
end end
...@@ -39,9 +39,9 @@ class RunPipelineScheduleWorker ...@@ -39,9 +39,9 @@ class RunPipelineScheduleWorker
"schedule_id: #{schedule.id} message: #{error.message}" "schedule_id: #{schedule.id} message: #{error.message}"
Gitlab::Sentry Gitlab::Sentry
.track_exception(error, .track_and_raise_for_dev_exception(error,
issue_url: 'https://gitlab.com/gitlab-org/gitlab-foss/issues/41231', issue_url: 'https://gitlab.com/gitlab-org/gitlab-foss/issues/41231',
extra: { schedule_id: schedule.id }) schedule_id: schedule.id)
end end
# rubocop:enable Gitlab/RailsLogger # rubocop:enable Gitlab/RailsLogger
......
...@@ -80,12 +80,12 @@ class StuckCiJobsWorker ...@@ -80,12 +80,12 @@ class StuckCiJobsWorker
end end
def track_exception_for_build(ex, build) def track_exception_for_build(ex, build)
Gitlab::Sentry.track_acceptable_exception(ex, extra: { Gitlab::Sentry.track_exception(ex,
build_id: build.id, build_id: build.id,
build_name: build.name, build_name: build.name,
build_stage: build.stage, build_stage: build.stage,
pipeline_id: build.pipeline_id, pipeline_id: build.pipeline_id,
project_id: build.project_id project_id: build.project_id
}) )
end end
end end
...@@ -29,7 +29,7 @@ module Sidekiq ...@@ -29,7 +29,7 @@ module Sidekiq
MSG MSG
rescue Sidekiq::Worker::EnqueueFromTransactionError => e rescue Sidekiq::Worker::EnqueueFromTransactionError => e
::Rails.logger.error(e.message) if ::Rails.env.production? ::Rails.logger.error(e.message) if ::Rails.env.production?
Gitlab::Sentry.track_exception(e) Gitlab::Sentry.track_and_raise_for_dev_exception(e)
end end
end end
......
...@@ -2,35 +2,4 @@ ...@@ -2,35 +2,4 @@
require 'gitlab/current_settings' require 'gitlab/current_settings'
def configure_sentry Gitlab::Sentry.configure
if Gitlab::Sentry.enabled?
Raven.configure do |config|
config.dsn = Gitlab.config.sentry.dsn
config.release = Gitlab.revision
config.current_environment = Gitlab.config.sentry.environment
# Sanitize fields based on those sanitized from Rails.
config.sanitize_fields = Rails.application.config.filter_parameters.map(&:to_s)
# Sanitize authentication headers
config.sanitize_http_headers = %w[Authorization Private-Token]
config.tags = { program: Gitlab.process_name }
# Debugging for https://gitlab.com/gitlab-org/gitlab-foss/issues/57727
config.before_send = lambda do |event, hint|
if ActiveModel::MissingAttributeError === hint[:exception]
columns_hash = ActiveRecord::Base
.connection
.schema_cache
.instance_variable_get(:@columns_hash)
.map { |k, v| [k, v.map(&:first)] }
.to_h
event.extra.merge!(columns_hash)
end
event
end
end
end
end
configure_sentry if Rails.env.production? || Rails.env.development?
...@@ -127,6 +127,68 @@ importer progresses. Here's what to do: ...@@ -127,6 +127,68 @@ importer progresses. Here's what to do:
logger.info(message: "Import error", error_code: 1, error: "I/O failure") logger.info(message: "Import error", error_code: 1, error: "I/O failure")
``` ```
## Exception Handling
It often happens that you catch the exception and want to track it.
It should be noted that manual logging of exceptions is not allowed, as:
1. Manual logged exceptions can leak confidential data,
1. Manual logged exception very often require to clean backtrace
which reduces the boilerplate,
1. Very often manually logged exception needs to be tracked to Sentry as well,
1. Manually logged exceptions does not use `correlation_id`, which makes hard
to pin them to request, user and context in which this exception was raised,
1. It is very likely that manually logged exceptions will end-up across
multiple files, which increases burden scraping all logging files.
To avoid duplicating and having consistent behavior the `Gitlab::Sentry`
provides helper methods to track exceptions:
1. `Gitlab::Sentry.track_and_raise_exception`: this method logs,
sends exception to Sentry (if configured) and re-raises the exception,
1. `Gitlab::Sentry.track_exception`: this method only logs
and sends exception to Sentry (if configured),
1. `Gitlab::Sentry.log_exception`: this method only logs the exception,
and DOES NOT send the exception to Sentry,
1. `Gitlab::Sentry.track_and_raise_for_dev_exception`: this method logs,
sends exception to Sentry (if configured) and re-raises the exception
for development and test enviroments.
It is advised to only use `Gitlab::Sentry.track_and_raise_exception`
and `Gitlab::Sentry.track_exception` as presented on below examples.
Consider adding additional extra parameters to provide more context
for each tracked exception.
### Example
```ruby
class MyService < ::BaseService
def execute
project.perform_expensive_operation
success
rescue => e
Gitlab::Sentry.track_exception(e, project_id: project.id)
error('Exception occurred')
end
end
```
```ruby
class MyService < ::BaseService
def execute
project.perform_expensive_operation
success
rescue => e
Gitlab::Sentry.track_and_raise_exception(e, project_id: project.id)
end
end
```
## Additional steps with new log files ## Additional steps with new log files
1. Consider log retention settings. By default, Omnibus will rotate any 1. Consider log retention settings. By default, Omnibus will rotate any
......
...@@ -126,7 +126,7 @@ module EE ...@@ -126,7 +126,7 @@ module EE
status: :error status: :error
}.merge(opts) }.merge(opts)
rescue Kubeclient::HttpError => e rescue Kubeclient::HttpError => e
::Gitlab::Sentry.track_acceptable_exception(e) ::Gitlab::Sentry.track_exception(e)
{ {
error: _('Kubernetes API returned status code: %{error_code}') % { error: _('Kubernetes API returned status code: %{error_code}') % {
......
...@@ -106,7 +106,7 @@ class JenkinsDeprecatedService < CiService ...@@ -106,7 +106,7 @@ class JenkinsDeprecatedService < CiService
begin begin
src = Nokogiri.parse(response).css('img.build-caption-status-icon,.build-caption>img').first.attributes['src'].value src = Nokogiri.parse(response).css('img.build-caption-status-icon,.build-caption>img').first.attributes['src'].value
rescue NoMethodError => ex rescue NoMethodError => ex
Raven.capture_exception(ex, extra: { 'response' => response }) Gitlab::Sentry.track_exception(ex, response: response)
return :error return :error
end end
......
...@@ -392,7 +392,7 @@ module API ...@@ -392,7 +392,7 @@ module API
# conan sends two upload requests, the first has no file, so we skip record creation if file.size == 0 # conan sends two upload requests, the first has no file, so we skip record creation if file.size == 0
::Packages::Conan::CreatePackageFileService.new(current_package, uploaded_file, params.merge(conan_file_type: file_type)).execute unless params['file.size'] == 0 ::Packages::Conan::CreatePackageFileService.new(current_package, uploaded_file, params.merge(conan_file_type: file_type)).execute unless params['file.size'] == 0
rescue ObjectStorage::RemoteStoreError => e rescue ObjectStorage::RemoteStoreError => e
Gitlab::Sentry.track_acceptable_exception(e, extra: { file_name: params[:file_name], project_id: project.id }) Gitlab::Sentry.track_exception(e, file_name: params[:file_name], project_id: project.id)
forbidden! forbidden!
end end
......
...@@ -28,7 +28,7 @@ module EE ...@@ -28,7 +28,7 @@ module EE
error = LimitExceededError.new(message) error = LimitExceededError.new(message)
# TODO: change this to Gitlab::Sentry.log_exception(error, extra_context) # TODO: change this to Gitlab::Sentry.log_exception(error, extra_context)
# https://gitlab.com/gitlab-org/gitlab/issues/32906 # https://gitlab.com/gitlab-org/gitlab/issues/32906
::Gitlab::Sentry.track_acceptable_exception(error, extra: extra_context) ::Gitlab::Sentry.track_exception(error, extra_context)
end end
end end
end end
......
...@@ -18,7 +18,7 @@ module Gitlab ...@@ -18,7 +18,7 @@ module Gitlab
rescue JSON::ParserError rescue JSON::ParserError
raise LicenseScanningParserError, 'JSON parsing failed' raise LicenseScanningParserError, 'JSON parsing failed'
rescue => e rescue => e
Gitlab::Sentry.track_exception(e) Gitlab::Sentry.track_and_raise_for_dev_exception(e)
raise LicenseScanningParserError, 'License scanning report parsing failed' raise LicenseScanningParserError, 'License scanning report parsing failed'
end end
end end
......
...@@ -19,7 +19,7 @@ module Gitlab ...@@ -19,7 +19,7 @@ module Gitlab
metrics_report.add_metric(name, metric_values.first) metrics_report.add_metric(name, metric_values.first)
rescue => e rescue => e
Gitlab::Sentry.track_exception(e) Gitlab::Sentry.track_and_raise_for_dev_exception(e)
raise MetricsParserError, "Metrics parsing failed" raise MetricsParserError, "Metrics parsing failed"
end end
end end
......
...@@ -17,7 +17,7 @@ module Gitlab ...@@ -17,7 +17,7 @@ module Gitlab
rescue JSON::ParserError rescue JSON::ParserError
raise SecurityReportParserError, 'JSON parsing failed' raise SecurityReportParserError, 'JSON parsing failed'
rescue => e rescue => e
Gitlab::Sentry.track_exception(e) Gitlab::Sentry.track_and_raise_for_dev_exception(e)
raise SecurityReportParserError, "#{report.type} security report parsing failed" raise SecurityReportParserError, "#{report.type} security report parsing failed"
end end
......
...@@ -54,9 +54,9 @@ describe ::Gitlab::Ci::Pipeline::Chain::Limit::Activity do ...@@ -54,9 +54,9 @@ describe ::Gitlab::Ci::Pipeline::Chain::Limit::Activity do
end end
it 'logs the error' do it 'logs the error' do
expect(Gitlab::Sentry).to receive(:track_acceptable_exception).with( expect(Gitlab::Sentry).to receive(:track_exception).with(
instance_of(EE::Gitlab::Ci::Limit::LimitExceededError), instance_of(EE::Gitlab::Ci::Limit::LimitExceededError),
extra: { project_id: project.id, plan: namespace.actual_plan_name } project_id: project.id, plan: namespace.actual_plan_name
) )
subject subject
...@@ -83,7 +83,7 @@ describe ::Gitlab::Ci::Pipeline::Chain::Limit::Activity do ...@@ -83,7 +83,7 @@ describe ::Gitlab::Ci::Pipeline::Chain::Limit::Activity do
end end
it 'does not log any error' do it 'does not log any error' do
expect(Gitlab::Sentry).not_to receive(:track_acceptable_exception) expect(Gitlab::Sentry).not_to receive(:track_exception)
subject subject
end end
......
...@@ -56,9 +56,9 @@ describe ::Gitlab::Ci::Pipeline::Chain::Limit::JobActivity do ...@@ -56,9 +56,9 @@ describe ::Gitlab::Ci::Pipeline::Chain::Limit::JobActivity do
end end
it 'logs the error' do it 'logs the error' do
expect(Gitlab::Sentry).to receive(:track_acceptable_exception).with( expect(Gitlab::Sentry).to receive(:track_exception).with(
instance_of(EE::Gitlab::Ci::Limit::LimitExceededError), instance_of(EE::Gitlab::Ci::Limit::LimitExceededError),
extra: { project_id: project.id, plan: namespace.actual_plan_name } project_id: project.id, plan: namespace.actual_plan_name
) )
subject subject
...@@ -85,7 +85,7 @@ describe ::Gitlab::Ci::Pipeline::Chain::Limit::JobActivity do ...@@ -85,7 +85,7 @@ describe ::Gitlab::Ci::Pipeline::Chain::Limit::JobActivity do
end end
it 'does not log any error' do it 'does not log any error' do
expect(Gitlab::Sentry).not_to receive(:track_acceptable_exception) expect(Gitlab::Sentry).not_to receive(:track_exception)
subject subject
end end
......
...@@ -72,9 +72,9 @@ describe ::Gitlab::Ci::Pipeline::Chain::Limit::Size do ...@@ -72,9 +72,9 @@ describe ::Gitlab::Ci::Pipeline::Chain::Limit::Size do
end end
it 'logs the error' do it 'logs the error' do
expect(Gitlab::Sentry).to receive(:track_acceptable_exception).with( expect(Gitlab::Sentry).to receive(:track_exception).with(
instance_of(EE::Gitlab::Ci::Limit::LimitExceededError), instance_of(EE::Gitlab::Ci::Limit::LimitExceededError),
extra: { project_id: project.id, plan: namespace.actual_plan_name } project_id: project.id, plan: namespace.actual_plan_name
) )
subject subject
...@@ -124,7 +124,7 @@ describe ::Gitlab::Ci::Pipeline::Chain::Limit::Size do ...@@ -124,7 +124,7 @@ describe ::Gitlab::Ci::Pipeline::Chain::Limit::Size do
end end
it 'does not log any error' do it 'does not log any error' do
expect(Gitlab::Sentry).not_to receive(:track_acceptable_exception) expect(Gitlab::Sentry).not_to receive(:track_exception)
subject subject
end end
......
...@@ -384,8 +384,9 @@ module API ...@@ -384,8 +384,9 @@ module API
def handle_api_exception(exception) def handle_api_exception(exception)
if report_exception?(exception) if report_exception?(exception)
define_params_for_grape_middleware define_params_for_grape_middleware
Gitlab::Sentry.context(current_user) Gitlab::Sentry.with_context(current_user) do
Gitlab::Sentry.track_acceptable_exception(exception, extra: params) Gitlab::Sentry.track_exception(exception, params)
end
end end
# This is used with GrapeLogging::Loggers::ExceptionLogger # This is used with GrapeLogging::Loggers::ExceptionLogger
......
...@@ -11,7 +11,6 @@ module Gitlab ...@@ -11,7 +11,6 @@ module Gitlab
{ title: 'task', color: '#7F8C8D' }].freeze { title: 'task', color: '#7F8C8D' }].freeze
attr_reader :project, :client, :errors, :users attr_reader :project, :client, :errors, :users
attr_accessor :logger
def initialize(project) def initialize(project)
@project = project @project = project
...@@ -20,7 +19,6 @@ module Gitlab ...@@ -20,7 +19,6 @@ module Gitlab
@labels = {} @labels = {}
@errors = [] @errors = []
@users = {} @users = {}
@logger = Gitlab::Import::Logger.build
end end
def execute def execute
...@@ -47,7 +45,8 @@ module Gitlab ...@@ -47,7 +45,8 @@ module Gitlab
backtrace = Gitlab::Profiler.clean_backtrace(ex.backtrace) backtrace = Gitlab::Profiler.clean_backtrace(ex.backtrace)
error = { type: :pull_request, iid: pull_request.iid, errors: ex.message, trace: backtrace, raw_response: pull_request.raw } error = { type: :pull_request, iid: pull_request.iid, errors: ex.message, trace: backtrace, raw_response: pull_request.raw }
log_error(error) Gitlab::Sentry.log_exception(ex, error)
# Omit the details from the database to avoid blowing up usage in the error column # Omit the details from the database to avoid blowing up usage in the error column
error.delete(:trace) error.delete(:trace)
error.delete(:raw_response) error.delete(:raw_response)
...@@ -275,10 +274,6 @@ module Gitlab ...@@ -275,10 +274,6 @@ module Gitlab
author.to_s + comment.note.to_s author.to_s + comment.note.to_s
end end
def log_error(details)
logger.error(log_base_data.merge(details))
end
def log_base_data def log_base_data
{ {
class: self.class.name, class: self.class.name,
......
...@@ -133,7 +133,10 @@ module Gitlab ...@@ -133,7 +133,10 @@ module Gitlab
log_info(stage: 'import_repository', message: 'finished import') log_info(stage: 'import_repository', message: 'finished import')
rescue Gitlab::Shell::Error => e rescue Gitlab::Shell::Error => e
log_error(stage: 'import_repository', message: 'failed import', error: e.message) Gitlab::Sentry.log_exception(
e,
stage: 'import_repository', message: 'failed import', error: e.message
)
# Expire cache to prevent scenarios such as: # Expire cache to prevent scenarios such as:
# 1. First import failed, but the repo was imported successfully, so +exists?+ returns true # 1. First import failed, but the repo was imported successfully, so +exists?+ returns true
...@@ -164,8 +167,10 @@ module Gitlab ...@@ -164,8 +167,10 @@ module Gitlab
batch.each do |pull_request| batch.each do |pull_request|
import_bitbucket_pull_request(pull_request) import_bitbucket_pull_request(pull_request)
rescue StandardError => e rescue StandardError => e
backtrace = Gitlab::Profiler.clean_backtrace(e.backtrace) Gitlab::Sentry.log_exception(
log_error(stage: 'import_pull_requests', iid: pull_request.iid, error: e.message, backtrace: backtrace) e,
stage: 'import_pull_requests', iid: pull_request.iid, error: e.message
)
errors << { type: :pull_request, iid: pull_request.iid, errors: e.message, backtrace: backtrace.join("\n"), raw_response: pull_request.raw } errors << { type: :pull_request, iid: pull_request.iid, errors: e.message, backtrace: backtrace.join("\n"), raw_response: pull_request.raw }
end end
...@@ -177,7 +182,11 @@ module Gitlab ...@@ -177,7 +182,11 @@ module Gitlab
client.delete_branch(project_key, repository_slug, branch.name, branch.sha) client.delete_branch(project_key, repository_slug, branch.name, branch.sha)
project.repository.delete_branch(branch.name) project.repository.delete_branch(branch.name)
rescue BitbucketServer::Connection::ConnectionError => e rescue BitbucketServer::Connection::ConnectionError => e
log_error(stage: 'delete_temp_branches', branch: branch.name, error: e.message) Gitlab::Sentry.log_exception(
e,
stage: 'delete_temp_branches', branch: branch.name, error: e.message
)
@errors << { type: :delete_temp_branches, branch_name: branch.name, errors: e.message } @errors << { type: :delete_temp_branches, branch_name: branch.name, errors: e.message }
end end
end end
...@@ -288,7 +297,11 @@ module Gitlab ...@@ -288,7 +297,11 @@ module Gitlab
# a regular note. # a regular note.
create_fallback_diff_note(merge_request, comment, position) create_fallback_diff_note(merge_request, comment, position)
rescue StandardError => e rescue StandardError => e
log_error(stage: 'create_diff_note', comment_id: comment.id, error: e.message) Gitlab::Sentry.log_exception(
e,
stage: 'create_diff_note', comment_id: comment.id, error: e.message
)
errors << { type: :pull_request, id: comment.id, errors: e.message } errors << { type: :pull_request, id: comment.id, errors: e.message }
nil nil
end end
...@@ -325,7 +338,11 @@ module Gitlab ...@@ -325,7 +338,11 @@ module Gitlab
merge_request.notes.create!(pull_request_comment_attributes(replies)) merge_request.notes.create!(pull_request_comment_attributes(replies))
end end
rescue StandardError => e rescue StandardError => e
log_error(stage: 'import_standalone_pr_comments', merge_request_id: merge_request.id, comment_id: comment.id, error: e.message) Gitlab::Sentry.log_exception(
e,
stage: 'import_standalone_pr_comments', merge_request_id: merge_request.id, comment_id: comment.id, error: e.message
)
errors << { type: :pull_request, comment_id: comment.id, errors: e.message } errors << { type: :pull_request, comment_id: comment.id, errors: e.message }
end end
end end
...@@ -360,10 +377,6 @@ module Gitlab ...@@ -360,10 +377,6 @@ module Gitlab
logger.info(log_base_data.merge(details)) logger.info(log_base_data.merge(details))
end end
def log_error(details)
logger.error(log_base_data.merge(details))
end
def log_warn(details) def log_warn(details)
logger.warn(log_base_data.merge(details)) logger.warn(log_base_data.merge(details))
end end
......
...@@ -67,11 +67,11 @@ module Gitlab ...@@ -67,11 +67,11 @@ module Gitlab
build_config(config) build_config(config)
rescue Gitlab::Config::Loader::Yaml::DataTooLargeError => e rescue Gitlab::Config::Loader::Yaml::DataTooLargeError => e
track_exception(e) track_and_raise_for_dev_exception(e)
raise Config::ConfigError, e.message raise Config::ConfigError, e.message
rescue Gitlab::Ci::Config::External::Context::TimeoutError => e rescue Gitlab::Ci::Config::External::Context::TimeoutError => e
track_exception(e) track_and_raise_for_dev_exception(e)
raise Config::ConfigError, TIMEOUT_MESSAGE raise Config::ConfigError, TIMEOUT_MESSAGE
end end
...@@ -94,8 +94,8 @@ module Gitlab ...@@ -94,8 +94,8 @@ module Gitlab
user: user) user: user)
end end
def track_exception(error) def track_and_raise_for_dev_exception(error)
Gitlab::Sentry.track_exception(error, extra: @context.sentry_payload) Gitlab::Sentry.track_and_raise_for_dev_exception(error, @context.sentry_payload)
end end
# Overriden in EE # Overriden in EE
......
...@@ -21,10 +21,10 @@ module Gitlab ...@@ -21,10 +21,10 @@ module Gitlab
rescue Gitlab::Ci::YamlProcessor::ValidationError => ex rescue Gitlab::Ci::YamlProcessor::ValidationError => ex
error(ex.message, config_error: true) error(ex.message, config_error: true)
rescue => ex rescue => ex
Gitlab::Sentry.track_acceptable_exception(ex, extra: { Gitlab::Sentry.track_exception(ex,
project_id: project.id, project_id: project.id,
sha: @pipeline.sha sha: @pipeline.sha
}) )
error("Undefined error (#{Labkit::Correlation::CorrelationId.current_id})", error("Undefined error (#{Labkit::Correlation::CorrelationId.current_id})",
config_error: true) config_error: true)
......
...@@ -35,7 +35,7 @@ module Gitlab ...@@ -35,7 +35,7 @@ module Gitlab
# match the blob, which is a bug. But we shouldn't fail to render # match the blob, which is a bug. But we shouldn't fail to render
# completely in that case, even though we want to report the error. # completely in that case, even though we want to report the error.
rescue RangeError => e rescue RangeError => e
Gitlab::Sentry.track_exception(e, issue_url: 'https://gitlab.com/gitlab-org/gitlab-foss/issues/45441') Gitlab::Sentry.track_and_raise_for_dev_exception(e, issue_url: 'https://gitlab.com/gitlab-org/gitlab-foss/issues/45441')
end end
end end
......
...@@ -67,8 +67,7 @@ module Gitlab ...@@ -67,8 +67,7 @@ module Gitlab
File.read(cert_file).scan(PEM_REGEX).map do |cert| File.read(cert_file).scan(PEM_REGEX).map do |cert|
OpenSSL::X509::Certificate.new(cert).to_pem OpenSSL::X509::Certificate.new(cert).to_pem
rescue OpenSSL::OpenSSLError => e rescue OpenSSL::OpenSSLError => e
Rails.logger.error "Could not load certificate #{cert_file} #{e}" # rubocop:disable Gitlab/RailsLogger Gitlab::Sentry.track_and_raise_for_dev_exception(e, cert_file: cert_file)
Gitlab::Sentry.track_exception(e, extra: { cert_file: cert_file })
nil nil
end.compact end.compact
end.uniq.join("\n") end.uniq.join("\n")
......
...@@ -91,12 +91,10 @@ module Gitlab ...@@ -91,12 +91,10 @@ module Gitlab
project.repository.add_branch(project.creator, source_branch, pull_request.source_branch_sha) project.repository.add_branch(project.creator, source_branch, pull_request.source_branch_sha)
rescue Gitlab::Git::CommandError => e rescue Gitlab::Git::CommandError => e
Gitlab::Sentry.track_acceptable_exception(e, Gitlab::Sentry.track_exception(e,
extra: {
source_branch: source_branch, source_branch: source_branch,
project_id: merge_request.project.id, project_id: merge_request.project.id,
merge_request_id: merge_request.id merge_request_id: merge_request.id)
})
end end
end end
end end
......
...@@ -110,9 +110,9 @@ module Gitlab ...@@ -110,9 +110,9 @@ module Gitlab
folder_contents = Dir.children(tmp_dir) folder_contents = Dir.children(tmp_dir)
# This means we left a GPG-agent process hanging. Logging the problem in # This means we left a GPG-agent process hanging. Logging the problem in
# sentry will make this more visible. # sentry will make this more visible.
Gitlab::Sentry.track_exception(e, Gitlab::Sentry.track_and_raise_for_dev_exception(e,
issue_url: 'https://gitlab.com/gitlab-org/gitlab/issues/20918', issue_url: 'https://gitlab.com/gitlab-org/gitlab/issues/20918',
extra: { tmp_dir: tmp_dir, contents: folder_contents }) tmp_dir: tmp_dir, contents: folder_contents)
end end
tmp_keychains_removed.increment unless File.exist?(tmp_dir) tmp_keychains_removed.increment unless File.exist?(tmp_dir)
......
...@@ -32,7 +32,7 @@ module Gitlab ...@@ -32,7 +32,7 @@ module Gitlab
# Will inform you if there needs to be `calls_gitaly: true` as a kwarg in the field declaration # Will inform you if there needs to be `calls_gitaly: true` as a kwarg in the field declaration
# if there is at least 1 Gitaly call involved with the field resolution. # if there is at least 1 Gitaly call involved with the field resolution.
error = RuntimeError.new("Gitaly is called for field '#{type_object.name}' on #{type_object.owner.try(:name)} - please either specify a constant complexity or add `calls_gitaly: true` to the field declaration") error = RuntimeError.new("Gitaly is called for field '#{type_object.name}' on #{type_object.owner.try(:name)} - please either specify a constant complexity or add `calls_gitaly: true` to the field declaration")
Gitlab::Sentry.track_exception(error) Gitlab::Sentry.track_and_raise_for_dev_exception(error)
end end
end end
end end
......
...@@ -18,7 +18,7 @@ module Gitlab ...@@ -18,7 +18,7 @@ module Gitlab
variables: variables variables: variables
}) })
rescue => e rescue => e
Gitlab::Sentry.track_exception(e) Gitlab::Sentry.track_and_raise_for_dev_exception(e)
default_initial_values(query) default_initial_values(query)
end end
...@@ -38,7 +38,7 @@ module Gitlab ...@@ -38,7 +38,7 @@ module Gitlab
GraphqlLogger.info(memo.except!(:time_started, :query)) GraphqlLogger.info(memo.except!(:time_started, :query))
rescue => e rescue => e
Gitlab::Sentry.track_exception(e) Gitlab::Sentry.track_and_raise_for_dev_exception(e)
end end
private private
......
...@@ -61,7 +61,7 @@ module Gitlab ...@@ -61,7 +61,7 @@ module Gitlab
tokens = lexer.lex(text, continue: continue) tokens = lexer.lex(text, continue: continue)
Timeout.timeout(timeout_time) { @formatter.format(tokens, tag: tag).html_safe } Timeout.timeout(timeout_time) { @formatter.format(tokens, tag: tag).html_safe }
rescue Timeout::Error => e rescue Timeout::Error => e
Gitlab::Sentry.track_exception(e) Gitlab::Sentry.track_and_raise_for_dev_exception(e)
highlight_plain(text) highlight_plain(text)
rescue rescue
highlight_plain(text) highlight_plain(text)
......
...@@ -82,11 +82,8 @@ module Gitlab ...@@ -82,11 +82,8 @@ module Gitlab
end end
def log_import_failure(relation_key, relation_index, exception) def log_import_failure(relation_key, relation_index, exception)
Gitlab::Sentry.track_acceptable_exception( Gitlab::Sentry.track_exception(exception,
exception, project_id: @importable.id, relation_key: relation_key, relation_index: relation_index)
extra: { project_id: @importable.id,
relation_key: relation_key,
relation_index: relation_index })
ImportFailure.create( ImportFailure.create(
project: @importable, project: @importable,
......
...@@ -56,11 +56,7 @@ module Gitlab ...@@ -56,11 +56,7 @@ module Gitlab
end end
def error(error) def error(error)
error_payload = { message: error.message } Gitlab::Sentry.track_exception(error, log_base_data)
error_payload[:error_backtrace] = Gitlab::Profiler.clean_backtrace(error.backtrace) if error.backtrace
log_error(error_payload)
Gitlab::Sentry.track_acceptable_exception(error, extra: log_base_data)
add_error_message(error.message) add_error_message(error.message)
end end
......
...@@ -118,6 +118,8 @@ module Gitlab ...@@ -118,6 +118,8 @@ module Gitlab
end end
def self.clean_backtrace(backtrace) def self.clean_backtrace(backtrace)
return unless backtrace
Array(Rails.backtrace_cleaner.clean(backtrace)).reject do |line| Array(Rails.backtrace_cleaner.clean(backtrace)).reject do |line|
line.match(Regexp.union(IGNORE_BACKTRACES)) line.match(Regexp.union(IGNORE_BACKTRACES))
end end
......
...@@ -2,23 +2,51 @@ ...@@ -2,23 +2,51 @@
module Gitlab module Gitlab
module Sentry module Sentry
def self.enabled? class << self
(Rails.env.production? || Rails.env.development?) && def configure
Gitlab.config.sentry.enabled Raven.configure do |config|
config.dsn = sentry_dsn
config.release = Gitlab.revision
config.current_environment = Gitlab.config.sentry.environment
# Sanitize fields based on those sanitized from Rails.
config.sanitize_fields = Rails.application.config.filter_parameters.map(&:to_s)
# Sanitize authentication headers
config.sanitize_http_headers = %w[Authorization Private-Token]
config.tags = { program: Gitlab.process_name }
# Debugging for https://gitlab.com/gitlab-org/gitlab-foss/issues/57727
config.before_send = method(:add_context_from_exception_type)
end
end end
def self.context(current_user = nil) def with_context(current_user = nil)
return unless enabled? last_user_context = Raven.context.user
user_context = {
id: current_user&.id,
email: current_user&.email,
username: current_user&.username
}.compact
Raven.tags_context(default_tags) Raven.tags_context(default_tags)
Raven.user_context(user_context)
if current_user yield
Raven.user_context( ensure
id: current_user.id, Raven.user_context(last_user_context)
email: current_user.email,
username: current_user.username
)
end end
# This should be used when you want to passthrough exception handling:
# rescue and raise to be catched in upper layers of the application.
#
# If the exception implements the method `sentry_extra_data` and that method
# returns a Hash, then the return value of that method will be merged into
# `extra`. Exceptions can use this mechanism to provide structured data
# to sentry in addition to their message and back-trace.
def track_and_raise_exception(exception, extra = {})
process_exception(exception, sentry: true, extra: extra)
raise exception
end end
# This can be used for investigating exceptions that can be recovered from in # This can be used for investigating exceptions that can be recovered from in
...@@ -28,50 +56,99 @@ module Gitlab ...@@ -28,50 +56,99 @@ module Gitlab
# That way we can track down these exceptions with as much information as we # That way we can track down these exceptions with as much information as we
# need to resolve them. # need to resolve them.
# #
# If the exception implements the method `sentry_extra_data` and that method
# returns a Hash, then the return value of that method will be merged into
# `extra`. Exceptions can use this mechanism to provide structured data
# to sentry in addition to their message and back-trace.
#
# Provide an issue URL for follow up. # Provide an issue URL for follow up.
def self.track_exception(exception, issue_url: nil, extra: {}) # as `issue_url: 'http://gitlab.com/gitlab-org/gitlab/issues/111'`
track_acceptable_exception(exception, issue_url: issue_url, extra: extra) def track_and_raise_for_dev_exception(exception, extra = {})
process_exception(exception, sentry: true, extra: extra)
raise exception if should_raise_for_dev? raise exception if should_raise_for_dev?
end end
# This should be used when you do not want to raise an exception in # This should be used when you only want to track the exception.
# development and test. If you need development and test to behave #
# just the same as production you can use this instead of # If the exception implements the method `sentry_extra_data` and that method
# track_exception. # returns a Hash, then the return value of that method will be merged into
# `extra`. Exceptions can use this mechanism to provide structured data
# to sentry in addition to their message and back-trace.
def track_exception(exception, extra = {})
process_exception(exception, sentry: true, extra: extra)
end
# This should be used when you only want to log the exception,
# but not send it to Sentry.
# #
# If the exception implements the method `sentry_extra_data` and that method # If the exception implements the method `sentry_extra_data` and that method
# returns a Hash, then the return value of that method will be merged into # returns a Hash, then the return value of that method will be merged into
# `extra`. Exceptions can use this mechanism to provide structured data # `extra`. Exceptions can use this mechanism to provide structured data
# to sentry in addition to their message and back-trace. # to sentry in addition to their message and back-trace.
def self.track_acceptable_exception(exception, issue_url: nil, extra: {}) def log_exception(exception, extra = {})
if enabled? process_exception(exception, extra: extra)
extra = build_extra_data(exception, issue_url, extra) end
context # Make sure we've set everything we know in the context
private
def process_exception(exception, sentry: false, logging: true, extra:)
exception.try(:sentry_extra_data)&.tap do |data|
extra = extra.merge(data) if data.is_a?(Hash)
end
if sentry && Raven.configuration.server
Raven.capture_exception(exception, tags: default_tags, extra: extra) Raven.capture_exception(exception, tags: default_tags, extra: extra)
end end
if logging
# TODO: this logic could migrate into `Gitlab::ExceptionLogFormatter`
# and we could also flatten deep nested hashes if required for search
# (e.g. if `extra` includes hash of hashes).
# In the current implementation, we don't flatten multi-level folded hashes.
log_hash = {}
Raven.context.tags.each { |name, value| log_hash["tags.#{name}"] = value }
Raven.context.user.each { |name, value| log_hash["user.#{name}"] = value }
Raven.context.extra.merge(extra).each { |name, value| log_hash["extra.#{name}"] = value }
Gitlab::ExceptionLogFormatter.format!(exception, log_hash)
Gitlab::Sentry::Logger.error(log_hash)
end
end
def sentry_dsn
return unless Rails.env.production? || Rails.env.development?
return unless Gitlab.config.sentry.enabled
Gitlab.config.sentry.dsn
end end
def self.should_raise_for_dev? def should_raise_for_dev?
Rails.env.development? || Rails.env.test? Rails.env.development? || Rails.env.test?
end end
def self.default_tags def default_tags
{ {
Labkit::Correlation::CorrelationId::LOG_KEY.to_sym => Labkit::Correlation::CorrelationId.current_id, Labkit::Correlation::CorrelationId::LOG_KEY.to_sym => Labkit::Correlation::CorrelationId.current_id,
locale: I18n.locale locale: I18n.locale
} }
end end
def self.build_extra_data(exception, issue_url, extra) def add_context_from_exception_type(event, hint)
exception.try(:sentry_extra_data)&.tap do |data| if ActiveModel::MissingAttributeError === hint[:exception]
extra.merge!(data) if data.is_a?(Hash) columns_hash = ActiveRecord::Base
end .connection
.schema_cache
.instance_variable_get(:@columns_hash)
.map { |k, v| [k, v.map(&:first)] }
.to_h
extra.merge({ issue_url: issue_url }.compact) event.extra.merge!(columns_hash)
end end
private_class_method :build_extra_data event
end
end
end end
end end
# frozen_string_literal: true
module Gitlab
module Sentry
class Logger < ::Gitlab::JsonLogger
def self.file_name_noext
'exceptions_json'
end
end
end
end
...@@ -126,7 +126,7 @@ module Gitlab ...@@ -126,7 +126,7 @@ module Gitlab
true true
rescue => e rescue => e
Gitlab::Sentry.track_acceptable_exception(e, extra: { path: path, new_path: new_path, storage: storage }) Gitlab::Sentry.track_exception(e, path: path, new_path: new_path, storage: storage)
false false
end end
...@@ -158,7 +158,7 @@ module Gitlab ...@@ -158,7 +158,7 @@ module Gitlab
true true
rescue => e rescue => e
Rails.logger.warn("Repository does not exist: #{e} at: #{name}.git") # rubocop:disable Gitlab/RailsLogger Rails.logger.warn("Repository does not exist: #{e} at: #{name}.git") # rubocop:disable Gitlab/RailsLogger
Gitlab::Sentry.track_acceptable_exception(e, extra: { path: name, storage: storage }) Gitlab::Sentry.track_exception(e, path: name, storage: storage)
false false
end end
...@@ -267,7 +267,7 @@ module Gitlab ...@@ -267,7 +267,7 @@ module Gitlab
def mv_namespace(storage, old_name, new_name) def mv_namespace(storage, old_name, new_name)
Gitlab::GitalyClient::NamespaceService.new(storage).rename(old_name, new_name) Gitlab::GitalyClient::NamespaceService.new(storage).rename(old_name, new_name)
rescue GRPC::InvalidArgument => e rescue GRPC::InvalidArgument => e
Gitlab::Sentry.track_acceptable_exception(e, extra: { old_name: old_name, new_name: new_name, storage: storage }) Gitlab::Sentry.track_exception(e, old_name: old_name, new_name: new_name, storage: storage)
false false
end end
......
...@@ -8,7 +8,7 @@ module Gitlab::UsageDataCounters ...@@ -8,7 +8,7 @@ module Gitlab::UsageDataCounters
class << self class << self
def redis_key(event) def redis_key(event)
Gitlab::Sentry.track_exception(UnknownEvent, extra: { event: event }) unless known_events.include?(event.to_s) Gitlab::Sentry.track_and_raise_for_dev_exception(UnknownEvent.new, event: event) unless known_events.include?(event.to_s)
"USAGE_#{prefix}_#{event}".upcase "USAGE_#{prefix}_#{event}".upcase
end end
......
...@@ -62,7 +62,7 @@ module Sentry ...@@ -62,7 +62,7 @@ module Sentry
def handle_mapping_exceptions(&block) def handle_mapping_exceptions(&block)
yield yield
rescue KeyError => e rescue KeyError => e
Gitlab::Sentry.track_acceptable_exception(e) Gitlab::Sentry.track_exception(e)
raise MissingKeysError, "Sentry API response is missing keys. #{e.message}" raise MissingKeysError, "Sentry API response is missing keys. #{e.message}"
end end
...@@ -118,7 +118,7 @@ module Sentry ...@@ -118,7 +118,7 @@ module Sentry
def handle_request_exceptions def handle_request_exceptions
yield yield
rescue Gitlab::HTTP::Error => e rescue Gitlab::HTTP::Error => e
Gitlab::Sentry.track_acceptable_exception(e) Gitlab::Sentry.track_exception(e)
raise_error 'Error when connecting to Sentry' raise_error 'Error when connecting to Sentry'
rescue Net::OpenTimeout rescue Net::OpenTimeout
raise_error 'Connection to Sentry timed out' raise_error 'Connection to Sentry timed out'
...@@ -129,7 +129,7 @@ module Sentry ...@@ -129,7 +129,7 @@ module Sentry
rescue Errno::ECONNREFUSED rescue Errno::ECONNREFUSED
raise_error 'Connection refused' raise_error 'Connection refused'
rescue => e rescue => e
Gitlab::Sentry.track_acceptable_exception(e) Gitlab::Sentry.track_exception(e)
raise_error "Sentry request failed due to #{e.class}" raise_error "Sentry request failed due to #{e.class}"
end end
......
...@@ -19211,9 +19211,6 @@ msgstr "" ...@@ -19211,9 +19211,6 @@ msgstr ""
msgid "Upload file" msgid "Upload file"
msgstr "" msgstr ""
msgid "Upload file does not exist"
msgstr ""
msgid "Upload object map" msgid "Upload object map"
msgstr "" msgstr ""
......
...@@ -1159,7 +1159,7 @@ describe Projects::IssuesController do ...@@ -1159,7 +1159,7 @@ describe Projects::IssuesController do
end end
it "prevents deletion if destroy_confirm is not set" do it "prevents deletion if destroy_confirm is not set" do
expect(Gitlab::Sentry).to receive(:track_acceptable_exception).and_call_original expect(Gitlab::Sentry).to receive(:track_exception).and_call_original
delete :destroy, params: { namespace_id: project.namespace, project_id: project, id: issue.iid } delete :destroy, params: { namespace_id: project.namespace, project_id: project, id: issue.iid }
...@@ -1168,7 +1168,7 @@ describe Projects::IssuesController do ...@@ -1168,7 +1168,7 @@ describe Projects::IssuesController do
end end
it "prevents deletion in JSON format if destroy_confirm is not set" do it "prevents deletion in JSON format if destroy_confirm is not set" do
expect(Gitlab::Sentry).to receive(:track_acceptable_exception).and_call_original expect(Gitlab::Sentry).to receive(:track_exception).and_call_original
delete :destroy, params: { namespace_id: project.namespace, project_id: project, id: issue.iid, format: 'json' } delete :destroy, params: { namespace_id: project.namespace, project_id: project, id: issue.iid, format: 'json' }
......
...@@ -620,7 +620,7 @@ describe Projects::MergeRequestsController do ...@@ -620,7 +620,7 @@ describe Projects::MergeRequestsController do
end end
it "prevents deletion if destroy_confirm is not set" do it "prevents deletion if destroy_confirm is not set" do
expect(Gitlab::Sentry).to receive(:track_acceptable_exception).and_call_original expect(Gitlab::Sentry).to receive(:track_exception).and_call_original
delete :destroy, params: { namespace_id: project.namespace, project_id: project, id: merge_request.iid } delete :destroy, params: { namespace_id: project.namespace, project_id: project, id: merge_request.iid }
...@@ -629,7 +629,7 @@ describe Projects::MergeRequestsController do ...@@ -629,7 +629,7 @@ describe Projects::MergeRequestsController do
end end
it "prevents deletion in JSON format if destroy_confirm is not set" do it "prevents deletion in JSON format if destroy_confirm is not set" do
expect(Gitlab::Sentry).to receive(:track_acceptable_exception).and_call_original expect(Gitlab::Sentry).to receive(:track_exception).and_call_original
delete :destroy, params: { namespace_id: project.namespace, project_id: project, id: merge_request.iid, format: 'json' } delete :destroy, params: { namespace_id: project.namespace, project_id: project, id: merge_request.iid, format: 'json' }
......
...@@ -157,7 +157,7 @@ describe Gitlab::Ci::Config do ...@@ -157,7 +157,7 @@ describe Gitlab::Ci::Config do
describe '.new' do describe '.new' do
it 'raises error' do it 'raises error' do
expect(Gitlab::Sentry).to receive(:track_exception) expect(Gitlab::Sentry).to receive(:track_and_raise_for_dev_exception)
expect { config }.to raise_error( expect { config }.to raise_error(
described_class::ConfigError, described_class::ConfigError,
...@@ -367,7 +367,7 @@ describe Gitlab::Ci::Config do ...@@ -367,7 +367,7 @@ describe Gitlab::Ci::Config do
end end
it 'raises error TimeoutError' do it 'raises error TimeoutError' do
expect(Gitlab::Sentry).to receive(:track_exception) expect(Gitlab::Sentry).to receive(:track_and_raise_for_dev_exception)
expect { config }.to raise_error( expect { config }.to raise_error(
described_class::ConfigError, described_class::ConfigError,
......
...@@ -105,7 +105,7 @@ describe Gitlab::Diff::Highlight do ...@@ -105,7 +105,7 @@ describe Gitlab::Diff::Highlight do
end end
it 'keeps the original rich line' do it 'keeps the original rich line' do
allow(Gitlab::Sentry).to receive(:track_exception) allow(Gitlab::Sentry).to receive(:track_and_raise_for_dev_exception)
code = %q{+ raise RuntimeError, "System commands must be given as an array of strings"} code = %q{+ raise RuntimeError, "System commands must be given as an array of strings"}
...@@ -114,7 +114,7 @@ describe Gitlab::Diff::Highlight do ...@@ -114,7 +114,7 @@ describe Gitlab::Diff::Highlight do
end end
it 'reports to Sentry if configured' do it 'reports to Sentry if configured' do
expect(Gitlab::Sentry).to receive(:track_exception).and_call_original expect(Gitlab::Sentry).to receive(:track_and_raise_for_dev_exception).and_call_original
expect { subject }. to raise_exception(RangeError) expect { subject }. to raise_exception(RangeError)
end end
......
...@@ -86,12 +86,11 @@ describe Gitlab::GitalyClient do ...@@ -86,12 +86,11 @@ describe Gitlab::GitalyClient do
describe '.stub_certs' do describe '.stub_certs' do
it 'skips certificates if OpenSSLError is raised and report it' do it 'skips certificates if OpenSSLError is raised and report it' do
expect(Rails.logger).to receive(:error).at_least(:once)
expect(Gitlab::Sentry) expect(Gitlab::Sentry)
.to receive(:track_exception) .to receive(:track_and_raise_for_dev_exception)
.with( .with(
a_kind_of(OpenSSL::X509::CertificateError), a_kind_of(OpenSSL::X509::CertificateError),
extra: { cert_file: a_kind_of(String) }).at_least(:once) cert_file: a_kind_of(String)).at_least(:once)
expect(OpenSSL::X509::Certificate) expect(OpenSSL::X509::Certificate)
.to receive(:new) .to receive(:new)
......
...@@ -301,7 +301,7 @@ describe Gitlab::GithubImport::Importer::PullRequestImporter, :clean_gitlab_redi ...@@ -301,7 +301,7 @@ describe Gitlab::GithubImport::Importer::PullRequestImporter, :clean_gitlab_redi
it 'ignores Git command errors when creating a branch' do it 'ignores Git command errors when creating a branch' do
expect(project.repository).to receive(:add_branch).and_raise(Gitlab::Git::CommandError) expect(project.repository).to receive(:add_branch).and_raise(Gitlab::Git::CommandError)
expect(Gitlab::Sentry).to receive(:track_acceptable_exception).and_call_original expect(Gitlab::Sentry).to receive(:track_exception).and_call_original
mr = insert_git_data mr = insert_git_data
......
...@@ -182,17 +182,17 @@ describe Gitlab::Gpg do ...@@ -182,17 +182,17 @@ describe Gitlab::Gpg do
expected_tmp_dir = nil expected_tmp_dir = nil
expect(described_class).to receive(:cleanup_tmp_dir).and_raise(expected_exception) expect(described_class).to receive(:cleanup_tmp_dir).and_raise(expected_exception)
allow(Gitlab::Sentry).to receive(:track_exception) allow(Gitlab::Sentry).to receive(:track_and_raise_for_dev_exception)
described_class.using_tmp_keychain do described_class.using_tmp_keychain do
expected_tmp_dir = described_class.current_home_dir expected_tmp_dir = described_class.current_home_dir
FileUtils.touch(File.join(expected_tmp_dir, 'dummy.file')) FileUtils.touch(File.join(expected_tmp_dir, 'dummy.file'))
end end
expect(Gitlab::Sentry).to have_received(:track_exception).with( expect(Gitlab::Sentry).to have_received(:track_and_raise_for_dev_exception).with(
expected_exception, expected_exception,
issue_url: 'https://gitlab.com/gitlab-org/gitlab/issues/20918', issue_url: 'https://gitlab.com/gitlab-org/gitlab/issues/20918',
extra: { tmp_dir: expected_tmp_dir, contents: ['dummy.file'] } tmp_dir: expected_tmp_dir, contents: ['dummy.file']
) )
end end
......
...@@ -49,24 +49,9 @@ describe Gitlab::ImportExport::Shared do ...@@ -49,24 +49,9 @@ describe Gitlab::ImportExport::Shared do
it 'updates the import JID' do it 'updates the import JID' do
import_state = create(:import_state, project: project, jid: 'jid-test') import_state = create(:import_state, project: project, jid: 'jid-test')
expect_next_instance_of(Gitlab::Import::Logger) do |logger| expect(Gitlab::Sentry)
expect(logger).to receive(:error).with(hash_including(import_jid: import_state.jid)) .to receive(:track_exception)
end .with(error, hash_including(import_jid: import_state.jid))
subject.error(error)
end
it 'calls the error logger without a backtrace' do
expect(subject).to receive(:log_error).with(message: error.message)
subject.error(error)
end
it 'calls the error logger with the full message' do
backtrace = caller
allow(error).to receive(:backtrace).and_return(caller)
expect(subject).to receive(:log_error).with(message: error.message, error_backtrace: Gitlab::Profiler.clean_backtrace(backtrace))
subject.error(error) subject.error(error)
end end
......
...@@ -8,10 +8,20 @@ describe Gitlab::ImportExport::VersionChecker do ...@@ -8,10 +8,20 @@ describe Gitlab::ImportExport::VersionChecker do
describe 'bundle a project Git repo' do describe 'bundle a project Git repo' do
let(:version) { Gitlab::ImportExport.version } let(:version) { Gitlab::ImportExport.version }
let(:version_file) { Tempfile.new('VERSION') }
before do before do
allow_any_instance_of(Gitlab::ImportExport::Shared).to receive(:relative_archive_path).and_return('') allow_any_instance_of(Gitlab::ImportExport::Shared).to receive(:relative_archive_path).and_return('')
allow(File).to receive(:open).and_return(version)
version_file.write(version)
version_file.rewind
allow_any_instance_of(described_class).to receive(:version_file).and_return(version_file.path)
end
after do
version_file.close
version_file.unlink
end end
it 'returns true if Import/Export have the same version' do it 'returns true if Import/Export have the same version' do
......
...@@ -3,12 +3,31 @@ ...@@ -3,12 +3,31 @@
require 'spec_helper' require 'spec_helper'
describe Gitlab::Sentry do describe Gitlab::Sentry do
describe '.context' do let(:exception) { RuntimeError.new('boom') }
it 'adds the expected tags' do let(:issue_url) { 'http://gitlab.com/gitlab-org/gitlab-foss/issues/1' }
expect(described_class).to receive(:enabled?).and_return(true)
let(:expected_payload_includes) do
[
{ 'exception.class' => 'RuntimeError' },
{ 'exception.message' => 'boom' },
{ 'tags.correlation_id' => 'cid' },
{ 'extra.some_other_info' => 'info' },
{ 'extra.issue_url' => 'http://gitlab.com/gitlab-org/gitlab-foss/issues/1' }
]
end
before do
stub_sentry_settings
allow(described_class).to receive(:sentry_dsn).and_return(Gitlab.config.sentry.dsn)
allow(Labkit::Correlation::CorrelationId).to receive(:current_id).and_return('cid') allow(Labkit::Correlation::CorrelationId).to receive(:current_id).and_return('cid')
described_class.context(nil) described_class.configure
end
describe '.with_context' do
it 'adds the expected tags' do
described_class.with_context {}
expect(Raven.tags_context[:locale].to_s).to eq(I18n.locale.to_s) expect(Raven.tags_context[:locale].to_s).to eq(I18n.locale.to_s)
expect(Raven.tags_context[Labkit::Correlation::CorrelationId::LOG_KEY.to_sym].to_s) expect(Raven.tags_context[Labkit::Correlation::CorrelationId::LOG_KEY.to_sym].to_s)
...@@ -16,23 +35,23 @@ describe Gitlab::Sentry do ...@@ -16,23 +35,23 @@ describe Gitlab::Sentry do
end end
end end
describe '.track_exception' do describe '.track_and_raise_for_dev_exception' do
let(:exception) { RuntimeError.new('boom') } context 'when exceptions for dev should be raised' do
before do before do
allow(described_class).to receive(:enabled?).and_return(true) expect(described_class).to receive(:should_raise_for_dev?).and_return(true)
end end
it 'raises the exception if it should' do it 'raises the exception' do
expect(described_class).to receive(:should_raise_for_dev?).and_return(true) expect(Raven).to receive(:capture_exception)
expect { described_class.track_exception(exception) }
expect { described_class.track_and_raise_for_dev_exception(exception) }
.to raise_error(RuntimeError) .to raise_error(RuntimeError)
end end
end
context 'when exceptions should not be raised' do context 'when exceptions for dev should not be raised' do
before do before do
allow(described_class).to receive(:should_raise_for_dev?).and_return(false) expect(described_class).to receive(:should_raise_for_dev?).and_return(false)
allow(Labkit::Correlation::CorrelationId).to receive(:current_id).and_return('cid')
end end
it 'logs the exception with all attributes passed' do it 'logs the exception with all attributes passed' do
...@@ -50,30 +69,49 @@ describe Gitlab::Sentry do ...@@ -50,30 +69,49 @@ describe Gitlab::Sentry do
tags: a_hash_including(expected_tags), tags: a_hash_including(expected_tags),
extra: a_hash_including(expected_extras)) extra: a_hash_including(expected_extras))
described_class.track_exception( described_class.track_and_raise_for_dev_exception(
exception, exception,
issue_url: 'http://gitlab.com/gitlab-org/gitlab-foss/issues/1', issue_url: issue_url,
extra: { some_other_info: 'info' } some_other_info: 'info'
) )
end end
it 'sets the context' do it 'calls Gitlab::Sentry::Logger.error with formatted payload' do
expect(described_class).to receive(:context) expect(Gitlab::Sentry::Logger).to receive(:error)
.with(a_hash_including(*expected_payload_includes))
described_class.track_exception(exception) described_class.track_and_raise_for_dev_exception(
exception,
issue_url: issue_url,
some_other_info: 'info'
)
end end
end end
end end
context '.track_acceptable_exception' do describe '.track_and_raise_exception' do
let(:exception) { RuntimeError.new('boom') } it 'always raises the exception' do
let(:issue_url) { 'http://gitlab.com/gitlab-org/gitlab-foss/issues/1' } expect(Raven).to receive(:capture_exception)
before do expect { described_class.track_and_raise_exception(exception) }
allow(described_class).to receive(:enabled?).and_return(true) .to raise_error(RuntimeError)
allow(Labkit::Correlation::CorrelationId).to receive(:current_id).and_return('cid') end
it 'calls Gitlab::Sentry::Logger.error with formatted payload' do
expect(Gitlab::Sentry::Logger).to receive(:error)
.with(a_hash_including(*expected_payload_includes))
expect do
described_class.track_and_raise_exception(
exception,
issue_url: issue_url,
some_other_info: 'info'
)
end.to raise_error(RuntimeError)
end
end end
describe '.track_exception' do
it 'calls Raven.capture_exception' do it 'calls Raven.capture_exception' do
expected_extras = { expected_extras = {
some_other_info: 'info', some_other_info: 'info',
...@@ -89,34 +127,45 @@ describe Gitlab::Sentry do ...@@ -89,34 +127,45 @@ describe Gitlab::Sentry do
tags: a_hash_including(expected_tags), tags: a_hash_including(expected_tags),
extra: a_hash_including(expected_extras)) extra: a_hash_including(expected_extras))
described_class.track_acceptable_exception( described_class.track_exception(
exception,
issue_url: issue_url,
some_other_info: 'info'
)
end
it 'calls Gitlab::Sentry::Logger.error with formatted payload' do
expect(Gitlab::Sentry::Logger).to receive(:error)
.with(a_hash_including(*expected_payload_includes))
described_class.track_exception(
exception, exception,
issue_url: issue_url, issue_url: issue_url,
extra: { some_other_info: 'info' } some_other_info: 'info'
) )
end end
context 'the exception implements :sentry_extra_data' do context 'the exception implements :sentry_extra_data' do
let(:extra_info) { { event: 'explosion', size: :massive } } let(:extra_info) { { event: 'explosion', size: :massive } }
let(:exception) { double(message: 'bang!', sentry_extra_data: extra_info) } let(:exception) { double(message: 'bang!', sentry_extra_data: extra_info, backtrace: caller) }
it 'includes the extra data from the exception in the tracking information' do it 'includes the extra data from the exception in the tracking information' do
expect(Raven).to receive(:capture_exception) expect(Raven).to receive(:capture_exception)
.with(exception, a_hash_including(extra: a_hash_including(extra_info))) .with(exception, a_hash_including(extra: a_hash_including(extra_info)))
described_class.track_acceptable_exception(exception) described_class.track_exception(exception)
end end
end end
context 'the exception implements :sentry_extra_data, which returns nil' do context 'the exception implements :sentry_extra_data, which returns nil' do
let(:exception) { double(message: 'bang!', sentry_extra_data: nil) } let(:exception) { double(message: 'bang!', sentry_extra_data: nil, backtrace: caller) }
it 'just includes the other extra info' do it 'just includes the other extra info' do
extra_info = { issue_url: issue_url } extra_info = { issue_url: issue_url }
expect(Raven).to receive(:capture_exception) expect(Raven).to receive(:capture_exception)
.with(exception, a_hash_including(extra: a_hash_including(extra_info))) .with(exception, a_hash_including(extra: a_hash_including(extra_info)))
described_class.track_acceptable_exception(exception, extra_info) described_class.track_exception(exception, extra_info)
end end
end end
end end
......
...@@ -3522,7 +3522,7 @@ describe Ci::Build do ...@@ -3522,7 +3522,7 @@ describe Ci::Build do
end end
it 'can drop the build' do it 'can drop the build' do
expect(Gitlab::Sentry).to receive(:track_exception) expect(Gitlab::Sentry).to receive(:track_and_raise_for_dev_exception)
expect { build.drop! }.not_to raise_error expect { build.drop! }.not_to raise_error
......
...@@ -1004,8 +1004,8 @@ describe Clusters::Cluster, :use_clean_rails_memory_store_caching do ...@@ -1004,8 +1004,8 @@ describe Clusters::Cluster, :use_clean_rails_memory_store_caching do
it { is_expected.to eq(connection_status: :unknown_failure) } it { is_expected.to eq(connection_status: :unknown_failure) }
it 'notifies Sentry' do it 'notifies Sentry' do
expect(Gitlab::Sentry).to receive(:track_acceptable_exception) expect(Gitlab::Sentry).to receive(:track_exception)
.with(instance_of(StandardError), hash_including(extra: { cluster_id: cluster.id })) .with(instance_of(StandardError), hash_including(cluster_id: cluster.id))
subject subject
end end
......
...@@ -82,7 +82,7 @@ describe GroupDescendant do ...@@ -82,7 +82,7 @@ describe GroupDescendant do
end end
it 'tracks the exception when a parent was not preloaded' do it 'tracks the exception when a parent was not preloaded' do
expect(Gitlab::Sentry).to receive(:track_exception).and_call_original expect(Gitlab::Sentry).to receive(:track_and_raise_for_dev_exception).and_call_original
expect { described_class.build_hierarchy([subsub_group]) }.to raise_error(ArgumentError) expect { described_class.build_hierarchy([subsub_group]) }.to raise_error(ArgumentError)
end end
...@@ -91,7 +91,7 @@ describe GroupDescendant do ...@@ -91,7 +91,7 @@ describe GroupDescendant do
expected_hierarchy = { parent => { subgroup => subsub_group } } expected_hierarchy = { parent => { subgroup => subsub_group } }
# this does not raise in production, so stubbing it here. # this does not raise in production, so stubbing it here.
allow(Gitlab::Sentry).to receive(:track_exception) allow(Gitlab::Sentry).to receive(:track_and_raise_for_dev_exception)
expect(described_class.build_hierarchy([subsub_group])).to eq(expected_hierarchy) expect(described_class.build_hierarchy([subsub_group])).to eq(expected_hierarchy)
end end
......
...@@ -171,8 +171,7 @@ describe Upload do ...@@ -171,8 +171,7 @@ describe Upload do
it 'sends a message to Sentry' do it 'sends a message to Sentry' do
upload = create(:upload, :issuable_upload) upload = create(:upload, :issuable_upload)
expect(Gitlab::Sentry).to receive(:enabled?).and_return(true) expect(Gitlab::Sentry).to receive(:track_exception).with(instance_of(RuntimeError), upload.attributes)
expect(Raven).to receive(:capture_message).with("Upload file does not exist", extra: upload.attributes)
upload.exist? upload.exist?
end end
......
...@@ -46,7 +46,7 @@ describe 'GraphQL' do ...@@ -46,7 +46,7 @@ describe 'GraphQL' do
end end
it 'logs the exception in Sentry and continues with the request' do it 'logs the exception in Sentry and continues with the request' do
expect(Gitlab::Sentry).to receive(:track_exception).at_least(1).times expect(Gitlab::Sentry).to receive(:track_and_raise_for_dev_exception).at_least(1).times
expect(Gitlab::GraphqlLogger).to receive(:info) expect(Gitlab::GraphqlLogger).to receive(:info)
post_graphql(query, variables: {}) post_graphql(query, variables: {})
...@@ -146,7 +146,7 @@ describe 'GraphQL' do ...@@ -146,7 +146,7 @@ describe 'GraphQL' do
end end
it "logs a warning that the 'calls_gitaly' field declaration is missing" do it "logs a warning that the 'calls_gitaly' field declaration is missing" do
expect(Gitlab::Sentry).to receive(:track_exception).once expect(Gitlab::Sentry).to receive(:track_and_raise_for_dev_exception).once
post_graphql(query, current_user: user) post_graphql(query, current_user: user)
end end
......
...@@ -226,11 +226,11 @@ describe API::Helpers do ...@@ -226,11 +226,11 @@ describe API::Helpers do
describe '.handle_api_exception' do describe '.handle_api_exception' do
before do before do
allow_any_instance_of(self.class).to receive(:rack_response) allow_any_instance_of(self.class).to receive(:rack_response)
allow(Gitlab::Sentry).to receive(:enabled?).and_return(true)
stub_sentry_settings stub_sentry_settings
configure_sentry expect(Gitlab::Sentry).to receive(:sentry_dsn).and_return(Gitlab.config.sentry.dsn)
Gitlab::Sentry.configure
Raven.client.configuration.encoding = 'json' Raven.client.configuration.encoding = 'json'
end end
......
...@@ -60,10 +60,10 @@ describe Ci::ArchiveTraceService, '#execute' do ...@@ -60,10 +60,10 @@ describe Ci::ArchiveTraceService, '#execute' do
it 'increments Prometheus counter, sends crash report to Sentry and ignore an error for continuing to archive' do it 'increments Prometheus counter, sends crash report to Sentry and ignore an error for continuing to archive' do
expect(Gitlab::Sentry) expect(Gitlab::Sentry)
.to receive(:track_exception) .to receive(:track_and_raise_for_dev_exception)
.with(::Gitlab::Ci::Trace::ArchiveError, .with(::Gitlab::Ci::Trace::ArchiveError,
issue_url: 'https://gitlab.com/gitlab-org/gitlab-foss/issues/51502', issue_url: 'https://gitlab.com/gitlab-org/gitlab-foss/issues/51502',
extra: { job_id: job.id } ).once job_id: job.id).once
expect(Sidekiq.logger).to receive(:warn).with( expect(Sidekiq.logger).to receive(:warn).with(
class: ArchiveTraceWorker.name, class: ArchiveTraceWorker.name,
......
...@@ -528,7 +528,7 @@ describe Ci::CreatePipelineService do ...@@ -528,7 +528,7 @@ describe Ci::CreatePipelineService do
end end
it 'logs error' do it 'logs error' do
expect(Gitlab::Sentry).to receive(:track_acceptable_exception).and_call_original expect(Gitlab::Sentry).to receive(:track_exception).and_call_original
execute_service execute_service
end end
...@@ -613,7 +613,7 @@ describe Ci::CreatePipelineService do ...@@ -613,7 +613,7 @@ describe Ci::CreatePipelineService do
end end
it 'logs error' do it 'logs error' do
expect(Gitlab::Sentry).to receive(:track_acceptable_exception).and_call_original expect(Gitlab::Sentry).to receive(:track_exception).and_call_original
execute_service execute_service
end end
......
...@@ -51,8 +51,8 @@ describe Ci::PrepareBuildService do ...@@ -51,8 +51,8 @@ describe Ci::PrepareBuildService do
it 'drops the build and notifies Sentry' do it 'drops the build and notifies Sentry' do
expect(build).to receive(:drop).with(:unmet_prerequisites).once expect(build).to receive(:drop).with(:unmet_prerequisites).once
expect(Gitlab::Sentry).to receive(:track_acceptable_exception) expect(Gitlab::Sentry).to receive(:track_exception)
.with(instance_of(Kubeclient::HttpError), hash_including(extra: { build_id: build.id })) .with(instance_of(Kubeclient::HttpError), hash_including(build_id: build.id))
subject subject
end end
......
...@@ -514,8 +514,8 @@ module Ci ...@@ -514,8 +514,8 @@ module Ci
subject { execute(specific_runner, {}) } subject { execute(specific_runner, {}) }
it 'does drop the build and logs both failures' do it 'does drop the build and logs both failures' do
expect(Gitlab::Sentry).to receive(:track_acceptable_exception) expect(Gitlab::Sentry).to receive(:track_exception)
.with(anything, a_hash_including(extra: a_hash_including(build_id: pending_job.id))) .with(anything, a_hash_including(build_id: pending_job.id))
.twice .twice
.and_call_original .and_call_original
...@@ -540,8 +540,8 @@ module Ci ...@@ -540,8 +540,8 @@ module Ci
subject { execute(specific_runner, {}) } subject { execute(specific_runner, {}) }
it 'does drop the build and logs failure' do it 'does drop the build and logs failure' do
expect(Gitlab::Sentry).to receive(:track_acceptable_exception) expect(Gitlab::Sentry).to receive(:track_exception)
.with(anything, a_hash_including(extra: a_hash_including(build_id: pending_job.id))) .with(anything, a_hash_including(build_id: pending_job.id))
.once .once
.and_call_original .and_call_original
......
...@@ -11,20 +11,10 @@ shared_examples 'logs kubernetes errors' do ...@@ -11,20 +11,10 @@ shared_examples 'logs kubernetes errors' do
} }
end end
let(:logger_hash) do
error_hash.merge(
exception: error_name,
message: error_message,
backtrace: instance_of(Array)
)
end
it 'logs into kubernetes.log and Sentry' do it 'logs into kubernetes.log and Sentry' do
expect(service.send(:logger)).to receive(:error).with(hash_including(logger_hash)) expect(Gitlab::Sentry).to receive(:track_exception).with(
expect(Gitlab::Sentry).to receive(:track_acceptable_exception).with(
error, error,
extra: hash_including(error_hash) hash_including(error_hash)
) )
service.execute service.execute
......
...@@ -63,7 +63,7 @@ describe Ci::ArchiveTracesCronWorker do ...@@ -63,7 +63,7 @@ describe Ci::ArchiveTracesCronWorker do
let!(:build) { create(:ci_build, :success, :trace_live, finished_at: finished_at) } let!(:build) { create(:ci_build, :success, :trace_live, finished_at: finished_at) }
before do before do
allow(Gitlab::Sentry).to receive(:track_exception) allow(Gitlab::Sentry).to receive(:track_and_raise_for_dev_exception)
allow_any_instance_of(Gitlab::Ci::Trace).to receive(:archive!).and_raise('Unexpected error') allow_any_instance_of(Gitlab::Ci::Trace).to receive(:archive!).and_raise('Unexpected error')
end end
......
...@@ -43,10 +43,10 @@ describe RunPipelineScheduleWorker do ...@@ -43,10 +43,10 @@ describe RunPipelineScheduleWorker do
allow(Ci::CreatePipelineService).to receive(:new) { raise ActiveRecord::StatementInvalid } allow(Ci::CreatePipelineService).to receive(:new) { raise ActiveRecord::StatementInvalid }
expect(Gitlab::Sentry) expect(Gitlab::Sentry)
.to receive(:track_exception) .to receive(:track_and_raise_for_dev_exception)
.with(ActiveRecord::StatementInvalid, .with(ActiveRecord::StatementInvalid,
issue_url: 'https://gitlab.com/gitlab-org/gitlab-foss/issues/41231', issue_url: 'https://gitlab.com/gitlab-org/gitlab-foss/issues/41231',
extra: { schedule_id: pipeline_schedule.id } ).once schedule_id: pipeline_schedule.id).once
end end
it 'increments Prometheus counter' do it 'increments Prometheus counter' do
......
...@@ -30,8 +30,8 @@ describe StuckCiJobsWorker do ...@@ -30,8 +30,8 @@ describe StuckCiJobsWorker do
it "does drop the job and logs the reason" do it "does drop the job and logs the reason" do
job.update_columns(yaml_variables: '[{"key" => "value"}]') job.update_columns(yaml_variables: '[{"key" => "value"}]')
expect(Gitlab::Sentry).to receive(:track_acceptable_exception) expect(Gitlab::Sentry).to receive(:track_exception)
.with(anything, a_hash_including(extra: a_hash_including(build_id: job.id))) .with(anything, a_hash_including(build_id: job.id))
.once .once
.and_call_original .and_call_original
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment