Commit b4da574a authored by Kerri Miller's avatar Kerri Miller

Prefer explicit methods over delegates for predicates

Predicate methods should return a boolean value, however when we
delegate with allow_nil to properly handle non-extant relations, we
end up with true, false, and nil as possible responses. While we
generally are loose about nil being equivalent to false in Ruby, a
predicate method is a special signal that we very much care about
the boolean-nature of its response.
parent 4d29bbac
...@@ -376,8 +376,7 @@ class MergeRequest < ApplicationRecord ...@@ -376,8 +376,7 @@ class MergeRequest < ApplicationRecord
alias_attribute :auto_merge_enabled, :merge_when_pipeline_succeeds alias_attribute :auto_merge_enabled, :merge_when_pipeline_succeeds
alias_method :issuing_parent, :target_project alias_method :issuing_parent, :target_project
delegate :active?, :builds_with_coverage, to: :head_pipeline, prefix: true, allow_nil: true delegate :builds_with_coverage, to: :head_pipeline, prefix: true, allow_nil: true
delegate :success?, :active?, to: :actual_head_pipeline, prefix: true, allow_nil: true
RebaseLockTimeout = Class.new(StandardError) RebaseLockTimeout = Class.new(StandardError)
...@@ -437,6 +436,18 @@ class MergeRequest < ApplicationRecord ...@@ -437,6 +436,18 @@ class MergeRequest < ApplicationRecord
target_project.latest_pipeline(target_branch, sha) target_project.latest_pipeline(target_branch, sha)
end end
def head_pipeline_active?
!!head_pipeline&.active?
end
def actual_head_pipeline_active?
!!actual_head_pipeline&.active?
end
def actual_head_pipeline_success?
!!actual_head_pipeline&.success?
end
# Pattern used to extract `!123` merge request references from text # Pattern used to extract `!123` merge request references from text
# #
# This pattern supports cross-project references. # This pattern supports cross-project references.
......
...@@ -52,8 +52,6 @@ module EE ...@@ -52,8 +52,6 @@ module EE
delegate :sha, to: :head_pipeline, prefix: :head_pipeline, allow_nil: true delegate :sha, to: :head_pipeline, prefix: :head_pipeline, allow_nil: true
delegate :sha, to: :base_pipeline, prefix: :base_pipeline, allow_nil: true delegate :sha, to: :base_pipeline, prefix: :base_pipeline, allow_nil: true
delegate :merge_requests_author_approval?, to: :target_project, allow_nil: true
delegate :merge_requests_disable_committers_approval?, to: :target_project, allow_nil: true
accepts_nested_attributes_for :approval_rules, allow_destroy: true accepts_nested_attributes_for :approval_rules, allow_destroy: true
...@@ -71,6 +69,14 @@ module EE ...@@ -71,6 +69,14 @@ module EE
scope :including_merge_train, -> do scope :including_merge_train, -> do
includes(:merge_train) includes(:merge_train)
end end
def merge_requests_author_approval?
!!target_project&.merge_requests_author_approval?
end
def merge_requests_disable_committers_approval?
!!target_project&.merge_requests_disable_committers_approval?
end
end end
class_methods do class_methods do
......
...@@ -93,6 +93,48 @@ RSpec.describe MergeRequest do ...@@ -93,6 +93,48 @@ RSpec.describe MergeRequest do
end end
end end
end end
describe '#merge_requests_author_approval?' do
context 'when project lacks a target_project relation' do
before do
merge_request.target_project = nil
end
it 'returns false' do
expect(merge_request.merge_requests_author_approval?).to be false
end
end
context 'when project has a target_project relation' do
it 'accesses the value from the target_project' do
expect(merge_request.target_project)
.to receive(:merge_requests_author_approval?)
merge_request.merge_requests_author_approval?
end
end
end
describe '#merge_requests_disable_committers_approval?' do
context 'when project lacks a target_project relation' do
before do
merge_request.target_project = nil
end
it 'returns false' do
expect(merge_request.merge_requests_disable_committers_approval?).to be false
end
end
context 'when project has a target_project relation' do
it 'accesses the value from the target_project' do
expect(merge_request.target_project)
.to receive(:merge_requests_disable_committers_approval?)
merge_request.merge_requests_disable_committers_approval?
end
end
end
end end
it_behaves_like 'an editable mentionable with EE-specific mentions' do it_behaves_like 'an editable mentionable with EE-specific mentions' do
......
...@@ -3082,32 +3082,83 @@ RSpec.describe MergeRequest, factory_default: :keep do ...@@ -3082,32 +3082,83 @@ RSpec.describe MergeRequest, factory_default: :keep do
end end
describe "#head_pipeline_active? " do describe "#head_pipeline_active? " do
it do context 'when project lacks a head_pipeline relation' do
is_expected before do
.to delegate_method(:active?) subject.head_pipeline = nil
.to(:head_pipeline) end
.with_prefix
.with_arguments(allow_nil: true) it 'returns false' do
expect(subject.head_pipeline_active?).to be false
end
end
context 'when project has a head_pipeline relation' do
let(:pipeline) { create(:ci_empty_pipeline) }
before do
allow(subject).to receive(:head_pipeline) { pipeline }
end
it 'accesses the value from the head_pipeline' do
expect(subject.head_pipeline)
.to receive(:active?)
subject.head_pipeline_active?
end
end end
end end
describe "#actual_head_pipeline_success? " do describe "#actual_head_pipeline_success? " do
it do context 'when project lacks an actual_head_pipeline relation' do
is_expected before do
.to delegate_method(:success?) allow(subject).to receive(:actual_head_pipeline) { nil }
.to(:actual_head_pipeline) end
.with_prefix
.with_arguments(allow_nil: true) it 'returns false' do
expect(subject.actual_head_pipeline_success?).to be false
end
end
context 'when project has a actual_head_pipeline relation' do
let(:pipeline) { create(:ci_empty_pipeline) }
before do
allow(subject).to receive(:actual_head_pipeline) { pipeline }
end
it 'accesses the value from the actual_head_pipeline' do
expect(subject.actual_head_pipeline)
.to receive(:success?)
subject.actual_head_pipeline_success?
end
end end
end end
describe "#actual_head_pipeline_active? " do describe "#actual_head_pipeline_active? " do
it do context 'when project lacks an actual_head_pipeline relation' do
is_expected before do
.to delegate_method(:active?) allow(subject).to receive(:actual_head_pipeline) { nil }
.to(:actual_head_pipeline) end
.with_prefix
.with_arguments(allow_nil: true) it 'returns false' do
expect(subject.actual_head_pipeline_active?).to be false
end
end
context 'when project has a actual_head_pipeline relation' do
let(:pipeline) { create(:ci_empty_pipeline) }
before do
allow(subject).to receive(:actual_head_pipeline) { pipeline }
end
it 'accesses the value from the actual_head_pipeline' do
expect(subject.actual_head_pipeline)
.to receive(:active?)
subject.actual_head_pipeline_active?
end
end end
end end
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment