Commit a979aeac authored by Ash McKenzie's avatar Ash McKenzie

Geo: LFS batch downloads handled by secondary

parent 9554796e
...@@ -3,12 +3,6 @@ module EE ...@@ -3,12 +3,6 @@ module EE
module GitHttpClientController module GitHttpClientController
extend ActiveSupport::Concern extend ActiveSupport::Concern
ALLOWED_CONTROLLER_AND_ACTIONS = {
'git_http' => %w{git_receive_pack},
'lfs_api' => %w{batch},
'lfs_locks_api' => %w{create unlock verify}
}.freeze
prepended do prepended do
before_action do before_action do
redirect_to(primary_full_url) if redirect? redirect_to(primary_full_url) if redirect?
...@@ -20,44 +14,52 @@ module EE ...@@ -20,44 +14,52 @@ module EE
class RouteHelper class RouteHelper
attr_reader :controller_name, :action_name attr_reader :controller_name, :action_name
def initialize(controller_name, action_name, params, allowed) ALLOWED_CONTROLLER_AND_ACTIONS = {
'git_http' => %w{git_receive_pack},
'lfs_locks_api' => %w{create unlock verify}
}.freeze
def initialize(controller_name, action_name, service)
@controller_name = controller_name @controller_name = controller_name
@action_name = action_name @action_name = action_name
@params = params @service = service
@allowed = allowed
end end
def match?(c_name, a_name) def match?(c_name, a_name)
controller_name == c_name && action_name == a_name controller_name == c_name && action_name == a_name
end end
def allowed_match? def redirect?
!!allowed[controller_name]&.include?(action_name) git_receive_pack? || controller_and_action_match?
end end
def service_or_action_name private
action_name == 'info_refs' ? params[:service] : action_name.dasherize
attr_reader :service
def git_receive_pack?
service_or_action_name == 'git-receive-pack'
end end
private def controller_and_action_match?
!!ALLOWED_CONTROLLER_AND_ACTIONS[controller_name]&.include?(action_name)
end
attr_reader :params, :allowed def service_or_action_name
action_name == 'info_refs' ? service : action_name.dasherize
end
end end
class GitLFSHelper class GitLFSHelper
MINIMUM_GIT_LFS_VERSION = '2.4.2'.freeze MINIMUM_GIT_LFS_VERSION = '2.4.2'.freeze
def initialize(current_version) def initialize(route_helper, operation, current_version)
@route_helper = route_helper
@operation = operation
@current_version = current_version @current_version = current_version
end end
def version_ok? def incorrect_version_response
return false unless current_version
::Gitlab::VersionInfo.parse(current_version) >= wanted_version
end
def incorrect_version_opts
{ {
json: { message: incorrect_version_message }, json: { message: incorrect_version_message },
content_type: ::LfsRequest::CONTENT_TYPE, content_type: ::LfsRequest::CONTENT_TYPE,
...@@ -65,27 +67,44 @@ module EE ...@@ -65,27 +67,44 @@ module EE
} }
end end
private def redirect?
return false unless route_helper.match?('lfs_api', 'batch')
return true if upload?
attr_reader :current_version false
end
def wanted_version def version_ok?
::Gitlab::VersionInfo.parse(MINIMUM_GIT_LFS_VERSION) return false unless current_version
::Gitlab::VersionInfo.parse(current_version) >= wanted_version
end end
private
attr_reader :route_helper, :operation, :current_version
def incorrect_version_message def incorrect_version_message
translation = _("You need git-lfs version %{min_git_lfs_version} (or greater) to continue. Please visit https://git-lfs.github.com") translation = _("You need git-lfs version %{min_git_lfs_version} (or greater) to continue. Please visit https://git-lfs.github.com")
translation % { min_git_lfs_version: MINIMUM_GIT_LFS_VERSION } translation % { min_git_lfs_version: MINIMUM_GIT_LFS_VERSION }
end end
def upload?
operation == 'upload'
end
def wanted_version
::Gitlab::VersionInfo.parse(MINIMUM_GIT_LFS_VERSION)
end
end end
def route_helper def route_helper
@route_helper ||= RouteHelper.new(controller_name, action_name, params, @route_helper ||= RouteHelper.new(controller_name, action_name, params[:service])
ALLOWED_CONTROLLER_AND_ACTIONS)
end end
def git_lfs_helper def git_lfs_helper
@git_lfs_helper ||= GitLFSHelper.new(current_git_lfs_version) # params[:operation] explained: https://github.com/git-lfs/git-lfs/blob/master/docs/api/batch.md#requests
@git_lfs_helper ||= GitLFSHelper.new(route_helper, params[:operation], request.headers['User-Agent'])
end end
def request_fullpath_for_primary def request_fullpath_for_primary
...@@ -97,23 +116,18 @@ module EE ...@@ -97,23 +116,18 @@ module EE
File.join(::Gitlab::Geo.primary_node.url, request_fullpath_for_primary) File.join(::Gitlab::Geo.primary_node.url, request_fullpath_for_primary)
end end
def current_git_lfs_version
request.headers['User-Agent']
end
def redirect? def redirect?
::Gitlab::Geo.secondary_with_primary? && match? && !filtered? return false unless ::Gitlab::Geo.secondary_with_primary?
end return true if route_helper.redirect?
def match? if git_lfs_helper.redirect?
route_helper.service_or_action_name == 'git-receive-pack' || return true if git_lfs_helper.version_ok?
route_helper.allowed_match?
end
def filtered? # git-lfs 2.4.2 is really only required for requests that involve
if route_helper.match?('lfs_api', 'batch') && !git_lfs_helper.version_ok? # redirection, so we only render if it's an LFS upload operation
render(git_lfs_helper.incorrect_version_opts) #
return true render(git_lfs_helper.incorrect_version_response)
return false
end end
false false
......
---
title: 'Geo: LFS batch downloads are OK to be handled by secondary'
merge_request: 7209
author:
type: fixed
...@@ -121,11 +121,10 @@ describe "Git HTTP requests (Geo)" do ...@@ -121,11 +121,10 @@ describe "Git HTTP requests (Geo)" do
context 'API' do context 'API' do
describe 'POST batch' do describe 'POST batch' do
def make_request def make_request
post url, {}, env.merge(extra_env) post url, args, env
end end
let(:extra_env) { {} } let(:args) { {} }
let(:incorrect_version_regex) { /You need git-lfs version 2.4.2/ }
let(:url) { "/#{project.full_path}.git/info/lfs/objects/batch" } let(:url) { "/#{project.full_path}.git/info/lfs/objects/batch" }
subject do subject do
...@@ -133,28 +132,82 @@ describe "Git HTTP requests (Geo)" do ...@@ -133,28 +132,82 @@ describe "Git HTTP requests (Geo)" do
response response
end end
context 'with the correct git-lfs version' do before do
let(:extra_env) { { 'User-Agent' => 'git-lfs/2.4.2 (GitHub; darwin amd64; go 1.10.2)' } } allow(Gitlab.config.lfs).to receive(:enabled).and_return(true)
project.update_attribute(:lfs_enabled, true)
env['Content-Type'] = LfsRequest::CONTENT_TYPE
end
it 'redirects to the primary' do context 'operation upload' do
is_expected.to have_gitlab_http_status(:redirect) let(:args) { { 'operation' => 'upload' }.to_json }
redirect_location = "#{primary.url.chomp('/')}#{url}"
expect(subject.header['Location']).to eq(redirect_location) context 'with the correct git-lfs version' do
before do
env['User-Agent'] = 'git-lfs/2.4.2 (GitHub; darwin amd64; go 1.10.2)'
end
it 'redirects to the primary' do
is_expected.to have_gitlab_http_status(:redirect)
redirect_location = "#{primary.url.chomp('/')}#{url}"
expect(subject.header['Location']).to eq(redirect_location)
end
end end
end
where(:description, :version) do context 'with an incorrect git-lfs version' do
'outdated' | 'git-lfs/2.4.1' where(:description, :version) do
'unknown' | 'git-lfs' 'outdated' | 'git-lfs/2.4.1'
'unknown' | 'git-lfs'
end
with_them do
context "that is #{description}" do
before do
env['User-Agent'] = "#{version} (GitHub; darwin amd64; go 1.10.2)"
end
it 'is forbidden' do
is_expected.to have_gitlab_http_status(:forbidden)
expect(json_response['message']).to match(/You need git-lfs version 2.4.2/)
end
end
end
end
end end
with_them do context 'operation download' do
context "with an #{description} git-lfs version" do let(:user) { create(:user) }
let(:extra_env) { { 'User-Agent' => "#{version} (GitHub; darwin amd64; go 1.10.2)" } } let(:authorization) { ActionController::HttpAuthentication::Basic.encode_credentials(user.username, user.password) }
let(:lfs_object) { create(:lfs_object, :with_file) }
let(:args) do
{
'operation' => 'download',
'objects' => [{ 'oid' => lfs_object.oid, 'size' => lfs_object.size }]
}.to_json
end
before do
project.add_maintainer(user)
env['Authorization'] = authorization
end
it 'is handled by the secondary' do
is_expected.to have_gitlab_http_status(:ok)
end
where(:description, :version) do
'outdated' | 'git-lfs/2.4.1'
'unknown' | 'git-lfs'
end
with_them do
context "with an #{description} git-lfs version" do
before do
env['User-Agent'] = "#{version} (GitHub; darwin amd64; go 1.10.2)"
end
it 'errors out' do it 'is handled by the secondary' do
is_expected.to have_gitlab_http_status(:forbidden) is_expected.to have_gitlab_http_status(:ok)
expect(json_response['message']).to match(incorrect_version_regex) end
end 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