From eedc4a1c1bcecd3e676f4f03eb1dcd1f72586bfd Mon Sep 17 00:00:00 2001 From: Matt Kasa <mkasa@gitlab.com> Date: Thu, 20 Jul 2023 12:02:08 -0700 Subject: [PATCH] Detect model scope changes as database changes in Danger Relates to https://gitlab.com/gitlab-org/gitlab/-/issues/419001 --- danger/database/Dangerfile | 4 +- spec/tooling/danger/database_spec.rb | 90 +++++++++++++++++++++++++++- tooling/danger/database.rb | 16 +++++ 3 files changed, 106 insertions(+), 4 deletions(-) diff --git a/danger/database/Dangerfile b/danger/database/Dangerfile index 1c0e9d721180..6861a04ffc6c 100644 --- a/danger/database/Dangerfile +++ b/danger/database/Dangerfile @@ -67,9 +67,7 @@ end return unless helper.ci? return if helper.mr_labels.include?(DATABASE_APPROVED_LABEL) -db_paths_to_review = helper.changes_by_category[:database] - -if helper.mr_labels.include?('database') || db_paths_to_review.any? +if helper.mr_labels.include?('database') || database.changes.any? message 'This merge request adds or changes files that require a ' \ 'review from the [Database team](https://gitlab.com/groups/gl-database/-/group_members).' diff --git a/spec/tooling/danger/database_spec.rb b/spec/tooling/danger/database_spec.rb index ddcfa279dc3c..a342014cf6bf 100644 --- a/spec/tooling/danger/database_spec.rb +++ b/spec/tooling/danger/database_spec.rb @@ -1,5 +1,6 @@ # frozen_string_literal: true +require 'rspec-parameterized' require 'gitlab-dangerfiles' require 'danger' require 'danger/plugins/internal/helper' @@ -41,11 +42,98 @@ let(:cutoff) { Date.parse('2022-10-01') - 21 } - subject(:database) { fake_danger.new } + subject(:database) { fake_danger.new(helper: fake_helper) } describe '#find_migration_files_before' do it 'returns migrations that are before the cutoff' do expect(database.find_migration_files_before(migration_files, cutoff).length).to eq(8) end end + + describe '#changes' do + using RSpec::Parameterized::TableSyntax + + where do + { + 'with database changes to a migration file' => { + modified_files: %w[ + db/migrate/20230720114001_test_migration.rb + db/schema_migrations/20230720114001 + db/structure.sql + app/models/test.rb + ], + changed_lines: [], + changes_by_category: { + database: %w[ + db/migrate/20230720114001_test_migration.rb + db/schema_migrations/20230720114001 + db/structure.sql + ] + }, + impacted_files: %w[ + db/migrate/20230720114001_test_migration.rb + db/schema_migrations/20230720114001 + db/structure.sql + ] + }, + 'with non-database changes' => { + modified_files: %w[ + app/models/test.rb + ], + changed_lines: %w[ + +# Comment explaining scope :blah + ], + changes_by_category: { + database: [] + }, + impacted_files: [] + }, + 'with database changes in a doc' => { + modified_files: %w[doc/development/database/test.md], + changed_lines: [ + '+scope :blah, ->() { where(hidden: false) }' + ], + changes_by_category: { + database: [] + }, + impacted_files: [] + }, + 'with database changes in a model' => { + modified_files: %w[app/models/test.rb], + changed_lines: [ + '+# Comment explaining scope :blah', + '+scope :blah, ->() { where(hidden: false) }' + ], + changes_by_category: { + database: [] + }, + impacted_files: %w[app/models/test.rb] + }, + 'with database changes in a concern' => { + modified_files: %w[app/models/concerns/test.rb], + changed_lines: [ + '- .where(hidden: false)', + '+ .where(hidden: true)' + ], + changes_by_category: { + database: [] + }, + impacted_files: %w[app/models/concerns/test.rb] + } + } + end + + with_them do + before do + allow(fake_helper).to receive(:modified_files).and_return(modified_files) + allow(fake_helper).to receive(:all_changed_files).and_return(modified_files) + allow(fake_helper).to receive(:changed_lines).and_return(changed_lines) + allow(fake_helper).to receive(:changes_by_category).and_return(changes_by_category) + end + + it 'returns database changes' do + expect(database.changes).to match impacted_files + end + end + end end diff --git a/tooling/danger/database.rb b/tooling/danger/database.rb index 4cfac7c4af44..71b3ed1a1a5d 100644 --- a/tooling/danger/database.rb +++ b/tooling/danger/database.rb @@ -5,6 +5,8 @@ module Danger module Database TIMESTAMP_MATCHER = /(?<timestamp>\d{14})/ MIGRATION_MATCHER = %r{\A(ee/)?db/(geo/)?(post_)?migrate/} + MODEL_PATHS = %r{\A(ee/)?app/models/} + MODEL_CHANGES = %r{^[^#\n]*?(?:scope :|where\(|joins\()} def find_migration_files_before(file_names, cutoff) migrations = file_names.select { |f| f.match?(MIGRATION_MATCHER) } @@ -15,6 +17,20 @@ def find_migration_files_before(file_names, cutoff) timestamp < cutoff end end + + def changes + changed_database_paths + changed_model_paths + end + + def changed_database_paths + helper.changes_by_category[:database] + end + + def changed_model_paths + helper.all_changed_files.grep(MODEL_PATHS).select do |file| + helper.changed_lines(file).any? { |change| change =~ MODEL_CHANGES } + end + end end end end -- GitLab