Commit 3a77b9d2 authored by Matthias Käppler's avatar Matthias Käppler

Merge branch 'make-lifecycle-hooks-to-be-fatal' into 'master'

Make LifecycleEvents exceptions to be fatal

See merge request gitlab-org/gitlab!52881
parents e8525082 f6ea5406
---
title: Make LifecycleEvents exceptions to be fatal
merge_request: 52881
author:
type: fixed
# frozen_string_literal: true # frozen_string_literal: true
require_relative '../utils' # Gitlab::Utils
module Gitlab module Gitlab
module Cluster module Cluster
# #
...@@ -64,6 +66,10 @@ module Gitlab ...@@ -64,6 +66,10 @@ module Gitlab
# Blocks will be executed in the order in which they are registered. # Blocks will be executed in the order in which they are registered.
# #
class LifecycleEvents class LifecycleEvents
FatalError = Class.new(Exception) # rubocop:disable Lint/InheritException
USE_FATAL_LIFECYCLE_EVENTS = Gitlab::Utils.to_boolean(ENV.fetch('GITLAB_FATAL_LIFECYCLE_EVENTS', 'true'))
class << self class << self
# #
# Hook registration methods (called from initializers) # Hook registration methods (called from initializers)
...@@ -111,24 +117,24 @@ module Gitlab ...@@ -111,24 +117,24 @@ module Gitlab
# Lifecycle integration methods (called from unicorn.rb, puma.rb, etc.) # Lifecycle integration methods (called from unicorn.rb, puma.rb, etc.)
# #
def do_worker_start def do_worker_start
call(@worker_start_hooks) call(:worker_start_hooks, @worker_start_hooks)
end end
def do_before_fork def do_before_fork
call(@before_fork_hooks) call(:before_fork_hooks, @before_fork_hooks)
end end
def do_before_graceful_shutdown def do_before_graceful_shutdown
call(@master_blackout_period) call(:master_blackout_period, @master_blackout_period)
blackout_seconds = ::Settings.shutdown.blackout_seconds.to_i blackout_seconds = ::Settings.shutdown.blackout_seconds.to_i
sleep(blackout_seconds) if blackout_seconds > 0 sleep(blackout_seconds) if blackout_seconds > 0
call(@master_graceful_shutdown) call(:master_graceful_shutdown, @master_graceful_shutdown)
end end
def do_before_master_restart def do_before_master_restart
call(@master_restart_hooks) call(:master_restart_hooks, @master_restart_hooks)
end end
# DEPRECATED # DEPRECATED
...@@ -143,8 +149,18 @@ module Gitlab ...@@ -143,8 +149,18 @@ module Gitlab
private private
def call(hooks) def call(name, hooks)
hooks&.each(&:call) return unless hooks
hooks.each do |hook|
hook.call
rescue => e
Gitlab::ErrorTracking.track_exception(e, type: 'LifecycleEvents', hook: hook)
warn("ERROR: The hook #{name} failed with exception (#{e.class}) \"#{e.message}\".")
# we consider lifecycle hooks to be fatal errors
raise FatalError, e if USE_FATAL_LIFECYCLE_EVENTS
end
end end
def in_clustered_environment? def in_clustered_environment?
......
# frozen_string_literal: true
require 'fast_spec_helper'
require 'rspec-parameterized'
RSpec.describe Gitlab::Cluster::LifecycleEvents do
# we create a new instance to ensure that we do not touch existing hooks
let(:replica) { Class.new(described_class) }
context 'hooks execution' do
using RSpec::Parameterized::TableSyntax
where(:method, :hook_names) do
:do_worker_start | %i[worker_start_hooks]
:do_before_fork | %i[before_fork_hooks]
:do_before_graceful_shutdown | %i[master_blackout_period master_graceful_shutdown]
:do_before_master_restart | %i[master_restart_hooks]
end
before do
# disable blackout period to speed-up tests
stub_config(shutdown: { blackout_seconds: 0 })
end
with_them do
subject { replica.public_send(method) }
it 'executes all hooks' do
hook_names.each do |hook_name|
hook = double
replica.instance_variable_set(:"@#{hook_name}", [hook])
# ensure that proper hooks are called
expect(hook).to receive(:call)
expect(replica).to receive(:call).with(hook_name, anything).and_call_original
end
subject
end
end
end
describe '#call' do
let(:name) { :my_hooks }
subject { replica.send(:call, name, hooks) }
context 'when many hooks raise exception' do
let(:hooks) do
[
-> { raise 'Exception A' },
-> { raise 'Exception B' }
]
end
context 'USE_FATAL_LIFECYCLE_EVENTS is set to default' do
it 'only first hook is executed and is fatal' do
expect(hooks[0]).to receive(:call).and_call_original
expect(hooks[1]).not_to receive(:call)
expect(Gitlab::ErrorTracking).to receive(:track_exception).and_call_original
expect(replica).to receive(:warn).with('ERROR: The hook my_hooks failed with exception (RuntimeError) "Exception A".')
expect { subject }.to raise_error(described_class::FatalError, 'Exception A')
end
end
context 'when USE_FATAL_LIFECYCLE_EVENTS is disabled' do
before do
stub_const('Gitlab::Cluster::LifecycleEvents::USE_FATAL_LIFECYCLE_EVENTS', false)
end
it 'many hooks are executed and all exceptions are logged' do
expect(hooks[0]).to receive(:call).and_call_original
expect(hooks[1]).to receive(:call).and_call_original
expect(Gitlab::ErrorTracking).to receive(:track_exception).twice.and_call_original
expect(replica).to receive(:warn).twice.and_call_original
expect { subject }.not_to raise_error
end
end
end
end
end
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