276 lines
8.5 KiB
Ruby
276 lines
8.5 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
require 'spec_helper'
|
|
|
|
RSpec.describe ::Gitlab::Ci::Pipeline::Logger, feature_category: :continuous_integration do
|
|
let_it_be(:project) { build_stubbed(:project) }
|
|
let_it_be(:pipeline) { build_stubbed(:ci_pipeline, project: project) }
|
|
|
|
subject(:logger) { described_class.new(project: project) }
|
|
|
|
describe '#log_when' do
|
|
it 'stores blocks for later evaluation' do
|
|
logger.log_when { |obs| true }
|
|
|
|
expect(logger.send(:log_conditions).first).to be_a(Proc)
|
|
end
|
|
end
|
|
|
|
describe '#instrument' do
|
|
it "returns the block's value" do
|
|
expect(logger.instrument(:expensive_operation) { 123 }).to eq(123)
|
|
end
|
|
|
|
it 'records durations of instrumented operations' do
|
|
logger.instrument(:expensive_operation) { 123 }
|
|
|
|
expected_data = {
|
|
'expensive_operation_duration_s' => {
|
|
'count' => 1,
|
|
'max' => a_kind_of(Numeric),
|
|
'sum' => a_kind_of(Numeric)
|
|
}
|
|
}
|
|
expect(logger.observations_hash).to match(a_hash_including(expected_data))
|
|
end
|
|
|
|
it 'raises an error when block is not provided' do
|
|
expect { logger.instrument(:expensive_operation) }
|
|
.to raise_error(ArgumentError, 'block not given')
|
|
end
|
|
|
|
context 'when once: true' do
|
|
it 'logs only one observation' do
|
|
logger.instrument(:expensive_operation, once: true) { 123 }
|
|
logger.instrument(:expensive_operation, once: true) { 123 }
|
|
|
|
expected_data = {
|
|
'expensive_operation_duration_s' => a_kind_of(Numeric)
|
|
}
|
|
expect(logger.observations_hash).to match(a_hash_including(expected_data))
|
|
end
|
|
end
|
|
end
|
|
|
|
describe '#instrument_once_with_sql', :request_store do
|
|
subject(:instrument_once_with_sql) do
|
|
logger.instrument_once_with_sql(:expensive_operation, &operation)
|
|
end
|
|
|
|
def expected_data(count:, db_count: nil)
|
|
database_name = Ci::ApplicationRecord.connection.pool.db_config.name
|
|
|
|
total_db_count = count * db_count if db_count
|
|
|
|
{
|
|
"expensive_operation_duration_s" => a_kind_of(Numeric),
|
|
"expensive_operation_db_count" => total_db_count || a_kind_of(Numeric),
|
|
"expensive_operation_db_primary_count" => a_kind_of(Numeric),
|
|
"expensive_operation_db_primary_duration_s" => a_kind_of(Numeric),
|
|
"expensive_operation_db_#{database_name}_count" => a_kind_of(Numeric),
|
|
"expensive_operation_db_#{database_name}_duration_s" => a_kind_of(Numeric)
|
|
}
|
|
end
|
|
|
|
context 'with a single query' do
|
|
let(:operation) { -> { Ci::Pipeline.count } }
|
|
|
|
it { is_expected.to eq(operation.call) }
|
|
|
|
it 'includes SQL metrics' do
|
|
instrument_once_with_sql
|
|
|
|
expect(logger.observations_hash)
|
|
.to match(a_hash_including(expected_data(count: 1, db_count: 1)))
|
|
end
|
|
end
|
|
|
|
context 'with multiple queries' do
|
|
let(:operation) { -> { Ci::Build.count + Ci::Bridge.count } }
|
|
|
|
it { is_expected.to eq(operation.call) }
|
|
|
|
it 'includes SQL metrics' do
|
|
instrument_once_with_sql
|
|
|
|
expect(logger.observations_hash)
|
|
.to match(a_hash_including(expected_data(count: 1, db_count: 2)))
|
|
end
|
|
end
|
|
|
|
context 'when there are not SQL operations' do
|
|
let(:operation) { -> { 123 } }
|
|
|
|
it { is_expected.to eq(operation.call) }
|
|
|
|
it 'does not include SQL metrics' do
|
|
instrument_once_with_sql
|
|
|
|
expect(logger.observations_hash.keys)
|
|
.to match_array(['expensive_operation_duration_s'])
|
|
end
|
|
end
|
|
end
|
|
|
|
describe '#observe' do
|
|
it 'records durations of observed operations' do
|
|
expect(logger.observe(:pipeline_creation_duration_s, 30)).to be_truthy
|
|
|
|
expected_data = {
|
|
'pipeline_creation_duration_s' => {
|
|
'sum' => 30, 'count' => 1, 'max' => 30
|
|
}
|
|
}
|
|
expect(logger.observations_hash).to match(a_hash_including(expected_data))
|
|
end
|
|
|
|
context 'when once: true' do
|
|
it 'records the latest observation' do
|
|
expect(logger.observe(:pipeline_creation_duration_s, 20, once: true)).to be_truthy
|
|
expect(logger.observe(:pipeline_creation_duration_s, 30, once: true)).to be_truthy
|
|
|
|
expected_data = {
|
|
'pipeline_creation_duration_s' => 30
|
|
}
|
|
expect(logger.observations_hash).to match(a_hash_including(expected_data))
|
|
end
|
|
|
|
it 'logs data as expected' do
|
|
expect(logger.observe(:pipeline_creation_duration_s, 30, once: true)).to be_truthy
|
|
expect(logger.observe(:pipeline_operation_x_duration_s, 20)).to be_truthy
|
|
expect(logger.observe(:pipeline_operation_x_duration_s, 20)).to be_truthy
|
|
|
|
expected_data = {
|
|
'pipeline_creation_duration_s' => 30,
|
|
'pipeline_operation_x_duration_s' => {
|
|
'sum' => 40, 'count' => 2, 'max' => 20
|
|
}
|
|
}
|
|
expect(logger.observations_hash).to match(a_hash_including(expected_data))
|
|
end
|
|
end
|
|
end
|
|
|
|
describe '#commit' do
|
|
subject(:commit) { logger.commit(pipeline: pipeline, caller: 'source') }
|
|
|
|
before do
|
|
stub_feature_flags(ci_pipeline_creation_logger: flag)
|
|
allow(logger).to receive(:current_monotonic_time) { Time.current.to_i }
|
|
|
|
logger.instrument(:pipeline_save) { travel(60.seconds) }
|
|
logger.observe(:pipeline_creation_duration_s, 30)
|
|
logger.observe(:pipeline_creation_duration_s, 10)
|
|
end
|
|
|
|
context 'when the feature flag is enabled' do
|
|
let(:flag) { true }
|
|
|
|
let(:expected_data) do
|
|
{
|
|
'correlation_id' => a_kind_of(String),
|
|
'meta.project' => project.full_path,
|
|
'meta.root_namespace' => project.root_namespace.full_path,
|
|
'class' => described_class.name.to_s,
|
|
'pipeline_id' => pipeline.id,
|
|
'pipeline_persisted' => true,
|
|
'project_id' => project.id,
|
|
'pipeline_creation_service_duration_s' => a_kind_of(Numeric),
|
|
'pipeline_creation_caller' => 'source',
|
|
'pipeline_source' => pipeline.source,
|
|
'pipeline_save_duration_s' => {
|
|
'sum' => 60, 'count' => 1, 'max' => 60
|
|
},
|
|
'pipeline_creation_duration_s' => {
|
|
'sum' => 40, 'count' => 2, 'max' => 30
|
|
}
|
|
}
|
|
end
|
|
|
|
it 'logs to application.json' do
|
|
expect(Gitlab::AppJsonLogger)
|
|
.to receive(:info)
|
|
.with(a_hash_including(expected_data))
|
|
.and_call_original
|
|
|
|
expect(commit).to be_truthy
|
|
end
|
|
|
|
context 'with log conditions' do
|
|
it 'does not log when the conditions are false' do
|
|
logger.log_when { |_obs| false }
|
|
|
|
expect(Gitlab::AppJsonLogger).not_to receive(:info)
|
|
|
|
expect(commit).to be_falsey
|
|
end
|
|
|
|
it 'logs when a condition is true' do
|
|
logger.log_when { |_obs| true }
|
|
logger.log_when { |_obs| false }
|
|
|
|
expect(Gitlab::AppJsonLogger)
|
|
.to receive(:info)
|
|
.with(a_hash_including(expected_data))
|
|
.and_call_original
|
|
|
|
expect(commit).to be_truthy
|
|
end
|
|
|
|
context 'with unexistent observations in condition' do
|
|
it 'does not commit the log' do
|
|
logger.log_when do |observations|
|
|
value = observations['non_existent_value']
|
|
next false unless value
|
|
|
|
value > 0
|
|
end
|
|
|
|
expect(Gitlab::AppJsonLogger).not_to receive(:info)
|
|
|
|
expect(commit).to be_falsey
|
|
end
|
|
end
|
|
end
|
|
|
|
context 'when project is not passed and pipeline is not persisted' do
|
|
let(:project) {}
|
|
let(:pipeline) { build(:ci_pipeline) }
|
|
|
|
let(:expected_data) do
|
|
{
|
|
'class' => described_class.name.to_s,
|
|
'pipeline_persisted' => false,
|
|
'pipeline_creation_service_duration_s' => a_kind_of(Numeric),
|
|
'pipeline_creation_caller' => 'source',
|
|
'pipeline_save_duration_s' => {
|
|
'sum' => 60, 'count' => 1, 'max' => 60
|
|
},
|
|
'pipeline_creation_duration_s' => {
|
|
'sum' => 40, 'count' => 2, 'max' => 30
|
|
}
|
|
}
|
|
end
|
|
|
|
it 'logs to application.json' do
|
|
expect(Gitlab::AppJsonLogger)
|
|
.to receive(:info)
|
|
.with(a_hash_including(expected_data))
|
|
.and_call_original
|
|
|
|
expect(commit).to be_truthy
|
|
end
|
|
end
|
|
end
|
|
|
|
context 'when the feature flag is disabled' do
|
|
let(:flag) { false }
|
|
|
|
it 'does not log' do
|
|
expect(Gitlab::AppJsonLogger).not_to receive(:info)
|
|
|
|
expect(commit).to be_falsey
|
|
end
|
|
end
|
|
end
|
|
end
|