From 9acaa09ba87b3cbcdc371c32efad78fceaf1b370 Mon Sep 17 00:00:00 2001 From: Thomas Hutterer <thutterer@gitlab.com> Date: Tue, 6 Aug 2024 19:15:06 +0000 Subject: [PATCH] Add Danger warning about changed SettingBlock instances Settings sections can be searched through the command palette. But this only works if they are kept in sync with the files that drive that search. --- danger/plugins/settings_sections.rb | 9 +++ danger/settings_sections/Dangerfile | 3 + lib/search/group_settings.rb | 2 + lib/search/project_settings.rb | 2 + spec/tooling/danger/settings_sections_spec.rb | 57 +++++++++++++++++++ tooling/danger/settings_sections.rb | 55 ++++++++++++++++++ 6 files changed, 128 insertions(+) create mode 100644 danger/plugins/settings_sections.rb create mode 100644 danger/settings_sections/Dangerfile create mode 100644 spec/tooling/danger/settings_sections_spec.rb create mode 100644 tooling/danger/settings_sections.rb diff --git a/danger/plugins/settings_sections.rb b/danger/plugins/settings_sections.rb new file mode 100644 index 000000000000..c58e2832e4af --- /dev/null +++ b/danger/plugins/settings_sections.rb @@ -0,0 +1,9 @@ +# frozen_string_literal: true + +require_relative '../../tooling/danger/settings_sections' + +module Danger + class SettingsSections < ::Danger::Plugin + include Tooling::Danger::SettingsSections + end +end diff --git a/danger/settings_sections/Dangerfile b/danger/settings_sections/Dangerfile new file mode 100644 index 000000000000..05b5f8e1d5da --- /dev/null +++ b/danger/settings_sections/Dangerfile @@ -0,0 +1,3 @@ +# frozen_string_literal: true + +settings_sections.check! diff --git a/lib/search/group_settings.rb b/lib/search/group_settings.rb index 32903860e7db..96c3fb8878b3 100644 --- a/lib/search/group_settings.rb +++ b/lib/search/group_settings.rb @@ -1,6 +1,8 @@ # frozen_string_literal: true module Search + # Generates a list of all available setting sections of a group. + # This list is used by the command palette's search functionality. class GroupSettings include Rails.application.routes.url_helpers diff --git a/lib/search/project_settings.rb b/lib/search/project_settings.rb index 413b69a0ac8e..645783c83b94 100644 --- a/lib/search/project_settings.rb +++ b/lib/search/project_settings.rb @@ -1,6 +1,8 @@ # frozen_string_literal: true module Search + # Generates a list of all available setting sections of a project. + # This list is used by the command palette's search functionality. class ProjectSettings include Rails.application.routes.url_helpers diff --git a/spec/tooling/danger/settings_sections_spec.rb b/spec/tooling/danger/settings_sections_spec.rb new file mode 100644 index 000000000000..1fbea0e1f3a1 --- /dev/null +++ b/spec/tooling/danger/settings_sections_spec.rb @@ -0,0 +1,57 @@ +# frozen_string_literal: true + +require 'gitlab/dangerfiles/spec_helper' +require 'fast_spec_helper' + +require_relative '../../../tooling/danger/settings_sections' + +RSpec.describe Tooling::Danger::SettingsSections, feature_category: :tooling do + include_context 'with dangerfile' + + subject(:settings_section_check) { fake_danger.new(helper: fake_helper) } + + let(:fake_danger) { DangerSpecHelper.fake_danger.include(described_class) } + let(:matching_changed_files) { ['app/views/foo/bar.html.haml', 'app/assets/js/foo/bar.vue'] } + let(:changed_lines) { ['-render SettingsBlockComponent.new(id: "foo") do', '<settings-block id="foo">'] } + let(:stable_branch?) { false } + + before do + allow(fake_helper).to receive(:changed_files).and_return(matching_changed_files) + allow(fake_helper).to receive(:changed_lines).and_return(changed_lines) + allow(fake_helper).to receive(:stable_branch?).and_return(stable_branch?) + end + + context 'when on stable branch' do + let(:stable_branch?) { true } + + it 'does not write any markdown' do + expect(settings_section_check).not_to receive(:markdown) + settings_section_check.check! + end + end + + context 'when none of the changed files are Haml or Vue files' do + let(:matching_changed_files) { [] } + + it 'does not write any markdown' do + expect(settings_section_check).not_to receive(:markdown) + settings_section_check.check! + end + end + + context 'when none of the changed lines match the pattern' do + let(:changed_lines) { ['-foo', '+bar'] } + + it 'does not write any markdown' do + expect(settings_section_check).not_to receive(:markdown) + settings_section_check.check! + end + end + + it 'adds a new markdown section listing every matching line' do + expect(settings_section_check).to receive(:markdown).with(/Searchable setting sections/) + expect(settings_section_check).to receive(:markdown).with(/SettingsBlock/) + expect(settings_section_check).to receive(:markdown).with(/settings-block/) + settings_section_check.check! + end +end diff --git a/tooling/danger/settings_sections.rb b/tooling/danger/settings_sections.rb new file mode 100644 index 000000000000..b05fb721e94e --- /dev/null +++ b/tooling/danger/settings_sections.rb @@ -0,0 +1,55 @@ +# frozen_string_literal: true + +module Tooling + module Danger + module SettingsSections + def check! + return if helper.stable_branch? + + changed_code_files = helper.changed_files(/\.(haml|vue)$/) + return if changed_code_files.empty? + + vc_regexp = /(SettingsBlockComponent|settings-block)/ + lines_with_matches = filter_changed_lines(changed_code_files, vc_regexp) + return if lines_with_matches.empty? + + markdown(<<~MARKDOWN) + ## Searchable setting sections + + Looks like you have edited the template of some settings section. Please check that all changed sections are still searchable: + + - If you created a new section, make sure to add it to either `lib/search/project_settings.rb` or `lib/search/group_settings.rb`, or in their counterparts in `ee/` if this section is only available behind a licensed feature. + - If you removed a section, make sure to also remove it from the files above. + - If you changed a section's id, please update it also in the files above. + - If you just moved code around within the same page, there is nothing to do. + + MARKDOWN + + lines_with_matches.each do |file, lines| + markdown(<<~MARKDOWN) + #### `#{file}` + + ```shell + #{lines.join("\n")} + ``` + + MARKDOWN + end + end + + def filter_changed_lines(files, pattern) + files_with_lines = {} + files.each do |file| + next if file.start_with?('spec/', 'ee/spec/', 'qa/') + + matching_changed_lines = helper.changed_lines(file).select { |line| line =~ pattern } + next unless matching_changed_lines.any? + + files_with_lines[file] = matching_changed_lines + end + + files_with_lines + end + end + end +end -- GitLab