Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
G
gitlab-ce
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
1
Merge Requests
1
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
gitlab-ce
Commits
a979aeac
Commit
a979aeac
authored
Sep 03, 2018
by
Ash McKenzie
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Geo: LFS batch downloads handled by secondary
parent
9554796e
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
135 additions
and
63 deletions
+135
-63
ee/app/controllers/ee/projects/git_http_client_controller.rb
ee/app/controllers/ee/projects/git_http_client_controller.rb
+58
-44
ee/changelogs/unreleased/7419-running-git-clone-with-a-secondary-geo-node-url-has-lfs-batch-download-redirected-to-primary.yml
...node-url-has-lfs-batch-download-redirected-to-primary.yml
+5
-0
ee/spec/requests/git_http_geo_spec.rb
ee/spec/requests/git_http_geo_spec.rb
+72
-19
No files found.
ee/app/controllers/ee/projects/git_http_client_controller.rb
View file @
a979aeac
...
@@ -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
...
...
ee/changelogs/unreleased/7419-running-git-clone-with-a-secondary-geo-node-url-has-lfs-batch-download-redirected-to-primary.yml
0 → 100644
View file @
a979aeac
---
title
:
'
Geo:
LFS
batch
downloads
are
OK
to
be
handled
by
secondary'
merge_request
:
7209
author
:
type
:
fixed
ee/spec/requests/git_http_geo_spec.rb
View file @
a979aeac
...
@@ -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
)
e
xpect
(
json_response
[
'message'
]).
to
match
(
incorrect_version_regex
)
e
nd
end
end
end
end
end
end
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment