2021-11-18 22:05:49 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
require 'rspec-parameterized'
|
|
|
|
require 'gitlab-dangerfiles'
|
|
|
|
require 'danger'
|
2022-01-26 12:08:38 +05:30
|
|
|
require 'danger/plugins/internal/helper'
|
2021-11-18 22:05:49 +05:30
|
|
|
require 'gitlab/dangerfiles/spec_helper'
|
|
|
|
|
|
|
|
require_relative '../../../tooling/danger/specs'
|
|
|
|
require_relative '../../../tooling/danger/project_helper'
|
|
|
|
|
2023-03-04 22:38:38 +05:30
|
|
|
RSpec.describe Tooling::Danger::Specs, feature_category: :tooling do
|
2021-11-18 22:05:49 +05:30
|
|
|
include_context "with dangerfile"
|
|
|
|
|
|
|
|
let(:fake_danger) { DangerSpecHelper.fake_danger.include(described_class) }
|
2022-11-25 23:54:43 +05:30
|
|
|
let(:fake_project_helper) { instance_double('Tooling::Danger::ProjectHelper') }
|
|
|
|
let(:filename) { 'spec/foo_spec.rb' }
|
|
|
|
|
2021-11-18 22:05:49 +05:30
|
|
|
let(:file_lines) do
|
|
|
|
[
|
|
|
|
" describe 'foo' do",
|
|
|
|
" expect(foo).to match(['bar'])",
|
|
|
|
" end",
|
|
|
|
" expect(foo).to match(['bar'])", # same line as line 1 above, we expect two different suggestions
|
|
|
|
" ",
|
|
|
|
" expect(foo).to match ['bar']",
|
|
|
|
" expect(foo).to eq(['bar'])",
|
|
|
|
" expect(foo).to eq ['bar']",
|
|
|
|
" expect(foo).to(match(['bar']))",
|
|
|
|
" expect(foo).to(eq(['bar']))",
|
|
|
|
" foo.eq(['bar'])"
|
|
|
|
]
|
|
|
|
end
|
|
|
|
|
|
|
|
let(:matching_lines) do
|
|
|
|
[
|
2022-11-25 23:54:43 +05:30
|
|
|
"+ expect(foo).to match(['should not error'])",
|
2021-11-18 22:05:49 +05:30
|
|
|
"+ expect(foo).to match(['bar'])",
|
|
|
|
"+ expect(foo).to match(['bar'])",
|
|
|
|
"+ expect(foo).to match ['bar']",
|
|
|
|
"+ expect(foo).to eq(['bar'])",
|
|
|
|
"+ expect(foo).to eq ['bar']",
|
|
|
|
"+ expect(foo).to(match(['bar']))",
|
|
|
|
"+ expect(foo).to(eq(['bar']))"
|
|
|
|
]
|
|
|
|
end
|
|
|
|
|
2022-11-25 23:54:43 +05:30
|
|
|
let(:changed_lines) do
|
|
|
|
[
|
|
|
|
" expect(foo).to match(['bar'])",
|
|
|
|
" expect(foo).to match(['bar'])",
|
|
|
|
" expect(foo).to match ['bar']",
|
|
|
|
" expect(foo).to eq(['bar'])",
|
|
|
|
" expect(foo).to eq ['bar']",
|
|
|
|
"- expect(foo).to match(['bar'])",
|
|
|
|
"- expect(foo).to match(['bar'])",
|
|
|
|
"- expect(foo).to match ['bar']",
|
|
|
|
"- expect(foo).to eq(['bar'])",
|
|
|
|
"- expect(foo).to eq ['bar']",
|
|
|
|
"+ expect(foo).to eq([])"
|
|
|
|
] + matching_lines
|
|
|
|
end
|
|
|
|
|
2021-11-18 22:05:49 +05:30
|
|
|
subject(:specs) { fake_danger.new(helper: fake_helper) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
allow(specs).to receive(:project_helper).and_return(fake_project_helper)
|
2022-11-25 23:54:43 +05:30
|
|
|
allow(specs.helper).to receive(:changed_lines).with(filename).and_return(matching_lines)
|
|
|
|
allow(specs.project_helper).to receive(:file_lines).and_return(file_lines)
|
2021-11-18 22:05:49 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
describe '#changed_specs_files' do
|
|
|
|
let(:base_expected_files) { %w[spec/foo_spec.rb ee/spec/foo_spec.rb spec/bar_spec.rb ee/spec/bar_spec.rb spec/zab_spec.rb ee/spec/zab_spec.rb] }
|
|
|
|
|
|
|
|
before do
|
|
|
|
all_changed_files = %w[
|
|
|
|
app/workers/a.rb
|
|
|
|
app/workers/b.rb
|
|
|
|
app/workers/e.rb
|
|
|
|
spec/foo_spec.rb
|
|
|
|
ee/spec/foo_spec.rb
|
|
|
|
spec/bar_spec.rb
|
|
|
|
ee/spec/bar_spec.rb
|
|
|
|
spec/zab_spec.rb
|
|
|
|
ee/spec/zab_spec.rb
|
|
|
|
]
|
|
|
|
|
|
|
|
allow(specs.helper).to receive(:all_changed_files).and_return(all_changed_files)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns added, modified, and renamed_after files by default' do
|
|
|
|
expect(specs.changed_specs_files).to match_array(base_expected_files)
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with include_ee: :exclude' do
|
|
|
|
it 'returns spec files without EE-specific files' do
|
|
|
|
expect(specs.changed_specs_files(ee: :exclude)).not_to include(%w[ee/spec/foo_spec.rb ee/spec/bar_spec.rb ee/spec/zab_spec.rb])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with include_ee: :only' do
|
|
|
|
it 'returns EE-specific spec files only' do
|
|
|
|
expect(specs.changed_specs_files(ee: :only)).to match_array(%w[ee/spec/foo_spec.rb ee/spec/bar_spec.rb ee/spec/zab_spec.rb])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-11-25 23:54:43 +05:30
|
|
|
describe '#add_suggestions_for_match_with_array' do
|
|
|
|
let(:template) do
|
|
|
|
<<~MARKDOWN
|
|
|
|
```suggestion
|
|
|
|
%<suggested_line>s
|
|
|
|
```
|
|
|
|
|
|
|
|
If order of the result is not important, please consider using `match_array` to avoid flakiness.
|
|
|
|
MARKDOWN
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'adds suggestions at the correct lines' do
|
|
|
|
[
|
|
|
|
{ suggested_line: " expect(foo).to match_array(['bar'])", number: 2 },
|
|
|
|
{ suggested_line: " expect(foo).to match_array(['bar'])", number: 4 },
|
|
|
|
{ suggested_line: " expect(foo).to match_array ['bar']", number: 6 },
|
|
|
|
{ suggested_line: " expect(foo).to match_array(['bar'])", number: 7 },
|
|
|
|
{ suggested_line: " expect(foo).to match_array ['bar']", number: 8 },
|
|
|
|
{ suggested_line: " expect(foo).to(match_array(['bar']))", number: 9 },
|
|
|
|
{ suggested_line: " expect(foo).to(match_array(['bar']))", number: 10 }
|
|
|
|
].each do |test_case|
|
|
|
|
comment = format(template, suggested_line: test_case[:suggested_line])
|
|
|
|
expect(specs).to receive(:markdown).with(comment, file: filename, line: test_case[:number])
|
|
|
|
end
|
|
|
|
|
|
|
|
specs.add_suggestions_for_match_with_array(filename)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#add_suggestions_for_project_factory_usage' do
|
|
|
|
let(:template) do
|
|
|
|
<<~MARKDOWN
|
|
|
|
```suggestion
|
|
|
|
%<suggested_line>s
|
|
|
|
```
|
|
|
|
|
|
|
|
Project creations are very slow. Use `let_it_be`, `build` or `build_stubbed` if possible.
|
|
|
|
See [testing best practices](https://docs.gitlab.com/ee/development/testing_guide/best_practices.html#optimize-factory-usage)
|
|
|
|
for background information and alternative options.
|
|
|
|
MARKDOWN
|
|
|
|
end
|
|
|
|
|
|
|
|
let(:file_lines) do
|
|
|
|
[
|
|
|
|
" let(:project) { create(:project) }",
|
|
|
|
" let_it_be(:project) { create(:project, :repository)",
|
|
|
|
" let!(:project) { create(:project) }",
|
|
|
|
" let(:var) { create(:project) }",
|
|
|
|
" let(:merge_request) { create(:merge_request, project: project)",
|
|
|
|
" context 'when merge request exists' do",
|
|
|
|
" it { is_expected.to be_success }",
|
|
|
|
" end",
|
|
|
|
" let!(:var) { create(:project) }",
|
|
|
|
" let(:project) { create(:thing) }",
|
|
|
|
" let(:project) { build(:project) }",
|
|
|
|
" let(:project) do",
|
|
|
|
" create(:project)",
|
|
|
|
" end",
|
|
|
|
" let(:project) { create(:project, :repository) }",
|
|
|
|
" str = 'let(:project) { create(:project) }'",
|
|
|
|
" let(:project) { create(:project_empty_repo) }",
|
|
|
|
" let(:project) { create(:forked_project_with_submodules) }",
|
|
|
|
" let(:project) { create(:project_with_design) }",
|
|
|
|
" let(:authorization) { create(:project_authorization) }"
|
|
|
|
]
|
|
|
|
end
|
|
|
|
|
|
|
|
let(:matching_lines) do
|
|
|
|
[
|
|
|
|
"+ let(:should_not_error) { create(:project) }",
|
|
|
|
"+ let(:project) { create(:project) }",
|
|
|
|
"+ let!(:project) { create(:project) }",
|
|
|
|
"+ let(:var) { create(:project) }",
|
|
|
|
"+ let!(:var) { create(:project) }",
|
|
|
|
"+ let(:project) { create(:project, :repository) }",
|
|
|
|
"+ let(:project) { create(:project_empty_repo) }",
|
|
|
|
"+ let(:project) { create(:forked_project_with_submodules) }",
|
|
|
|
"+ let(:project) { create(:project_with_design) }"
|
|
|
|
]
|
|
|
|
end
|
|
|
|
|
2021-11-18 22:05:49 +05:30
|
|
|
let(:changed_lines) do
|
|
|
|
[
|
2022-11-25 23:54:43 +05:30
|
|
|
"+ line which doesn't exist in the file and should not cause an error",
|
|
|
|
"+ let_it_be(:project) { create(:project, :repository)",
|
|
|
|
"+ let(:project) { create(:thing) }",
|
|
|
|
"+ let(:project) do",
|
|
|
|
"+ create(:project)",
|
|
|
|
"+ end",
|
|
|
|
"+ str = 'let(:project) { create(:project) }'",
|
|
|
|
"+ let(:authorization) { create(:project_authorization) }"
|
2021-11-18 22:05:49 +05:30
|
|
|
] + matching_lines
|
|
|
|
end
|
|
|
|
|
2022-11-25 23:54:43 +05:30
|
|
|
it 'adds suggestions at the correct lines', :aggregate_failures do
|
|
|
|
[
|
|
|
|
{ suggested_line: " let_it_be(:project) { create(:project) }", number: 1 },
|
|
|
|
{ suggested_line: " let_it_be(:project) { create(:project) }", number: 3 },
|
|
|
|
{ suggested_line: " let_it_be(:var) { create(:project) }", number: 4 },
|
|
|
|
{ suggested_line: " let_it_be(:var) { create(:project) }", number: 9 },
|
|
|
|
{ suggested_line: " let_it_be(:project) { create(:project, :repository) }", number: 15 },
|
|
|
|
{ suggested_line: " let_it_be(:project) { create(:project_empty_repo) }", number: 17 },
|
|
|
|
{ suggested_line: " let_it_be(:project) { create(:forked_project_with_submodules) }", number: 18 },
|
|
|
|
{ suggested_line: " let_it_be(:project) { create(:project_with_design) }", number: 19 }
|
|
|
|
].each do |test_case|
|
|
|
|
comment = format(template, suggested_line: test_case[:suggested_line])
|
|
|
|
expect(specs).to receive(:markdown).with(comment, file: filename, line: test_case[:number])
|
|
|
|
end
|
2021-11-18 22:05:49 +05:30
|
|
|
|
2022-11-25 23:54:43 +05:30
|
|
|
specs.add_suggestions_for_project_factory_usage(filename)
|
2021-11-18 22:05:49 +05:30
|
|
|
end
|
|
|
|
end
|
2023-03-04 22:38:38 +05:30
|
|
|
|
|
|
|
describe '#add_suggestions_for_feature_category' do
|
|
|
|
let(:template) do
|
|
|
|
<<~SUGGESTION_MARKDOWN
|
|
|
|
```suggestion
|
|
|
|
%<suggested_line>s
|
|
|
|
```
|
|
|
|
|
|
|
|
Consider adding `feature_category: <feature_category_name>` for this example if it is not set already.
|
|
|
|
See [testing best practices](https://docs.gitlab.com/ee/development/testing_guide/best_practices.html#feature-category-metadata).
|
|
|
|
SUGGESTION_MARKDOWN
|
|
|
|
end
|
|
|
|
|
|
|
|
let(:file_lines) do
|
|
|
|
[
|
|
|
|
" require 'spec_helper'",
|
|
|
|
" \n",
|
|
|
|
" RSpec.describe Projects::Analytics::CycleAnalytics::SummaryController, feature_category: :planning_analytics do",
|
|
|
|
" end",
|
|
|
|
"RSpec.describe Projects::Analytics::CycleAnalytics::SummaryController do",
|
|
|
|
" let_it_be(:user) { create(:user) }",
|
|
|
|
" end",
|
|
|
|
" describe 'GET \"time_summary\"' do",
|
|
|
|
" end",
|
|
|
|
" RSpec.describe Projects::Analytics::CycleAnalytics::SummaryController do",
|
|
|
|
" let_it_be(:user) { create(:user) }",
|
|
|
|
" end",
|
|
|
|
" describe 'GET \"time_summary\"' do",
|
|
|
|
" end"
|
|
|
|
]
|
|
|
|
end
|
|
|
|
|
|
|
|
let(:matching_lines) do
|
|
|
|
[
|
|
|
|
"+ RSpec.describe Projects::Analytics::CycleAnalytics::SummaryController, feature_category: :planning_analytics do",
|
|
|
|
"+RSpec.describe Projects::Analytics::CycleAnalytics::SummaryController do",
|
|
|
|
"+ RSpec.describe Projects::Analytics::CycleAnalytics::SummaryController do"
|
|
|
|
]
|
|
|
|
end
|
|
|
|
|
|
|
|
let(:changed_lines) do
|
|
|
|
[
|
|
|
|
"+ RSpec.describe Projects::Analytics::CycleAnalytics::SummaryController, feature_category: :planning_analytics do",
|
|
|
|
"+RSpec.describe Projects::Analytics::CycleAnalytics::SummaryController do",
|
|
|
|
"+ let_it_be(:user) { create(:user) }",
|
|
|
|
"- end",
|
|
|
|
"+ describe 'GET \"time_summary\"' do",
|
|
|
|
"+ RSpec.describe Projects::Analytics::CycleAnalytics::SummaryController do"
|
|
|
|
]
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'adds suggestions at the correct lines', :aggregate_failures do
|
|
|
|
[
|
|
|
|
{ suggested_line: "RSpec.describe Projects::Analytics::CycleAnalytics::SummaryController do", number: 5 },
|
|
|
|
{ suggested_line: " RSpec.describe Projects::Analytics::CycleAnalytics::SummaryController do", number: 10 }
|
|
|
|
|
|
|
|
].each do |test_case|
|
|
|
|
comment = format(template, suggested_line: test_case[:suggested_line])
|
|
|
|
expect(specs).to receive(:markdown).with(comment, file: filename, line: test_case[:number])
|
|
|
|
end
|
|
|
|
|
|
|
|
specs.add_suggestions_for_feature_category(filename)
|
|
|
|
end
|
|
|
|
end
|
2021-11-18 22:05:49 +05:30
|
|
|
end
|