debian-mirror-gitlab/spec/models/projects/forks/details_spec.rb
2023-05-27 22:25:52 +05:30

162 lines
5.6 KiB
Ruby

# frozen_string_literal: true
require 'spec_helper'
RSpec.describe Projects::Forks::Details, feature_category: :source_code_management do
include ExclusiveLeaseHelpers
include ProjectForksHelper
let_it_be(:user) { create(:user) }
let_it_be(:source_repo) { create(:project, :repository, :public).repository }
let_it_be(:fork_repo) { fork_project(source_repo.project, user, { repository: true }).repository }
let(:fork_branch) { 'fork-branch' }
let(:cache_key) { ['project_fork_details', fork_repo.project.id, fork_branch].join(':') }
describe '#counts', :use_clean_rails_redis_caching do
def expect_cached_counts(value)
counts = described_class.new(fork_repo.project, fork_branch).counts
ahead, behind = value
expect(counts).to eq({ ahead: ahead, behind: behind })
cached_value = {
source_sha: source_repo.commit.sha,
sha: fork_repo.commit(fork_branch).sha,
counts: value
}
expect(Rails.cache.read(cache_key)).to eq(cached_value)
end
it 'shows how far behind/ahead a fork is from the upstream' do
fork_repo.create_branch(fork_branch)
expect_cached_counts([0, 0])
fork_repo.commit_files(
user,
branch_name: fork_branch, message: 'Committing something',
actions: [{ action: :create, file_path: 'encoding/CHANGELOG', content: 'New file' }]
)
expect_cached_counts([1, 0])
fork_repo.commit_files(
user,
branch_name: fork_branch, message: 'Committing something else',
actions: [{ action: :create, file_path: 'encoding/ONE-MORE-CHANGELOG', content: 'One more new file' }]
)
expect_cached_counts([2, 0])
source_repo.commit_files(
user,
branch_name: source_repo.root_ref, message: 'Commit to root ref',
actions: [{ action: :create, file_path: 'encoding/CHANGELOG', content: 'One more' }]
)
expect_cached_counts([2, 1])
source_repo.commit_files(
user,
branch_name: source_repo.root_ref, message: 'Another commit to root ref',
actions: [{ action: :create, file_path: 'encoding/NEW-CHANGELOG', content: 'One more time' }]
)
expect_cached_counts([2, 2])
# When the fork is too far ahead
stub_const("#{described_class}::LATEST_COMMITS_COUNT", 1)
fork_repo.commit_files(
user,
branch_name: fork_branch, message: 'Another commit to fork',
actions: [{ action: :create, file_path: 'encoding/TOO-NEW-CHANGELOG', content: 'New file' }]
)
expect_cached_counts(nil)
end
context 'when counts calculated from a branch that exists upstream' do
let_it_be(:source_repo) { create(:project, :repository, :public).repository }
let_it_be(:fork_repo) { fork_project(source_repo.project, user, { repository: true }).repository }
let(:fork_branch) { 'feature' }
it 'compares the fork branch to upstream default branch' do
# The branch itself diverges from the upstream default branch
expect_cached_counts([1, 29])
source_repo.commit_files(
user,
branch_name: source_repo.root_ref, message: 'Commit to root ref',
actions: [{ action: :create, file_path: 'encoding/CHANGELOG', content: 'New file' }]
)
fork_repo.commit_files(
user,
branch_name: fork_branch, message: 'Committing to feature branch',
actions: [{ action: :create, file_path: 'encoding/FEATURE-BRANCH', content: 'New file' }]
)
# It takes into account diverged commits from upstream AND from fork
expect_cached_counts([2, 30])
end
end
context 'when specified branch does not exist' do
it 'returns nils as counts' do
counts = described_class.new(fork_repo.project, 'non-existent-branch').counts
expect(counts).to eq({ ahead: nil, behind: nil })
end
end
end
describe '#update!', :use_clean_rails_redis_caching do
it 'updates the cache with the specified value' do
value = { source_sha: source_repo.commit.sha, sha: fork_repo.commit.sha, counts: [0, 0], has_conflicts: true }
described_class.new(fork_repo.project, fork_branch).update!(value)
expect(Rails.cache.read(cache_key)).to eq(value)
end
end
describe '#has_conflicts', :use_clean_rails_redis_caching do
it 'returns whether merge for the stored commits failed due to conflicts' do
details = described_class.new(fork_repo.project, fork_branch)
expect do
value = { source_sha: source_repo.commit.sha, sha: fork_repo.commit.sha, counts: [0, 0], has_conflicts: true }
details.update!(value)
end.to change { details.has_conflicts? }.from(false).to(true)
end
end
describe '#exclusive_lease' do
it 'returns exclusive lease to the details' do
key = ['project_details', fork_repo.project.id, fork_branch].join(':')
uuid = SecureRandom.uuid
details = described_class.new(fork_repo.project, fork_branch)
expect(Gitlab::ExclusiveLease).to receive(:get_uuid).with(key).and_return(uuid)
expect(Gitlab::ExclusiveLease).to receive(:new).with(
key, uuid: uuid, timeout: described_class::LEASE_TIMEOUT
).and_call_original
expect(details.exclusive_lease).to be_a(Gitlab::ExclusiveLease)
end
end
describe 'syncing?', :use_clean_rails_redis_caching do
it 'returns whether there is a sync in progress' do
details = described_class.new(fork_repo.project, fork_branch)
expect(details.exclusive_lease.try_obtain).to be_present
expect(details.syncing?).to eq(true)
details.exclusive_lease.cancel
expect(details.syncing?).to eq(false)
end
end
end