mirror of
https://gitlab.com/gitlab-org/gitlab-foss.git
synced 2025-07-29 12:00:32 +00:00
246 lines
8.2 KiB
Ruby
246 lines
8.2 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
require 'fast_spec_helper'
|
|
require_relative '../../scripts/failed_tests'
|
|
|
|
RSpec.describe FailedTests do
|
|
let(:report_file) { 'spec/fixtures/scripts/test_report.json' }
|
|
let(:options) { described_class::DEFAULT_OPTIONS.merge(previous_tests_report_path: report_file) }
|
|
let(:frontend_failure_path) { 'path/to/fail_file_spec.js' }
|
|
let(:failure_path) { 'path/to/fail_file_spec.rb' }
|
|
let(:other_failure_path) { 'path/to/fail_file_spec_2.rb' }
|
|
let(:output_directory) { described_class::DEFAULT_OPTIONS[:output_directory] }
|
|
let(:user_spec) { 'spec/models/user_spec.rb' }
|
|
let(:project_spec) { 'spec/models/project_spec.rb' }
|
|
let(:issue_spec) { 'spec/models/issue_spec.rb' }
|
|
let(:file_contents_as_json) do
|
|
{
|
|
'suites' => [
|
|
{
|
|
'failed_count' => 1,
|
|
'name' => 'rspec unit pg16 10/12',
|
|
'test_cases' => [
|
|
{
|
|
'status' => 'failed',
|
|
'file' => failure_path
|
|
}
|
|
]
|
|
},
|
|
{
|
|
'failed_count' => 1,
|
|
'name' => 'rspec-ee unit pg16',
|
|
'test_cases' => [
|
|
{
|
|
'status' => 'failed',
|
|
'file' => failure_path
|
|
}
|
|
]
|
|
},
|
|
{
|
|
'failed_count' => 1,
|
|
'name' => 'rspec unit pg16 10/12',
|
|
'test_cases' => [
|
|
{
|
|
'status' => 'failed',
|
|
'file' => other_failure_path
|
|
}
|
|
]
|
|
},
|
|
{
|
|
'failed_count' => 1,
|
|
'name' => 'jest unit 1/11',
|
|
'test_cases' => [
|
|
{
|
|
'status' => 'failed',
|
|
'file' => frontend_failure_path
|
|
}
|
|
]
|
|
}
|
|
]
|
|
}
|
|
end
|
|
|
|
subject { described_class.new(options) }
|
|
|
|
def failed_test_case(file_path)
|
|
{ 'status' => 'failed', 'file' => file_path, 'job_url' => nil }
|
|
end
|
|
|
|
describe '#output_failed_tests' do
|
|
context 'with a valid report file' do
|
|
before do
|
|
allow(subject).to receive(:file_contents_as_json).and_return(file_contents_as_json)
|
|
end
|
|
|
|
it 'writes files for respective test suites' do
|
|
expect(File).to receive(:write)
|
|
.with(File.join(output_directory, "rspec_failed_tests.txt"),
|
|
"#{failure_path} #{other_failure_path}")
|
|
expect(File).to receive(:write)
|
|
.with(File.join(output_directory, "rspec_all_failed_tests.txt"),
|
|
"#{failure_path} #{other_failure_path}")
|
|
expect(File).to receive(:write)
|
|
.with(File.join(output_directory, "rspec_ee_failed_tests.txt"),
|
|
failure_path)
|
|
expect(File).to receive(:write)
|
|
.with(File.join(output_directory, "jest_failed_tests.txt"),
|
|
frontend_failure_path)
|
|
|
|
subject.output_failed_tests
|
|
end
|
|
|
|
context 'when given a valid format(json)' do
|
|
subject { described_class.new(options.merge(format: :json)) }
|
|
|
|
it 'writes list of failed test files for respective tests suites' do
|
|
expected_rspec_content = Gitlab::Json.pretty_generate([
|
|
{ 'status' => 'failed', 'file' => failure_path, 'job_url' => nil },
|
|
{ 'status' => 'failed', 'file' => other_failure_path, 'job_url' => nil }
|
|
])
|
|
expected_rspec_ee_content = Gitlab::Json.pretty_generate([
|
|
{ 'status' => 'failed', 'file' => failure_path, 'job_url' => nil }
|
|
])
|
|
expected_rspec_all_content = Gitlab::Json.pretty_generate([
|
|
{ 'status' => 'failed', 'file' => failure_path, 'job_url' => nil },
|
|
{ 'status' => 'failed', 'file' => failure_path, 'job_url' => nil },
|
|
{ 'status' => 'failed', 'file' => other_failure_path, 'job_url' => nil }
|
|
])
|
|
expected_jest_content = Gitlab::Json.pretty_generate([
|
|
{ 'status' => 'failed', 'file' => frontend_failure_path, 'job_url' => nil }
|
|
])
|
|
|
|
expect(File).to receive(:write)
|
|
.with(File.join(output_directory, "rspec_failed_tests.json"),
|
|
expected_rspec_content)
|
|
expect(File).to receive(:write)
|
|
.with(File.join(output_directory, "rspec_ee_failed_tests.json"),
|
|
expected_rspec_ee_content)
|
|
expect(File).to receive(:write)
|
|
.with(File.join(output_directory, "rspec_all_failed_tests.json"),
|
|
expected_rspec_all_content)
|
|
expect(File).to receive(:write)
|
|
.with(File.join(output_directory, "jest_failed_tests.json"),
|
|
expected_jest_content)
|
|
|
|
subject.output_failed_tests
|
|
end
|
|
end
|
|
|
|
context 'when given an invalid format' do
|
|
subject { described_class.new(options.merge(format: :foo)) }
|
|
|
|
it 'raises an exception' do
|
|
expect { subject.output_failed_tests }
|
|
.to raise_error '[FailedTests] Unsupported format `foo` (allowed formats: `oneline` and `json`)!'
|
|
end
|
|
end
|
|
|
|
describe 'empty report' do
|
|
let(:file_contents_as_json) do
|
|
{}
|
|
end
|
|
|
|
it 'does not fail for output files' do
|
|
subject.output_failed_tests
|
|
end
|
|
|
|
it 'returns empty results for suite failures' do
|
|
result = subject.failed_cases_for_suite_collection
|
|
|
|
expect(result.values.flatten).to be_empty
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
describe 'missing report file' do
|
|
subject { described_class.new(options.merge(previous_tests_report_path: 'unknownfile.json')) }
|
|
|
|
it 'does not fail for output files' do
|
|
subject.output_failed_tests
|
|
end
|
|
|
|
it 'returns empty results for suite failures' do
|
|
result = subject.failed_cases_for_suite_collection
|
|
|
|
expect(result.values.flatten).to be_empty
|
|
end
|
|
end
|
|
|
|
describe 'deduplication behavior' do
|
|
let(:duplicate_failures_json) do
|
|
{
|
|
'suites' => [
|
|
{
|
|
'failed_count' => 2,
|
|
'name' => 'rspec unit pg16 1/12',
|
|
'test_cases' => [
|
|
{ 'status' => 'failed', 'file' => user_spec },
|
|
{ 'status' => 'failed', 'file' => project_spec }
|
|
]
|
|
},
|
|
{
|
|
'failed_count' => 2,
|
|
'name' => 'rspec-ee unit pg16 2/12',
|
|
'test_cases' => [
|
|
{ 'status' => 'failed', 'file' => user_spec },
|
|
{ 'status' => 'failed', 'file' => issue_spec }
|
|
]
|
|
}
|
|
]
|
|
}
|
|
end
|
|
|
|
context 'with duplicate test files in multiple test suites' do
|
|
subject { described_class.new(options) }
|
|
|
|
before do
|
|
allow(subject).to receive(:file_contents_as_json).and_return(duplicate_failures_json)
|
|
allow(File).to receive(:write)
|
|
end
|
|
|
|
it 'deduplicates test file paths in oneline format' do
|
|
expect(File).to receive(:write)
|
|
.with(File.join(output_directory, "rspec_all_failed_tests.txt"),
|
|
"#{user_spec} #{project_spec} #{issue_spec}")
|
|
|
|
subject.output_failed_tests
|
|
end
|
|
end
|
|
end
|
|
|
|
describe 'CLI option parsing' do
|
|
describe '.parse_cli_options' do
|
|
it 'parses all options when provided' do
|
|
options = described_class.parse_cli_options([
|
|
'--previous-tests-report-path', '/custom/report.json',
|
|
'--output-directory', '/custom/output',
|
|
'--format', 'json',
|
|
'--rspec-pg-regex', 'custom.*pg',
|
|
'--rspec-ee-pg-regex', 'custom.*ee'
|
|
])
|
|
|
|
expect(options[:previous_tests_report_path]).to eq '/custom/report.json'
|
|
expect(options[:output_directory]).to eq '/custom/output'
|
|
expect(options[:format]).to eq 'json'
|
|
expect(options[:rspec_pg_regex]).to eq(/custom.*pg/)
|
|
expect(options[:rspec_ee_pg_regex]).to eq(/custom.*ee/)
|
|
end
|
|
|
|
it 'uses default values when no options provided' do
|
|
options = described_class.parse_cli_options([])
|
|
|
|
expect(options[:previous_tests_report_path]).to eq 'test_results/previous/test_reports.json'
|
|
expect(options[:output_directory]).to eq 'tmp/previous_failed_tests/'
|
|
expect(options[:format]).to eq :oneline
|
|
expect(options[:rspec_pg_regex]).to eq(/rspec .+ pg16( .+)?/)
|
|
expect(options[:rspec_ee_pg_regex]).to eq(/rspec-ee .+ pg16( .+)?/)
|
|
end
|
|
|
|
it 'handles help option by exiting' do
|
|
expect { described_class.parse_cli_options(['--help']) }.to raise_error(SystemExit)
|
|
end
|
|
end
|
|
end
|
|
end
|