mirror of
https://gitlab.com/gitlab-org/gitlab-foss.git
synced 2025-08-20 14:11:11 +00:00
87 lines
2.3 KiB
Ruby
87 lines
2.3 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
module BulkImports
|
|
class NetworkError < Error
|
|
TRACKER_COUNTER_KEY = 'bulk_imports/%{entity_id}/%{stage}/%{tracker_id}/network_error/%{error}'
|
|
ENTITY_COUNTER_KEY = 'bulk_imports/%{entity_id}/network_error/%{error}'
|
|
|
|
NO_SPACE_LEFT_EXCEPTION = Errno::ENOSPC
|
|
DECOMPRESSION_FAILURE_EXCEPTION = Zlib::Error
|
|
|
|
EXCEPTIONS_RETRY_DELAY = {
|
|
NO_SPACE_LEFT_EXCEPTION => 120,
|
|
DECOMPRESSION_FAILURE_EXCEPTION => 60
|
|
}.freeze
|
|
|
|
RETRIABLE_EXCEPTIONS = Gitlab::HTTP::HTTP_TIMEOUT_ERRORS + EXCEPTIONS_RETRY_DELAY.keys + [
|
|
EOFError, SocketError, OpenSSL::SSL::SSLError, OpenSSL::OpenSSLError,
|
|
Errno::ECONNRESET, Errno::ECONNREFUSED, Errno::EHOSTUNREACH, Errno::ENETUNREACH
|
|
].freeze
|
|
|
|
RETRIABLE_HTTP_CODES = [429, 500, 502, 503, 504].freeze
|
|
|
|
DEFAULT_RETRY_DELAY_SECONDS = 30
|
|
|
|
MAX_RETRIABLE_COUNT = 10
|
|
|
|
attr_reader :response
|
|
|
|
def initialize(message = nil, response: nil)
|
|
raise ArgumentError, 'message or response required' if message.blank? && response.blank?
|
|
|
|
super(message)
|
|
|
|
@response = response
|
|
end
|
|
|
|
def retriable?(object)
|
|
if retriable_exception? || retriable_http_code?
|
|
increment(object) <= MAX_RETRIABLE_COUNT
|
|
else
|
|
false
|
|
end
|
|
end
|
|
|
|
def retry_delay
|
|
if response&.code == 429
|
|
response.headers.fetch('Retry-After', DEFAULT_RETRY_DELAY_SECONDS).to_i
|
|
else
|
|
EXCEPTIONS_RETRY_DELAY[cause&.class] || DEFAULT_RETRY_DELAY_SECONDS
|
|
end.seconds
|
|
end
|
|
|
|
private
|
|
|
|
def retriable_exception?
|
|
RETRIABLE_EXCEPTIONS.include?(cause&.class)
|
|
end
|
|
|
|
def retriable_http_code?
|
|
RETRIABLE_HTTP_CODES.include?(response&.code.to_i)
|
|
end
|
|
|
|
def increment(object)
|
|
key = object.is_a?(BulkImports::Entity) ? entity_cache_key(object) : tracker_cache_key(object)
|
|
|
|
Gitlab::Cache::Import::Caching.increment(key)
|
|
end
|
|
|
|
def tracker_cache_key(tracker)
|
|
TRACKER_COUNTER_KEY % {
|
|
stage: tracker.stage,
|
|
tracker_id: tracker.id,
|
|
entity_id: tracker.entity.id,
|
|
error: cause.class.name
|
|
}
|
|
end
|
|
|
|
def entity_cache_key(entity)
|
|
ENTITY_COUNTER_KEY % {
|
|
import_id: entity.bulk_import_id,
|
|
entity_id: entity.id,
|
|
error: cause.class.name
|
|
}
|
|
end
|
|
end
|
|
end
|