Commit 27f5e271 authored by Jan Provaznik's avatar Jan Provaznik

Merge branch 'db-testing-catch-system-exceptions' into 'master'

Rescue low-level errors during migration testing

See merge request gitlab-org/gitlab!79445
parents d1c38b5b d668c0d9
...@@ -15,30 +15,26 @@ module Gitlab ...@@ -15,30 +15,26 @@ module Gitlab
end end
def observe(version:, name:, connection:, &block) def observe(version:, name:, connection:, &block)
observation = Observation.new(version, name) observation = Observation.new(version: version, name: name, success: false)
observation.success = true
observers = observer_classes.map { |c| c.new(observation, @result_dir, connection) } observers = observer_classes.map { |c| c.new(observation, @result_dir, connection) }
exception = nil
on_each_observer(observers) { |observer| observer.before } on_each_observer(observers) { |observer| observer.before }
observation.walltime = Benchmark.realtime do start = Process.clock_gettime(Process::CLOCK_MONOTONIC)
yield yield
rescue StandardError => e
exception = e observation.success = true
observation.success = false
end observation
ensure
observation.walltime = Process.clock_gettime(Process::CLOCK_MONOTONIC) - start
on_each_observer(observers) { |observer| observer.after } on_each_observer(observers) { |observer| observer.after }
on_each_observer(observers) { |observer| observer.record } on_each_observer(observers) { |observer| observer.record }
record_observation(observation) record_observation(observation)
raise exception if exception
observation
end end
private private
......
...@@ -10,7 +10,8 @@ module Gitlab ...@@ -10,7 +10,8 @@ module Gitlab
:walltime, :walltime,
:success, :success,
:total_database_size_change, :total_database_size_change,
:query_statistics :query_statistics,
keyword_init: true
) )
end end
end end
......
...@@ -66,58 +66,46 @@ RSpec.describe Gitlab::Database::Migrations::Instrumentation do ...@@ -66,58 +66,46 @@ RSpec.describe Gitlab::Database::Migrations::Instrumentation do
context 'on successful execution' do context 'on successful execution' do
subject { described_class.new(result_dir: result_dir).observe(version: migration_version, name: migration_name, connection: connection) {} } subject { described_class.new(result_dir: result_dir).observe(version: migration_version, name: migration_name, connection: connection) {} }
it 'records walltime' do it 'records a valid observation', :aggregate_failures do
expect(subject.walltime).not_to be_nil expect(subject.walltime).not_to be_nil
end
it 'records success' do
expect(subject.success).to be_truthy expect(subject.success).to be_truthy
end
it 'records the migration version' do
expect(subject.version).to eq(migration_version) expect(subject.version).to eq(migration_version)
end
it 'records the migration name' do
expect(subject.name).to eq(migration_name) expect(subject.name).to eq(migration_name)
end end
end end
context 'upon failure' do context 'upon failure' do
subject { described_class.new(result_dir: result_dir).observe(version: migration_version, name: migration_name, connection: connection) { raise 'something went wrong' } } where(exception: ['something went wrong', SystemStackError, Interrupt])
with_them do
let(:instance) { described_class.new(result_dir: result_dir) }
subject(:observe) { instance.observe(version: migration_version, name: migration_name, connection: connection) { raise exception } }
it 'raises the exception' do it 'raises the exception' do
expect { subject }.to raise_error(/something went wrong/) expect { observe }.to raise_error(exception)
end end
context 'retrieving observations' do context 'retrieving observations' do
subject { instance.observations.first } subject { instance.observations.first }
before do before do
instance.observe(version: migration_version, name: migration_name, connection: connection) { raise 'something went wrong' } observe
rescue StandardError # rubocop:disable Lint/RescueException
# ignore rescue Exception
# rubocop:enable Lint/RescueException
# ignore (we expect this exception)
end end
let(:instance) { described_class.new(result_dir: result_dir) } it 'records a valid observation', :aggregate_failures do
it 'records walltime' do
expect(subject.walltime).not_to be_nil expect(subject.walltime).not_to be_nil
end
it 'records failure' do
expect(subject.success).to be_falsey expect(subject.success).to be_falsey
end
it 'records the migration version' do
expect(subject.version).to eq(migration_version) expect(subject.version).to eq(migration_version)
end
it 'records the migration name' do
expect(subject.name).to eq(migration_name) expect(subject.name).to eq(migration_name)
end end
end end
end end
end
context 'sequence of migrations with failures' do context 'sequence of migrations with failures' do
subject { described_class.new(result_dir: result_dir) } subject { described_class.new(result_dir: result_dir) }
......
...@@ -5,7 +5,7 @@ RSpec.describe Gitlab::Database::Migrations::Observers::QueryDetails do ...@@ -5,7 +5,7 @@ RSpec.describe Gitlab::Database::Migrations::Observers::QueryDetails do
subject { described_class.new(observation, directory_path, connection) } subject { described_class.new(observation, directory_path, connection) }
let(:connection) { ActiveRecord::Migration.connection } let(:connection) { ActiveRecord::Migration.connection }
let(:observation) { Gitlab::Database::Migrations::Observation.new(migration_version, migration_name) } let(:observation) { Gitlab::Database::Migrations::Observation.new(version: migration_version, name: migration_name) }
let(:query) { "select date_trunc('day', $1::timestamptz) + $2 * (interval '1 hour')" } let(:query) { "select date_trunc('day', $1::timestamptz) + $2 * (interval '1 hour')" }
let(:query_binds) { [Time.current, 3] } let(:query_binds) { [Time.current, 3] }
let(:directory_path) { Dir.mktmpdir } let(:directory_path) { Dir.mktmpdir }
......
...@@ -4,7 +4,7 @@ require 'spec_helper' ...@@ -4,7 +4,7 @@ require 'spec_helper'
RSpec.describe Gitlab::Database::Migrations::Observers::QueryLog do RSpec.describe Gitlab::Database::Migrations::Observers::QueryLog do
subject { described_class.new(observation, directory_path, connection) } subject { described_class.new(observation, directory_path, connection) }
let(:observation) { Gitlab::Database::Migrations::Observation.new(migration_version, migration_name) } let(:observation) { Gitlab::Database::Migrations::Observation.new(version: migration_version, name: migration_name) }
let(:connection) { ActiveRecord::Migration.connection } let(:connection) { ActiveRecord::Migration.connection }
let(:query) { 'select 1' } let(:query) { 'select 1' }
let(:directory_path) { Dir.mktmpdir } let(:directory_path) { Dir.mktmpdir }
......
...@@ -5,7 +5,7 @@ RSpec.describe Gitlab::Database::Migrations::Observers::TransactionDuration do ...@@ -5,7 +5,7 @@ RSpec.describe Gitlab::Database::Migrations::Observers::TransactionDuration do
subject(:transaction_duration_observer) { described_class.new(observation, directory_path, connection) } subject(:transaction_duration_observer) { described_class.new(observation, directory_path, connection) }
let(:connection) { ActiveRecord::Migration.connection } let(:connection) { ActiveRecord::Migration.connection }
let(:observation) { Gitlab::Database::Migrations::Observation.new(migration_version, migration_name) } let(:observation) { Gitlab::Database::Migrations::Observation.new(version: migration_version, name: migration_name) }
let(:directory_path) { Dir.mktmpdir } let(:directory_path) { Dir.mktmpdir }
let(:log_file) { "#{directory_path}/#{migration_version}_#{migration_name}-transaction-duration.json" } let(:log_file) { "#{directory_path}/#{migration_version}_#{migration_name}-transaction-duration.json" }
let(:transaction_duration) { Gitlab::Json.parse(File.read(log_file)) } let(:transaction_duration) { Gitlab::Json.parse(File.read(log_file)) }
......
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