Commit 3f8c78ec authored by Sean McGivern's avatar Sean McGivern

Merge branch 'pks-find-remote-root-ref-inmemory-remote' into 'master'

Use in-memory remotes to find remote root refs [RUN ALL RSPEC] [RUN AS-IF-FOSS]

See merge request gitlab-org/gitlab!60583
parents c92f0085 ad3c036a
...@@ -481,7 +481,7 @@ end ...@@ -481,7 +481,7 @@ end
gem 'spamcheck', '~> 0.0.5' gem 'spamcheck', '~> 0.0.5'
# Gitaly GRPC protocol definitions # Gitaly GRPC protocol definitions
gem 'gitaly', '~> 13.11.0.pre.rc1' gem 'gitaly', '~> 13.12.0.pre.rc1'
gem 'grpc', '~> 1.30.2' gem 'grpc', '~> 1.30.2'
......
...@@ -439,7 +439,7 @@ GEM ...@@ -439,7 +439,7 @@ GEM
rails (>= 3.2.0) rails (>= 3.2.0)
git (1.7.0) git (1.7.0)
rchardet (~> 1.8) rchardet (~> 1.8)
gitaly (13.11.0.pre.rc1) gitaly (13.12.0.pre.rc1)
grpc (~> 1.0) grpc (~> 1.0)
github-markup (1.7.0) github-markup (1.7.0)
gitlab (4.16.1) gitlab (4.16.1)
...@@ -1446,7 +1446,7 @@ DEPENDENCIES ...@@ -1446,7 +1446,7 @@ DEPENDENCIES
gettext (~> 3.3) gettext (~> 3.3)
gettext_i18n_rails (~> 1.8.0) gettext_i18n_rails (~> 1.8.0)
gettext_i18n_rails_js (~> 1.3) gettext_i18n_rails_js (~> 1.3)
gitaly (~> 13.11.0.pre.rc1) gitaly (~> 13.12.0.pre.rc1)
github-markup (~> 1.7.0) github-markup (~> 1.7.0)
gitlab-chronic (~> 0.10.5) gitlab-chronic (~> 0.10.5)
gitlab-dangerfiles (~> 2.0.0) gitlab-dangerfiles (~> 2.0.0)
......
---
name: find_remote_root_refs_inmemory
introduced_by_url: https://gitlab.com/gitlab-org/gitlab/-/merge_requests/60583
rollout_issue_url: https://gitlab.com/gitlab-org/gitlab/-/issues/329664
milestone: '13.12'
type: development
group: group::gitaly
default_enabled: true
...@@ -707,8 +707,8 @@ module EE ...@@ -707,8 +707,8 @@ module EE
end end
# Update the default branch querying the remote to determine its HEAD # Update the default branch querying the remote to determine its HEAD
def update_root_ref(remote_name) def update_root_ref(remote, remote_url, authorization)
root_ref = repository.find_remote_root_ref(remote_name) root_ref = repository.find_remote_root_ref(remote, remote_url, authorization)
change_head(root_ref) if root_ref.present? change_head(root_ref) if root_ref.present?
end end
......
...@@ -50,9 +50,16 @@ module Geo ...@@ -50,9 +50,16 @@ module Geo
end end
def update_root_ref def update_root_ref
# Find the remote root ref, using a JWT header for authentication if Feature.enabled?(:find_remote_root_refs_inmemory, default_enabled: :yaml)
repository.with_config(jwt_authentication_header) do authorization = ::Gitlab::Geo::RepoSyncRequest.new(
project.update_root_ref(GEO_REMOTE_NAME) scope: repository.full_path
).authorization
project.update_root_ref(GEO_REMOTE_NAME, remote_url, authorization)
else
repository_with_config(jwt_authentication_header) do
project.update_root_ref(GEO_REMOTE_NAME, remote_url)
end
end end
end end
......
...@@ -2147,11 +2147,13 @@ RSpec.describe Project do ...@@ -2147,11 +2147,13 @@ RSpec.describe Project do
describe '#update_root_ref' do describe '#update_root_ref' do
let(:project) { create(:project, :repository) } let(:project) { create(:project, :repository) }
let(:url) { 'http://git.example.com/remote-repo.git' }
let(:auth) { 'Basic secret' }
it 'updates the default branch when HEAD has changed' do it 'updates the default branch when HEAD has changed' do
stub_find_remote_root_ref(project, ref: 'feature') stub_find_remote_root_ref(project, ref: 'feature')
expect { project.update_root_ref('origin') } expect { project.update_root_ref('origin', url, auth) }
.to change { project.default_branch } .to change { project.default_branch }
.from('master') .from('master')
.to('feature') .to('feature')
...@@ -2162,7 +2164,7 @@ RSpec.describe Project do ...@@ -2162,7 +2164,7 @@ RSpec.describe Project do
expect(project).to receive(:change_head).with('master').and_call_original expect(project).to receive(:change_head).with('master').and_call_original
project.update_root_ref('origin') project.update_root_ref('origin', url, auth)
# For good measure, expunge the root ref cache and reload. # For good measure, expunge the root ref cache and reload.
project.repository.expire_all_method_caches project.repository.expire_all_method_caches
...@@ -2172,14 +2174,14 @@ RSpec.describe Project do ...@@ -2172,14 +2174,14 @@ RSpec.describe Project do
it 'does not update the default branch when HEAD does not exist' do it 'does not update the default branch when HEAD does not exist' do
stub_find_remote_root_ref(project, ref: 'foo') stub_find_remote_root_ref(project, ref: 'foo')
expect { project.update_root_ref('origin') } expect { project.update_root_ref('origin', url, auth) }
.not_to change { project.default_branch } .not_to change { project.default_branch }
end end
def stub_find_remote_root_ref(project, ref:) def stub_find_remote_root_ref(project, ref:)
allow(project.repository) allow(project.repository)
.to receive(:find_remote_root_ref) .to receive(:find_remote_root_ref)
.with('origin') .with('origin', url, auth)
.and_return(ref) .and_return(ref)
end end
end end
......
...@@ -36,7 +36,7 @@ RSpec.describe Geo::RepositorySyncService, :geo do ...@@ -36,7 +36,7 @@ RSpec.describe Geo::RepositorySyncService, :geo do
allow_any_instance_of(Repository) allow_any_instance_of(Repository)
.to receive(:find_remote_root_ref) .to receive(:find_remote_root_ref)
.with('geo') .with('geo', url_to_repo, anything)
.and_return('master') .and_return('master')
allow_any_instance_of(Geo::ProjectHousekeepingService).to receive(:execute) allow_any_instance_of(Geo::ProjectHousekeepingService).to receive(:execute)
...@@ -48,7 +48,6 @@ RSpec.describe Geo::RepositorySyncService, :geo do ...@@ -48,7 +48,6 @@ RSpec.describe Geo::RepositorySyncService, :geo do
it 'fetches project repository with JWT credentials' do it 'fetches project repository with JWT credentials' do
expect(repository).to receive(:with_config) expect(repository).to receive(:with_config)
.with("http.#{url_to_repo}.extraHeader" => anything) .with("http.#{url_to_repo}.extraHeader" => anything)
.twice
.and_call_original .and_call_original
expect(repository).to receive(:fetch_as_mirror) expect(repository).to receive(:fetch_as_mirror)
...@@ -238,7 +237,7 @@ RSpec.describe Geo::RepositorySyncService, :geo do ...@@ -238,7 +237,7 @@ RSpec.describe Geo::RepositorySyncService, :geo do
before do before do
allow(project.repository) allow(project.repository)
.to receive(:find_remote_root_ref) .to receive(:find_remote_root_ref)
.with('geo') .with('geo', url_to_repo, anything)
.and_return('feature') .and_return('feature')
end end
...@@ -251,7 +250,6 @@ RSpec.describe Geo::RepositorySyncService, :geo do ...@@ -251,7 +250,6 @@ RSpec.describe Geo::RepositorySyncService, :geo do
it 'updates the default branch with JWT credentials' do it 'updates the default branch with JWT credentials' do
expect(repository).to receive(:with_config) expect(repository).to receive(:with_config)
.with("http.#{url_to_repo}.extraHeader" => anything) .with("http.#{url_to_repo}.extraHeader" => anything)
.twice
.and_call_original .and_call_original
expect(project).to receive(:change_head).with('feature').once expect(project).to receive(:change_head).with('feature').once
...@@ -264,7 +262,7 @@ RSpec.describe Geo::RepositorySyncService, :geo do ...@@ -264,7 +262,7 @@ RSpec.describe Geo::RepositorySyncService, :geo do
before do before do
allow(project.repository) allow(project.repository)
.to receive(:find_remote_root_ref) .to receive(:find_remote_root_ref)
.with('geo') .with('geo', url_to_repo, anything)
.and_return(project.default_branch) .and_return(project.default_branch)
end end
...@@ -277,7 +275,6 @@ RSpec.describe Geo::RepositorySyncService, :geo do ...@@ -277,7 +275,6 @@ RSpec.describe Geo::RepositorySyncService, :geo do
it 'updates the default branch with JWT credentials' do it 'updates the default branch with JWT credentials' do
expect(repository).to receive(:with_config) expect(repository).to receive(:with_config)
.with("http.#{url_to_repo}.extraHeader" => anything) .with("http.#{url_to_repo}.extraHeader" => anything)
.twice
.and_call_original .and_call_original
expect(project).to receive(:change_head).with('master').once expect(project).to receive(:change_head).with('master').once
......
...@@ -700,11 +700,11 @@ module Gitlab ...@@ -700,11 +700,11 @@ module Gitlab
end end
end end
def find_remote_root_ref(remote_name) def find_remote_root_ref(remote_name, remote_url, authorization = nil)
return unless remote_name.present? return unless remote_name.present? && remote_url.present?
wrapped_gitaly_errors do wrapped_gitaly_errors do
gitaly_remote_client.find_remote_root_ref(remote_name) gitaly_remote_client.find_remote_root_ref(remote_name, remote_url, authorization)
end end
end end
......
...@@ -43,11 +43,20 @@ module Gitlab ...@@ -43,11 +43,20 @@ module Gitlab
GitalyClient.call(@storage, :remote_service, :remove_remote, request, timeout: GitalyClient.long_timeout).result GitalyClient.call(@storage, :remote_service, :remove_remote, request, timeout: GitalyClient.long_timeout).result
end end
def find_remote_root_ref(remote_name) # The remote_name parameter is deprecated and will be removed soon.
request = Gitaly::FindRemoteRootRefRequest.new( def find_remote_root_ref(remote_name, remote_url, authorization)
repository: @gitaly_repo, request = if Feature.enabled?(:find_remote_root_refs_inmemory, default_enabled: :yaml)
remote: remote_name Gitaly::FindRemoteRootRefRequest.new(
) repository: @gitaly_repo,
remote_url: remote_url,
http_authorization_header: authorization
)
else
Gitaly::FindRemoteRootRefRequest.new(
repository: @gitaly_repo,
remote: remote_name
)
end
response = GitalyClient.call(@storage, :remote_service, response = GitalyClient.call(@storage, :remote_service,
:find_remote_root_ref, request, timeout: GitalyClient.medium_timeout) :find_remote_root_ref, request, timeout: GitalyClient.medium_timeout)
......
...@@ -604,29 +604,29 @@ RSpec.describe Gitlab::Git::Repository, :seed_helper do ...@@ -604,29 +604,29 @@ RSpec.describe Gitlab::Git::Repository, :seed_helper do
expect_any_instance_of(Gitlab::GitalyClient::RemoteService) expect_any_instance_of(Gitlab::GitalyClient::RemoteService)
.to receive(:find_remote_root_ref).and_call_original .to receive(:find_remote_root_ref).and_call_original
expect(repository.find_remote_root_ref('origin')).to eq 'master' expect(repository.find_remote_root_ref('origin', SeedHelper::GITLAB_GIT_TEST_REPO_URL)).to eq 'master'
end end
it 'returns UTF-8' do it 'returns UTF-8' do
expect(repository.find_remote_root_ref('origin')).to be_utf8 expect(repository.find_remote_root_ref('origin', SeedHelper::GITLAB_GIT_TEST_REPO_URL)).to be_utf8
end end
it 'returns nil when remote name is nil' do it 'returns nil when remote name is nil' do
expect_any_instance_of(Gitlab::GitalyClient::RemoteService) expect_any_instance_of(Gitlab::GitalyClient::RemoteService)
.not_to receive(:find_remote_root_ref) .not_to receive(:find_remote_root_ref)
expect(repository.find_remote_root_ref(nil)).to be_nil expect(repository.find_remote_root_ref(nil, nil)).to be_nil
end end
it 'returns nil when remote name is empty' do it 'returns nil when remote name is empty' do
expect_any_instance_of(Gitlab::GitalyClient::RemoteService) expect_any_instance_of(Gitlab::GitalyClient::RemoteService)
.not_to receive(:find_remote_root_ref) .not_to receive(:find_remote_root_ref)
expect(repository.find_remote_root_ref('')).to be_nil expect(repository.find_remote_root_ref('', '')).to be_nil
end end
it_behaves_like 'wrapping gRPC errors', Gitlab::GitalyClient::RemoteService, :find_remote_root_ref do it_behaves_like 'wrapping gRPC errors', Gitlab::GitalyClient::RemoteService, :find_remote_root_ref do
subject { repository.find_remote_root_ref('origin') } subject { repository.find_remote_root_ref('origin', SeedHelper::GITLAB_GIT_TEST_REPO_URL) }
end end
end end
......
...@@ -35,22 +35,50 @@ RSpec.describe Gitlab::GitalyClient::RemoteService do ...@@ -35,22 +35,50 @@ RSpec.describe Gitlab::GitalyClient::RemoteService do
end end
describe '#find_remote_root_ref' do describe '#find_remote_root_ref' do
it 'sends an find_remote_root_ref message and returns the root ref' do let(:remote) { 'origin' }
expect_any_instance_of(Gitaly::RemoteService::Stub) let(:url) { 'http://git.example.com/my-repo.git' }
.to receive(:find_remote_root_ref) let(:auth) { 'Basic secret' }
.with(gitaly_request_with_path(storage_name, relative_path), kind_of(Hash))
.and_return(double(ref: 'master')) shared_examples 'a find_remote_root_ref call' do
it 'sends an find_remote_root_ref message and returns the root ref' do
expect_any_instance_of(Gitaly::RemoteService::Stub)
.to receive(:find_remote_root_ref)
.with(gitaly_request_with_path(storage_name, relative_path), kind_of(Hash))
.with(gitaly_request_with_params(expected_params), kind_of(Hash))
.and_return(double(ref: 'master'))
expect(client.find_remote_root_ref(remote, url, auth)).to eq 'master'
end
it 'ensure ref is a valid UTF-8 string' do
expect_any_instance_of(Gitaly::RemoteService::Stub)
.to receive(:find_remote_root_ref)
.with(gitaly_request_with_path(storage_name, relative_path), kind_of(Hash))
.with(gitaly_request_with_params(expected_params), kind_of(Hash))
.and_return(double(ref: "an_invalid_ref_\xE5"))
expect(client.find_remote_root_ref('origin')).to eq 'master' expect(client.find_remote_root_ref(remote, url, auth)).to eq "an_invalid_ref_å"
end
end end
it 'ensure ref is a valid UTF-8 string' do context 'with inmemory feature enabled' do
expect_any_instance_of(Gitaly::RemoteService::Stub) before do
.to receive(:find_remote_root_ref) stub_feature_flags(find_remote_root_refs_inmemory: true)
.with(gitaly_request_with_path(storage_name, relative_path), kind_of(Hash)) end
.and_return(double(ref: "an_invalid_ref_\xE5"))
it_behaves_like 'a find_remote_root_ref call' do
let(:expected_params) { { remote_url: url, http_authorization_header: auth } }
end
end
expect(client.find_remote_root_ref('origin')).to eq "an_invalid_ref_å" context 'with inmemory feature disabled' do
before do
stub_feature_flags(find_remote_root_refs_inmemory: false)
end
it_behaves_like 'a find_remote_root_ref call' do
let(:expected_params) { { remote: remote } }
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