debian-mirror-gitlab/spec/lib/gitlab/memory/reporter_spec.rb

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

194 lines
5.8 KiB
Ruby
Raw Normal View History

2023-03-04 22:38:38 +05:30
# frozen_string_literal: true
require 'spec_helper'
RSpec.describe Gitlab::Memory::Reporter, :aggregate_failures, feature_category: :application_performance do
let(:fake_report) do
Class.new do
def name
'fake_report'
end
def active?
true
end
def run(writer)
writer << 'I ran'
end
end
end
let(:logger) { instance_double(::Logger) }
let(:report) { fake_report.new }
after do
FileUtils.rm_rf(reports_path)
end
2023-03-17 16:20:25 +05:30
describe '#run_report', time_travel_to: '2020-02-02 10:30:45 +0000' do
2023-03-04 22:38:38 +05:30
let(:report_duration_counter) { instance_double(::Prometheus::Client::Counter) }
let(:file_size) { 1_000_000 }
let(:report_file) { "#{reports_path}/fake_report.2020-02-02.10:30:45:000.worker_1.abc123.gz" }
2023-03-17 16:20:25 +05:30
let(:output) { File.read(report_file) }
2023-03-04 22:38:38 +05:30
before do
2023-03-17 16:20:25 +05:30
stub_const('Gitlab::Memory::Reporter::COMPRESS_CMD', %w[cat])
2023-03-04 22:38:38 +05:30
allow(SecureRandom).to receive(:uuid).and_return('abc123')
allow(Gitlab::Metrics).to receive(:counter).and_return(report_duration_counter)
allow(report_duration_counter).to receive(:increment)
allow(::Prometheus::PidProvider).to receive(:worker_id).and_return('worker_1')
allow(File).to receive(:size).with(report_file).and_return(file_size)
allow(logger).to receive(:info)
end
shared_examples 'runs and stores reports' do
it 'runs the given report and returns true' do
expect(reporter.run_report(report)).to be(true)
2023-03-17 16:20:25 +05:30
expect(output).to eq('I ran')
2023-03-04 22:38:38 +05:30
end
it 'logs start and finish event' do
expect(logger).to receive(:info).ordered.with(
hash_including(
message: 'started',
pid: Process.pid,
worker_id: 'worker_1',
perf_report_worker_uuid: 'abc123',
perf_report: 'fake_report'
))
expect(logger).to receive(:info).ordered.with(
hash_including(
:duration_s,
:cpu_s,
perf_report_file: report_file,
perf_report_size_bytes: file_size,
message: 'finished',
pid: Process.pid,
worker_id: 'worker_1',
perf_report_worker_uuid: 'abc123',
perf_report: 'fake_report'
))
reporter.run_report(report)
end
it 'increments Prometheus duration counter' do
expect(report_duration_counter).to receive(:increment).with({ report: 'fake_report' }, an_instance_of(Float))
reporter.run_report(report)
end
context 'when the report returns invalid file path' do
before do
allow(File).to receive(:size).with(report_file).and_raise(Errno::ENOENT)
end
it 'logs `0` as `perf_report_size_bytes`' do
expect(logger).to receive(:info).ordered.with(
hash_including(message: 'started')
)
expect(logger).to receive(:info).ordered.with(
hash_including(message: 'finished', perf_report_size_bytes: 0)
)
reporter.run_report(report)
end
end
context 'when an error occurs' do
2023-03-17 16:20:25 +05:30
shared_examples 'handles errors gracefully' do
it 'logs the error and returns false' do
expect(logger).to receive(:info).ordered.with(hash_including(message: 'started'))
expect(logger).to receive(:error).ordered.with(
hash_including(
message: 'failed', error: match(error_message)
))
expect(reporter.run_report(report)).to be(false)
end
context 'when compression process is still running' do
it 'terminates the process' do
allow(logger).to receive(:info)
allow(logger).to receive(:error)
2023-03-04 22:38:38 +05:30
2023-03-17 16:20:25 +05:30
expect(Gitlab::ProcessManagement).to receive(:signal).with(an_instance_of(Integer), :KILL)
2023-03-04 22:38:38 +05:30
2023-03-17 16:20:25 +05:30
reporter.run_report(report)
end
end
2023-03-04 22:38:38 +05:30
end
2023-03-17 16:20:25 +05:30
context 'when cause was an error being raised' do
let(:error_message) { 'report failed' }
2023-03-04 22:38:38 +05:30
2023-03-17 16:20:25 +05:30
before do
allow(report).to receive(:run).and_raise(RuntimeError.new('report failed'))
end
2023-03-04 22:38:38 +05:30
2023-03-17 16:20:25 +05:30
it_behaves_like 'handles errors gracefully'
2023-03-04 22:38:38 +05:30
end
2023-03-17 16:20:25 +05:30
context 'when cause was compression command failing' do
let(:error_message) { "StandardError: exit 1: cat:" }
2023-03-04 22:38:38 +05:30
2023-03-17 16:20:25 +05:30
before do
stub_const('Gitlab::Memory::Reporter::COMPRESS_CMD', %w[cat --bad-flag])
2023-03-04 22:38:38 +05:30
end
2023-03-17 16:20:25 +05:30
it_behaves_like 'handles errors gracefully'
2023-03-04 22:38:38 +05:30
end
end
context 'when a report is disabled' do
it 'does nothing and returns false' do
expect(report).to receive(:active?).and_return(false)
expect(report).not_to receive(:run)
expect(logger).not_to receive(:info)
expect(report_duration_counter).not_to receive(:increment)
reporter.run_report(report)
end
end
end
context 'when reports path is specified directly' do
let(:reports_path) { Dir.mktmpdir }
subject(:reporter) { described_class.new(reports_path: reports_path, logger: logger) }
it_behaves_like 'runs and stores reports'
end
context 'when reports path is specified via environment' do
let(:reports_path) { Dir.mktmpdir }
subject(:reporter) { described_class.new(logger: logger) }
before do
stub_env('GITLAB_DIAGNOSTIC_REPORTS_PATH', reports_path)
end
it_behaves_like 'runs and stores reports'
end
context 'when reports path is not specified' do
let(:reports_path) { reporter.reports_path }
subject(:reporter) { described_class.new(logger: logger) }
it 'defaults to a temporary location' do
expect(reports_path).not_to be_empty
end
it_behaves_like 'runs and stores reports'
end
end
end