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
b2695395
Commit
b2695395
authored
Nov 30, 2021
by
Albert Salim
Committed by
Aleksei Lipniagov
Nov 30, 2021
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Add test coverage and refactor existing tests on job artifacts
parent
cb74e49b
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
679 additions
and
624 deletions
+679
-624
spec/requests/api/ci/job_artifacts_spec.rb
spec/requests/api/ci/job_artifacts_spec.rb
+629
-0
spec/requests/api/ci/jobs_spec.rb
spec/requests/api/ci/jobs_spec.rb
+0
-589
spec/services/ci/job_artifacts/destroy_batch_service_spec.rb
spec/services/ci/job_artifacts/destroy_batch_service_spec.rb
+50
-35
No files found.
spec/requests/api/ci/job_artifacts_spec.rb
0 → 100644
View file @
b2695395
# frozen_string_literal: true
require
'spec_helper'
RSpec
.
describe
API
::
Ci
::
JobArtifacts
do
include
HttpBasicAuthHelpers
include
DependencyProxyHelpers
include
HttpIOHelpers
let_it_be
(
:project
,
reload:
true
)
do
create
(
:project
,
:repository
,
public_builds:
false
)
end
let_it_be
(
:pipeline
,
reload:
true
)
do
create
(
:ci_pipeline
,
project:
project
,
sha:
project
.
commit
.
id
,
ref:
project
.
default_branch
)
end
let
(
:user
)
{
create
(
:user
)
}
let
(
:api_user
)
{
user
}
let
(
:reporter
)
{
create
(
:project_member
,
:reporter
,
project:
project
).
user
}
let
(
:guest
)
{
create
(
:project_member
,
:guest
,
project:
project
).
user
}
let!
(
:job
)
do
create
(
:ci_build
,
:success
,
:tags
,
pipeline:
pipeline
,
artifacts_expire_at:
1
.
day
.
since
)
end
before
do
project
.
add_developer
(
user
)
end
shared_examples
'returns unauthorized'
do
it
'returns unauthorized'
do
expect
(
response
).
to
have_gitlab_http_status
(
:unauthorized
)
end
end
describe
'DELETE /projects/:id/jobs/:job_id/artifacts'
do
let!
(
:job
)
{
create
(
:ci_build
,
:artifacts
,
pipeline:
pipeline
,
user:
api_user
)
}
before
do
delete
api
(
"/projects/
#{
project
.
id
}
/jobs/
#{
job
.
id
}
/artifacts"
,
api_user
)
end
context
'when user is anonymous'
do
let
(
:api_user
)
{
nil
}
it
'does not delete artifacts'
do
expect
(
job
.
job_artifacts
.
size
).
to
eq
2
end
it
'returns status 401 (unauthorized)'
do
expect
(
response
).
to
have_gitlab_http_status
(
:unauthorized
)
end
end
context
'with developer'
do
it
'does not delete artifacts'
do
expect
(
job
.
job_artifacts
.
size
).
to
eq
2
end
it
'returns status 403 (forbidden)'
do
expect
(
response
).
to
have_gitlab_http_status
(
:forbidden
)
end
end
context
'with authorized user'
do
let
(
:maintainer
)
{
create
(
:project_member
,
:maintainer
,
project:
project
).
user
}
let!
(
:api_user
)
{
maintainer
}
it
'deletes artifacts'
do
expect
(
job
.
job_artifacts
.
size
).
to
eq
0
end
it
'returns status 204 (no content)'
do
expect
(
response
).
to
have_gitlab_http_status
(
:no_content
)
end
end
end
describe
'GET /projects/:id/jobs/:job_id/artifacts/:artifact_path'
do
context
'when job has artifacts'
do
let
(
:job
)
{
create
(
:ci_build
,
:artifacts
,
pipeline:
pipeline
)
}
let
(
:artifact
)
do
'other_artifacts_0.1.2/another-subdirectory/banana_sample.gif'
end
context
'when user is anonymous'
do
let
(
:api_user
)
{
nil
}
context
'when project is public'
do
it
'allows to access artifacts'
do
project
.
update_column
(
:visibility_level
,
Gitlab
::
VisibilityLevel
::
PUBLIC
)
project
.
update_column
(
:public_builds
,
true
)
get_artifact_file
(
artifact
)
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
end
end
context
'when project is public with artifacts that are non public'
do
let
(
:job
)
{
create
(
:ci_build
,
:artifacts
,
:non_public_artifacts
,
pipeline:
pipeline
)
}
it
'rejects access to artifacts'
do
project
.
update_column
(
:visibility_level
,
Gitlab
::
VisibilityLevel
::
PUBLIC
)
project
.
update_column
(
:public_builds
,
true
)
get_artifact_file
(
artifact
)
expect
(
response
).
to
have_gitlab_http_status
(
:forbidden
)
end
context
'with the non_public_artifacts feature flag disabled'
do
before
do
stub_feature_flags
(
non_public_artifacts:
false
)
end
it
'allows access to artifacts'
do
project
.
update_column
(
:visibility_level
,
Gitlab
::
VisibilityLevel
::
PUBLIC
)
project
.
update_column
(
:public_builds
,
true
)
get_artifact_file
(
artifact
)
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
end
end
end
context
'when project is public with builds access disabled'
do
it
'rejects access to artifacts'
do
project
.
update_column
(
:visibility_level
,
Gitlab
::
VisibilityLevel
::
PUBLIC
)
project
.
update_column
(
:public_builds
,
false
)
get_artifact_file
(
artifact
)
expect
(
response
).
to
have_gitlab_http_status
(
:forbidden
)
end
end
context
'when project is private'
do
it
'rejects access and hides existence of artifacts'
do
project
.
update_column
(
:visibility_level
,
Gitlab
::
VisibilityLevel
::
PRIVATE
)
project
.
update_column
(
:public_builds
,
true
)
get_artifact_file
(
artifact
)
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
end
end
end
context
'when user is authorized'
do
it
'returns a specific artifact file for a valid path'
do
expect
(
Gitlab
::
Workhorse
)
.
to
receive
(
:send_artifacts_entry
)
.
and_call_original
get_artifact_file
(
artifact
)
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
expect
(
response
.
headers
.
to_h
)
.
to
include
(
'Content-Type'
=>
'application/json'
,
'Gitlab-Workhorse-Send-Data'
=>
/artifacts-entry/
)
expect
(
response
.
parsed_body
).
to
be_empty
end
context
'when artifacts are locked'
do
it
'allows access to expired artifact'
do
pipeline
.
artifacts_locked!
job
.
update!
(
artifacts_expire_at:
Time
.
now
-
7
.
days
)
get_artifact_file
(
artifact
)
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
end
end
end
end
context
'when job does not have artifacts'
do
it
'does not return job artifact file'
do
get_artifact_file
(
'some/artifact'
)
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
end
end
def
get_artifact_file
(
artifact_path
)
get
api
(
"/projects/
#{
project
.
id
}
/jobs/
#{
job
.
id
}
/"
\
"artifacts/
#{
artifact_path
}
"
,
api_user
)
end
end
describe
'GET /projects/:id/jobs/:job_id/artifacts'
do
shared_examples
'downloads artifact'
do
let
(
:download_headers
)
do
{
'Content-Transfer-Encoding'
=>
'binary'
,
'Content-Disposition'
=>
%q(attachment; filename="ci_build_artifacts.zip"; filename*=UTF-8''ci_build_artifacts.zip)
}
end
it
'returns specific job artifacts'
do
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
expect
(
response
.
headers
.
to_h
).
to
include
(
download_headers
)
expect
(
response
.
body
).
to
match_file
(
job
.
artifacts_file
.
file
.
file
)
end
end
context
'normal authentication'
do
context
'job with artifacts'
do
context
'when artifacts are stored locally'
do
let
(
:job
)
{
create
(
:ci_build
,
:artifacts
,
pipeline:
pipeline
)
}
before
do
get
api
(
"/projects/
#{
project
.
id
}
/jobs/
#{
job
.
id
}
/artifacts"
,
api_user
)
end
context
'authorized user'
do
it_behaves_like
'downloads artifact'
end
context
'unauthorized user'
do
let
(
:api_user
)
{
nil
}
it
'does not return specific job artifacts'
do
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
end
end
end
context
'when artifacts are stored remotely'
do
let
(
:proxy_download
)
{
false
}
let
(
:job
)
{
create
(
:ci_build
,
pipeline:
pipeline
)
}
let
(
:artifact
)
{
create
(
:ci_job_artifact
,
:archive
,
:remote_store
,
job:
job
)
}
before
do
stub_artifacts_object_storage
(
proxy_download:
proxy_download
)
artifact
job
.
reload
get
api
(
"/projects/
#{
project
.
id
}
/jobs/
#{
job
.
id
}
/artifacts"
,
api_user
)
end
context
'when proxy download is enabled'
do
let
(
:proxy_download
)
{
true
}
it
'responds with the workhorse send-url'
do
expect
(
response
.
headers
[
Gitlab
::
Workhorse
::
SEND_DATA_HEADER
]).
to
start_with
(
"send-url:"
)
end
end
context
'when proxy download is disabled'
do
it
'returns location redirect'
do
expect
(
response
).
to
have_gitlab_http_status
(
:found
)
end
end
context
'authorized user'
do
it
'returns the file remote URL'
do
expect
(
response
).
to
redirect_to
(
artifact
.
file
.
url
)
end
end
context
'unauthorized user'
do
let
(
:api_user
)
{
nil
}
it
'does not return specific job artifacts'
do
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
end
end
end
context
'when public project guest and artifacts are non public'
do
let
(
:api_user
)
{
guest
}
let
(
:job
)
{
create
(
:ci_build
,
:artifacts
,
:non_public_artifacts
,
pipeline:
pipeline
)
}
before
do
project
.
update_column
(
:visibility_level
,
Gitlab
::
VisibilityLevel
::
PUBLIC
)
project
.
update_column
(
:public_builds
,
true
)
get
api
(
"/projects/
#{
project
.
id
}
/jobs/
#{
job
.
id
}
/artifacts"
,
api_user
)
end
it
'rejects access and hides existence of artifacts'
do
expect
(
response
).
to
have_gitlab_http_status
(
:forbidden
)
end
context
'with the non_public_artifacts feature flag disabled'
do
before
do
stub_feature_flags
(
non_public_artifacts:
false
)
get
api
(
"/projects/
#{
project
.
id
}
/jobs/
#{
job
.
id
}
/artifacts"
,
api_user
)
end
it
'allows access to artifacts'
do
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
end
end
end
it
'does not return job artifacts if not uploaded'
do
get
api
(
"/projects/
#{
project
.
id
}
/jobs/
#{
job
.
id
}
/artifacts"
,
api_user
)
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
end
end
end
end
describe
'GET /projects/:id/artifacts/:ref_name/download?job=name'
do
let
(
:api_user
)
{
reporter
}
let
(
:job
)
{
create
(
:ci_build
,
:artifacts
,
pipeline:
pipeline
,
user:
api_user
)
}
before
do
stub_artifacts_object_storage
job
.
success
end
def
get_for_ref
(
ref
=
pipeline
.
ref
,
job_name
=
job
.
name
)
get
api
(
"/projects/
#{
project
.
id
}
/jobs/artifacts/
#{
ref
}
/download"
,
api_user
),
params:
{
job:
job_name
}
end
context
'when not logged in'
do
let
(
:api_user
)
{
nil
}
before
do
get_for_ref
end
it
'does not find a resource in a private project'
do
expect
(
project
).
to
be_private
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
end
end
context
'when logging as guest'
do
let
(
:api_user
)
{
guest
}
before
do
get_for_ref
end
it
'gives 403'
do
expect
(
response
).
to
have_gitlab_http_status
(
:forbidden
)
end
end
context
'non-existing job'
do
shared_examples
'not found'
do
it
{
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
}
end
context
'has no such ref'
do
before
do
get_for_ref
(
'TAIL'
)
end
it_behaves_like
'not found'
end
context
'has no such job'
do
before
do
get_for_ref
(
pipeline
.
ref
,
'NOBUILD'
)
end
it_behaves_like
'not found'
end
end
context
'find proper job'
do
let
(
:job_with_artifacts
)
{
job
}
shared_examples
'a valid file'
do
context
'when artifacts are stored locally'
,
:sidekiq_might_not_need_inline
do
let
(
:download_headers
)
do
{
'Content-Transfer-Encoding'
=>
'binary'
,
'Content-Disposition'
=>
%Q(attachment; filename="
#{
job_with_artifacts
.
artifacts_file
.
filename
}
"; filename*=UTF-8''
#{
job
.
artifacts_file
.
filename
}
)
}
end
it
{
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
}
it
{
expect
(
response
.
headers
.
to_h
).
to
include
(
download_headers
)
}
end
context
'when artifacts are stored remotely'
do
let
(
:job
)
{
create
(
:ci_build
,
pipeline:
pipeline
,
user:
api_user
)
}
let!
(
:artifact
)
{
create
(
:ci_job_artifact
,
:archive
,
:remote_store
,
job:
job
)
}
before
do
job
.
reload
get
api
(
"/projects/
#{
project
.
id
}
/jobs/
#{
job
.
id
}
/artifacts"
,
api_user
)
end
it
'returns location redirect'
do
expect
(
response
).
to
have_gitlab_http_status
(
:found
)
end
end
end
context
'with regular branch'
do
before
do
pipeline
.
reload
pipeline
.
update!
(
ref:
'master'
,
sha:
project
.
commit
(
'master'
).
sha
)
get_for_ref
(
'master'
)
end
it_behaves_like
'a valid file'
end
context
'with branch name containing slash'
do
before
do
pipeline
.
reload
pipeline
.
update!
(
ref:
'improve/awesome'
,
sha:
project
.
commit
(
'improve/awesome'
).
sha
)
get_for_ref
(
'improve/awesome'
)
end
it_behaves_like
'a valid file'
end
context
'with job name in a child pipeline'
do
let
(
:child_pipeline
)
{
create
(
:ci_pipeline
,
child_of:
pipeline
)
}
let!
(
:child_job
)
{
create
(
:ci_build
,
:artifacts
,
:success
,
name:
'rspec'
,
pipeline:
child_pipeline
)
}
let
(
:job_with_artifacts
)
{
child_job
}
before
do
get_for_ref
(
'master'
,
child_job
.
name
)
end
it_behaves_like
'a valid file'
end
end
end
describe
'GET id/jobs/artifacts/:ref_name/raw/*artifact_path?job=name'
do
context
'when job has artifacts'
do
let
(
:job
)
{
create
(
:ci_build
,
:artifacts
,
pipeline:
pipeline
,
user:
api_user
)
}
let
(
:artifact
)
{
'other_artifacts_0.1.2/another-subdirectory/banana_sample.gif'
}
let
(
:visibility_level
)
{
Gitlab
::
VisibilityLevel
::
PUBLIC
}
let
(
:public_builds
)
{
true
}
before
do
stub_artifacts_object_storage
job
.
success
project
.
update!
(
visibility_level:
visibility_level
,
public_builds:
public_builds
)
get_artifact_file
(
artifact
)
end
context
'when user is anonymous'
do
let
(
:api_user
)
{
nil
}
context
'when project is public'
do
let
(
:visibility_level
)
{
Gitlab
::
VisibilityLevel
::
PUBLIC
}
let
(
:public_builds
)
{
true
}
it
'allows to access artifacts'
,
:sidekiq_might_not_need_inline
do
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
expect
(
response
.
headers
.
to_h
)
.
to
include
(
'Content-Type'
=>
'application/json'
,
'Gitlab-Workhorse-Send-Data'
=>
/artifacts-entry/
)
end
end
context
'when project is public with builds access disabled'
do
let
(
:visibility_level
)
{
Gitlab
::
VisibilityLevel
::
PUBLIC
}
let
(
:public_builds
)
{
false
}
it
'rejects access to artifacts'
do
expect
(
response
).
to
have_gitlab_http_status
(
:forbidden
)
expect
(
json_response
).
to
have_key
(
'message'
)
expect
(
response
.
headers
.
to_h
)
.
not_to
include
(
'Gitlab-Workhorse-Send-Data'
=>
/artifacts-entry/
)
end
end
context
'when project is public with non public artifacts'
do
let
(
:job
)
{
create
(
:ci_build
,
:artifacts
,
:non_public_artifacts
,
pipeline:
pipeline
,
user:
api_user
)
}
let
(
:visibility_level
)
{
Gitlab
::
VisibilityLevel
::
PUBLIC
}
let
(
:public_builds
)
{
true
}
it
'rejects access and hides existence of artifacts'
,
:sidekiq_might_not_need_inline
do
get_artifact_file
(
artifact
)
expect
(
response
).
to
have_gitlab_http_status
(
:forbidden
)
expect
(
json_response
).
to
have_key
(
'message'
)
expect
(
response
.
headers
.
to_h
)
.
not_to
include
(
'Gitlab-Workhorse-Send-Data'
=>
/artifacts-entry/
)
end
context
'with the non_public_artifacts feature flag disabled'
do
before
do
stub_feature_flags
(
non_public_artifacts:
false
)
end
it
'allows access to artifacts'
,
:sidekiq_might_not_need_inline
do
get_artifact_file
(
artifact
)
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
end
end
end
context
'when project is private'
do
let
(
:visibility_level
)
{
Gitlab
::
VisibilityLevel
::
PRIVATE
}
let
(
:public_builds
)
{
true
}
it
'rejects access and hides existence of artifacts'
do
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
expect
(
json_response
).
to
have_key
(
'message'
)
expect
(
response
.
headers
.
to_h
)
.
not_to
include
(
'Gitlab-Workhorse-Send-Data'
=>
/artifacts-entry/
)
end
end
end
context
'when user is authorized'
do
let
(
:visibility_level
)
{
Gitlab
::
VisibilityLevel
::
PRIVATE
}
let
(
:public_builds
)
{
true
}
it
'returns a specific artifact file for a valid path'
,
:sidekiq_might_not_need_inline
do
expect
(
Gitlab
::
Workhorse
)
.
to
receive
(
:send_artifacts_entry
)
.
and_call_original
get_artifact_file
(
artifact
)
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
expect
(
response
.
headers
.
to_h
)
.
to
include
(
'Content-Type'
=>
'application/json'
,
'Gitlab-Workhorse-Send-Data'
=>
/artifacts-entry/
)
expect
(
response
.
parsed_body
).
to
be_empty
end
end
context
'with branch name containing slash'
do
before
do
pipeline
.
reload
pipeline
.
update!
(
ref:
'improve/awesome'
,
sha:
project
.
commit
(
'improve/awesome'
).
sha
)
end
it
'returns a specific artifact file for a valid path'
,
:sidekiq_might_not_need_inline
do
get_artifact_file
(
artifact
,
'improve/awesome'
)
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
expect
(
response
.
headers
.
to_h
)
.
to
include
(
'Content-Type'
=>
'application/json'
,
'Gitlab-Workhorse-Send-Data'
=>
/artifacts-entry/
)
end
end
context
'non-existing job'
do
shared_examples
'not found'
do
it
{
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
}
end
context
'has no such ref'
do
before
do
get_artifact_file
(
'some/artifact'
,
'wrong-ref'
)
end
it_behaves_like
'not found'
end
context
'has no such job'
do
before
do
get_artifact_file
(
'some/artifact'
,
pipeline
.
ref
,
'wrong-job-name'
)
end
it_behaves_like
'not found'
end
end
end
context
'when job does not have artifacts'
do
let
(
:job
)
{
create
(
:ci_build
,
pipeline:
pipeline
,
user:
api_user
)
}
it
'does not return job artifact file'
do
get_artifact_file
(
'some/artifact'
)
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
end
end
def
get_artifact_file
(
artifact_path
,
ref
=
pipeline
.
ref
,
job_name
=
job
.
name
)
get
api
(
"/projects/
#{
project
.
id
}
/jobs/artifacts/
#{
ref
}
/raw/
#{
artifact_path
}
"
,
api_user
),
params:
{
job:
job_name
}
end
end
describe
'POST /projects/:id/jobs/:job_id/artifacts/keep'
do
before
do
post
api
(
"/projects/
#{
project
.
id
}
/jobs/
#{
job
.
id
}
/artifacts/keep"
,
user
)
end
context
'artifacts did not expire'
do
let
(
:job
)
do
create
(
:ci_build
,
:trace_artifact
,
:artifacts
,
:success
,
project:
project
,
pipeline:
pipeline
,
artifacts_expire_at:
Time
.
now
+
7
.
days
)
end
it
'keeps artifacts'
do
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
expect
(
job
.
reload
.
artifacts_expire_at
).
to
be_nil
end
end
context
'no artifacts'
do
let
(
:job
)
{
create
(
:ci_build
,
project:
project
,
pipeline:
pipeline
)
}
it
'responds with not found'
do
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
end
end
end
end
spec/requests/api/ci/jobs_spec.rb
View file @
b2695395
...
...
@@ -445,569 +445,6 @@ RSpec.describe API::Ci::Jobs do
end
end
describe
'DELETE /projects/:id/jobs/:job_id/artifacts'
do
let!
(
:job
)
{
create
(
:ci_build
,
:artifacts
,
pipeline:
pipeline
,
user:
api_user
)
}
before
do
delete
api
(
"/projects/
#{
project
.
id
}
/jobs/
#{
job
.
id
}
/artifacts"
,
api_user
)
end
context
'when user is anonymous'
do
let
(
:api_user
)
{
nil
}
it
'does not delete artifacts'
do
expect
(
job
.
job_artifacts
.
size
).
to
eq
2
end
it
'returns status 401 (unauthorized)'
do
expect
(
response
).
to
have_gitlab_http_status
(
:unauthorized
)
end
end
context
'with developer'
do
it
'does not delete artifacts'
do
expect
(
job
.
job_artifacts
.
size
).
to
eq
2
end
it
'returns status 403 (forbidden)'
do
expect
(
response
).
to
have_gitlab_http_status
(
:forbidden
)
end
end
context
'with authorized user'
do
let
(
:maintainer
)
{
create
(
:project_member
,
:maintainer
,
project:
project
).
user
}
let!
(
:api_user
)
{
maintainer
}
it
'deletes artifacts'
do
expect
(
job
.
job_artifacts
.
size
).
to
eq
0
end
it
'returns status 204 (no content)'
do
expect
(
response
).
to
have_gitlab_http_status
(
:no_content
)
end
end
end
describe
'GET /projects/:id/jobs/:job_id/artifacts/:artifact_path'
do
context
'when job has artifacts'
do
let
(
:job
)
{
create
(
:ci_build
,
:artifacts
,
pipeline:
pipeline
)
}
let
(
:artifact
)
do
'other_artifacts_0.1.2/another-subdirectory/banana_sample.gif'
end
context
'when user is anonymous'
do
let
(
:api_user
)
{
nil
}
context
'when project is public'
do
it
'allows to access artifacts'
do
project
.
update_column
(
:visibility_level
,
Gitlab
::
VisibilityLevel
::
PUBLIC
)
project
.
update_column
(
:public_builds
,
true
)
get_artifact_file
(
artifact
)
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
end
end
context
'when project is public with artifacts that are non public'
do
let
(
:job
)
{
create
(
:ci_build
,
:artifacts
,
:non_public_artifacts
,
pipeline:
pipeline
)
}
it
'rejects access to artifacts'
do
project
.
update_column
(
:visibility_level
,
Gitlab
::
VisibilityLevel
::
PUBLIC
)
project
.
update_column
(
:public_builds
,
true
)
get_artifact_file
(
artifact
)
expect
(
response
).
to
have_gitlab_http_status
(
:forbidden
)
end
context
'with the non_public_artifacts feature flag disabled'
do
before
do
stub_feature_flags
(
non_public_artifacts:
false
)
end
it
'allows access to artifacts'
do
project
.
update_column
(
:visibility_level
,
Gitlab
::
VisibilityLevel
::
PUBLIC
)
project
.
update_column
(
:public_builds
,
true
)
get_artifact_file
(
artifact
)
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
end
end
end
context
'when project is public with builds access disabled'
do
it
'rejects access to artifacts'
do
project
.
update_column
(
:visibility_level
,
Gitlab
::
VisibilityLevel
::
PUBLIC
)
project
.
update_column
(
:public_builds
,
false
)
get_artifact_file
(
artifact
)
expect
(
response
).
to
have_gitlab_http_status
(
:forbidden
)
end
end
context
'when project is private'
do
it
'rejects access and hides existence of artifacts'
do
project
.
update_column
(
:visibility_level
,
Gitlab
::
VisibilityLevel
::
PRIVATE
)
project
.
update_column
(
:public_builds
,
true
)
get_artifact_file
(
artifact
)
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
end
end
end
context
'when user is authorized'
do
it
'returns a specific artifact file for a valid path'
do
expect
(
Gitlab
::
Workhorse
)
.
to
receive
(
:send_artifacts_entry
)
.
and_call_original
get_artifact_file
(
artifact
)
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
expect
(
response
.
headers
.
to_h
)
.
to
include
(
'Content-Type'
=>
'application/json'
,
'Gitlab-Workhorse-Send-Data'
=>
/artifacts-entry/
)
expect
(
response
.
parsed_body
).
to
be_empty
end
context
'when artifacts are locked'
do
it
'allows access to expired artifact'
do
pipeline
.
artifacts_locked!
job
.
update!
(
artifacts_expire_at:
Time
.
now
-
7
.
days
)
get_artifact_file
(
artifact
)
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
end
end
end
end
context
'when job does not have artifacts'
do
it
'does not return job artifact file'
do
get_artifact_file
(
'some/artifact'
)
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
end
end
def
get_artifact_file
(
artifact_path
)
get
api
(
"/projects/
#{
project
.
id
}
/jobs/
#{
job
.
id
}
/"
\
"artifacts/
#{
artifact_path
}
"
,
api_user
)
end
end
describe
'GET /projects/:id/jobs/:job_id/artifacts'
do
shared_examples
'downloads artifact'
do
let
(
:download_headers
)
do
{
'Content-Transfer-Encoding'
=>
'binary'
,
'Content-Disposition'
=>
%q(attachment; filename="ci_build_artifacts.zip"; filename*=UTF-8''ci_build_artifacts.zip)
}
end
it
'returns specific job artifacts'
do
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
expect
(
response
.
headers
.
to_h
).
to
include
(
download_headers
)
expect
(
response
.
body
).
to
match_file
(
job
.
artifacts_file
.
file
.
file
)
end
end
context
'normal authentication'
do
context
'job with artifacts'
do
context
'when artifacts are stored locally'
do
let
(
:job
)
{
create
(
:ci_build
,
:artifacts
,
pipeline:
pipeline
)
}
before
do
get
api
(
"/projects/
#{
project
.
id
}
/jobs/
#{
job
.
id
}
/artifacts"
,
api_user
)
end
context
'authorized user'
do
it_behaves_like
'downloads artifact'
end
context
'unauthorized user'
do
let
(
:api_user
)
{
nil
}
it
'does not return specific job artifacts'
do
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
end
end
end
context
'when artifacts are stored remotely'
do
let
(
:proxy_download
)
{
false
}
let
(
:job
)
{
create
(
:ci_build
,
pipeline:
pipeline
)
}
let
(
:artifact
)
{
create
(
:ci_job_artifact
,
:archive
,
:remote_store
,
job:
job
)
}
before
do
stub_artifacts_object_storage
(
proxy_download:
proxy_download
)
artifact
job
.
reload
get
api
(
"/projects/
#{
project
.
id
}
/jobs/
#{
job
.
id
}
/artifacts"
,
api_user
)
end
context
'when proxy download is enabled'
do
let
(
:proxy_download
)
{
true
}
it
'responds with the workhorse send-url'
do
expect
(
response
.
headers
[
Gitlab
::
Workhorse
::
SEND_DATA_HEADER
]).
to
start_with
(
"send-url:"
)
end
end
context
'when proxy download is disabled'
do
it
'returns location redirect'
do
expect
(
response
).
to
have_gitlab_http_status
(
:found
)
end
end
context
'authorized user'
do
it
'returns the file remote URL'
do
expect
(
response
).
to
redirect_to
(
artifact
.
file
.
url
)
end
end
context
'unauthorized user'
do
let
(
:api_user
)
{
nil
}
it
'does not return specific job artifacts'
do
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
end
end
end
context
'when public project guest and artifacts are non public'
do
let
(
:api_user
)
{
guest
}
let
(
:job
)
{
create
(
:ci_build
,
:artifacts
,
:non_public_artifacts
,
pipeline:
pipeline
)
}
before
do
project
.
update_column
(
:visibility_level
,
Gitlab
::
VisibilityLevel
::
PUBLIC
)
project
.
update_column
(
:public_builds
,
true
)
get
api
(
"/projects/
#{
project
.
id
}
/jobs/
#{
job
.
id
}
/artifacts"
,
api_user
)
end
it
'rejects access and hides existence of artifacts'
do
expect
(
response
).
to
have_gitlab_http_status
(
:forbidden
)
end
context
'with the non_public_artifacts feature flag disabled'
do
before
do
stub_feature_flags
(
non_public_artifacts:
false
)
get
api
(
"/projects/
#{
project
.
id
}
/jobs/
#{
job
.
id
}
/artifacts"
,
api_user
)
end
it
'allows access to artifacts'
do
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
end
end
end
it
'does not return job artifacts if not uploaded'
do
get
api
(
"/projects/
#{
project
.
id
}
/jobs/
#{
job
.
id
}
/artifacts"
,
api_user
)
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
end
end
end
end
describe
'GET /projects/:id/artifacts/:ref_name/download?job=name'
do
let
(
:api_user
)
{
reporter
}
let
(
:job
)
{
create
(
:ci_build
,
:artifacts
,
pipeline:
pipeline
,
user:
api_user
)
}
before
do
stub_artifacts_object_storage
job
.
success
end
def
get_for_ref
(
ref
=
pipeline
.
ref
,
job_name
=
job
.
name
)
get
api
(
"/projects/
#{
project
.
id
}
/jobs/artifacts/
#{
ref
}
/download"
,
api_user
),
params:
{
job:
job_name
}
end
context
'when not logged in'
do
let
(
:api_user
)
{
nil
}
before
do
get_for_ref
end
it
'does not find a resource in a private project'
do
expect
(
project
).
to
be_private
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
end
end
context
'when logging as guest'
do
let
(
:api_user
)
{
guest
}
before
do
get_for_ref
end
it
'gives 403'
do
expect
(
response
).
to
have_gitlab_http_status
(
:forbidden
)
end
end
context
'non-existing job'
do
shared_examples
'not found'
do
it
{
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
}
end
context
'has no such ref'
do
before
do
get_for_ref
(
'TAIL'
)
end
it_behaves_like
'not found'
end
context
'has no such job'
do
before
do
get_for_ref
(
pipeline
.
ref
,
'NOBUILD'
)
end
it_behaves_like
'not found'
end
end
context
'find proper job'
do
let
(
:job_with_artifacts
)
{
job
}
shared_examples
'a valid file'
do
context
'when artifacts are stored locally'
,
:sidekiq_might_not_need_inline
do
let
(
:download_headers
)
do
{
'Content-Transfer-Encoding'
=>
'binary'
,
'Content-Disposition'
=>
%Q(attachment; filename="
#{
job_with_artifacts
.
artifacts_file
.
filename
}
"; filename*=UTF-8''
#{
job
.
artifacts_file
.
filename
}
)
}
end
it
{
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
}
it
{
expect
(
response
.
headers
.
to_h
).
to
include
(
download_headers
)
}
end
context
'when artifacts are stored remotely'
do
let
(
:job
)
{
create
(
:ci_build
,
pipeline:
pipeline
,
user:
api_user
)
}
let!
(
:artifact
)
{
create
(
:ci_job_artifact
,
:archive
,
:remote_store
,
job:
job
)
}
before
do
job
.
reload
get
api
(
"/projects/
#{
project
.
id
}
/jobs/
#{
job
.
id
}
/artifacts"
,
api_user
)
end
it
'returns location redirect'
do
expect
(
response
).
to
have_gitlab_http_status
(
:found
)
end
end
end
context
'with regular branch'
do
before
do
pipeline
.
reload
pipeline
.
update!
(
ref:
'master'
,
sha:
project
.
commit
(
'master'
).
sha
)
get_for_ref
(
'master'
)
end
it_behaves_like
'a valid file'
end
context
'with branch name containing slash'
do
before
do
pipeline
.
reload
pipeline
.
update!
(
ref:
'improve/awesome'
,
sha:
project
.
commit
(
'improve/awesome'
).
sha
)
get_for_ref
(
'improve/awesome'
)
end
it_behaves_like
'a valid file'
end
context
'with job name in a child pipeline'
do
let
(
:child_pipeline
)
{
create
(
:ci_pipeline
,
child_of:
pipeline
)
}
let!
(
:child_job
)
{
create
(
:ci_build
,
:artifacts
,
:success
,
name:
'rspec'
,
pipeline:
child_pipeline
)
}
let
(
:job_with_artifacts
)
{
child_job
}
before
do
get_for_ref
(
'master'
,
child_job
.
name
)
end
it_behaves_like
'a valid file'
end
end
end
describe
'GET id/jobs/artifacts/:ref_name/raw/*artifact_path?job=name'
do
context
'when job has artifacts'
do
let
(
:job
)
{
create
(
:ci_build
,
:artifacts
,
pipeline:
pipeline
,
user:
api_user
)
}
let
(
:artifact
)
{
'other_artifacts_0.1.2/another-subdirectory/banana_sample.gif'
}
let
(
:visibility_level
)
{
Gitlab
::
VisibilityLevel
::
PUBLIC
}
let
(
:public_builds
)
{
true
}
before
do
stub_artifacts_object_storage
job
.
success
project
.
update!
(
visibility_level:
visibility_level
,
public_builds:
public_builds
)
get_artifact_file
(
artifact
)
end
context
'when user is anonymous'
do
let
(
:api_user
)
{
nil
}
context
'when project is public'
do
let
(
:visibility_level
)
{
Gitlab
::
VisibilityLevel
::
PUBLIC
}
let
(
:public_builds
)
{
true
}
it
'allows to access artifacts'
,
:sidekiq_might_not_need_inline
do
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
expect
(
response
.
headers
.
to_h
)
.
to
include
(
'Content-Type'
=>
'application/json'
,
'Gitlab-Workhorse-Send-Data'
=>
/artifacts-entry/
)
end
end
context
'when project is public with builds access disabled'
do
let
(
:visibility_level
)
{
Gitlab
::
VisibilityLevel
::
PUBLIC
}
let
(
:public_builds
)
{
false
}
it
'rejects access to artifacts'
do
expect
(
response
).
to
have_gitlab_http_status
(
:forbidden
)
expect
(
json_response
).
to
have_key
(
'message'
)
expect
(
response
.
headers
.
to_h
)
.
not_to
include
(
'Gitlab-Workhorse-Send-Data'
=>
/artifacts-entry/
)
end
end
context
'when project is public with non public artifacts'
do
let
(
:job
)
{
create
(
:ci_build
,
:artifacts
,
:non_public_artifacts
,
pipeline:
pipeline
,
user:
api_user
)
}
let
(
:visibility_level
)
{
Gitlab
::
VisibilityLevel
::
PUBLIC
}
let
(
:public_builds
)
{
true
}
it
'rejects access and hides existence of artifacts'
,
:sidekiq_might_not_need_inline
do
get_artifact_file
(
artifact
)
expect
(
response
).
to
have_gitlab_http_status
(
:forbidden
)
expect
(
json_response
).
to
have_key
(
'message'
)
expect
(
response
.
headers
.
to_h
)
.
not_to
include
(
'Gitlab-Workhorse-Send-Data'
=>
/artifacts-entry/
)
end
context
'with the non_public_artifacts feature flag disabled'
do
before
do
stub_feature_flags
(
non_public_artifacts:
false
)
end
it
'allows access to artifacts'
,
:sidekiq_might_not_need_inline
do
get_artifact_file
(
artifact
)
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
end
end
end
context
'when project is private'
do
let
(
:visibility_level
)
{
Gitlab
::
VisibilityLevel
::
PRIVATE
}
let
(
:public_builds
)
{
true
}
it
'rejects access and hides existence of artifacts'
do
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
expect
(
json_response
).
to
have_key
(
'message'
)
expect
(
response
.
headers
.
to_h
)
.
not_to
include
(
'Gitlab-Workhorse-Send-Data'
=>
/artifacts-entry/
)
end
end
end
context
'when user is authorized'
do
let
(
:visibility_level
)
{
Gitlab
::
VisibilityLevel
::
PRIVATE
}
let
(
:public_builds
)
{
true
}
it
'returns a specific artifact file for a valid path'
,
:sidekiq_might_not_need_inline
do
expect
(
Gitlab
::
Workhorse
)
.
to
receive
(
:send_artifacts_entry
)
.
and_call_original
get_artifact_file
(
artifact
)
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
expect
(
response
.
headers
.
to_h
)
.
to
include
(
'Content-Type'
=>
'application/json'
,
'Gitlab-Workhorse-Send-Data'
=>
/artifacts-entry/
)
expect
(
response
.
parsed_body
).
to
be_empty
end
end
context
'with branch name containing slash'
do
before
do
pipeline
.
reload
pipeline
.
update!
(
ref:
'improve/awesome'
,
sha:
project
.
commit
(
'improve/awesome'
).
sha
)
end
it
'returns a specific artifact file for a valid path'
,
:sidekiq_might_not_need_inline
do
get_artifact_file
(
artifact
,
'improve/awesome'
)
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
expect
(
response
.
headers
.
to_h
)
.
to
include
(
'Content-Type'
=>
'application/json'
,
'Gitlab-Workhorse-Send-Data'
=>
/artifacts-entry/
)
end
end
context
'non-existing job'
do
shared_examples
'not found'
do
it
{
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
}
end
context
'has no such ref'
do
before
do
get_artifact_file
(
'some/artifact'
,
'wrong-ref'
)
end
it_behaves_like
'not found'
end
context
'has no such job'
do
before
do
get_artifact_file
(
'some/artifact'
,
pipeline
.
ref
,
'wrong-job-name'
)
end
it_behaves_like
'not found'
end
end
end
context
'when job does not have artifacts'
do
let
(
:job
)
{
create
(
:ci_build
,
pipeline:
pipeline
,
user:
api_user
)
}
it
'does not return job artifact file'
do
get_artifact_file
(
'some/artifact'
)
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
end
end
def
get_artifact_file
(
artifact_path
,
ref
=
pipeline
.
ref
,
job_name
=
job
.
name
)
get
api
(
"/projects/
#{
project
.
id
}
/jobs/artifacts/
#{
ref
}
/raw/
#{
artifact_path
}
"
,
api_user
),
params:
{
job:
job_name
}
end
end
describe
'GET /projects/:id/jobs/:job_id/trace'
do
before
do
get
api
(
"/projects/
#{
project
.
id
}
/jobs/
#{
job
.
id
}
/trace"
,
api_user
)
...
...
@@ -1249,32 +686,6 @@ RSpec.describe API::Ci::Jobs do
end
end
describe
'POST /projects/:id/jobs/:job_id/artifacts/keep'
do
before
do
post
api
(
"/projects/
#{
project
.
id
}
/jobs/
#{
job
.
id
}
/artifacts/keep"
,
user
)
end
context
'artifacts did not expire'
do
let
(
:job
)
do
create
(
:ci_build
,
:trace_artifact
,
:artifacts
,
:success
,
project:
project
,
pipeline:
pipeline
,
artifacts_expire_at:
Time
.
now
+
7
.
days
)
end
it
'keeps artifacts'
do
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
expect
(
job
.
reload
.
artifacts_expire_at
).
to
be_nil
end
end
context
'no artifacts'
do
let
(
:job
)
{
create
(
:ci_build
,
project:
project
,
pipeline:
pipeline
)
}
it
'responds with not found'
do
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
end
end
end
describe
'POST /projects/:id/jobs/:job_id/play'
do
before
do
post
api
(
"/projects/
#{
project
.
id
}
/jobs/
#{
job
.
id
}
/play"
,
api_user
)
...
...
spec/services/ci/job_artifacts/destroy_batch_service_spec.rb
View file @
b2695395
...
...
@@ -3,59 +3,74 @@
require
'spec_helper'
RSpec
.
describe
Ci
::
JobArtifacts
::
DestroyBatchService
do
let
(
:artifacts
)
{
Ci
::
JobArtifact
.
all
}
let
(
:artifacts
)
{
Ci
::
JobArtifact
.
where
(
id:
[
artifact_with_file
.
id
,
artifact_without_file
.
id
])
}
let
(
:service
)
{
described_class
.
new
(
artifacts
,
pick_up_at:
Time
.
current
)
}
let_it_be
(
:artifact_with_file
,
refind:
true
)
do
create
(
:ci_job_artifact
,
:zip
)
end
let_it_be
(
:artifact_without_file
,
refind:
true
)
do
create
(
:ci_job_artifact
)
end
let_it_be
(
:undeleted_artifact
,
refind:
true
)
do
create
(
:ci_job_artifact
)
end
describe
'.execute'
do
subject
(
:execute
)
{
service
.
execute
}
let_it_be
(
:artifact
,
refind:
true
)
do
create
(
:ci_job_artifact
)
it
'creates a deleted object for artifact with attached file'
do
expect
{
subject
}.
to
change
{
Ci
::
DeletedObject
.
count
}.
by
(
1
)
end
context
'when the artifact has a file attached to it'
do
before
do
artifact
.
file
=
fixture_file_upload
(
Rails
.
root
.
join
(
'spec/fixtures/ci_build_artifacts.zip'
),
'application/zip'
)
artifact
.
save!
end
it
'does not remove the attached file'
do
expect
{
execute
}.
not_to
change
{
artifact_with_file
.
file
.
exists?
}
end
it
'creates a deleted object
'
do
expect
{
subject
}.
to
change
{
Ci
::
DeletedObject
.
count
}.
by
(
1
)
end
it
'deletes the artifact records
'
do
expect
{
subject
}.
to
change
{
Ci
::
JobArtifact
.
count
}.
by
(
-
2
)
end
it
'does not remove the files'
do
expect
{
execute
}.
not_to
change
{
artifact
.
file
.
exists?
}
it
'reports metrics for destroyed artifacts'
do
expect_next_instance_of
(
Gitlab
::
Ci
::
Artifacts
::
Metrics
)
do
|
metrics
|
expect
(
metrics
).
to
receive
(
:increment_destroyed_artifacts_count
).
with
(
2
).
and_call_original
expect
(
metrics
).
to
receive
(
:increment_destroyed_artifacts_bytes
).
with
(
107464
).
and_call_original
end
it
'reports metrics for destroyed artifacts'
do
expect_next_instance_of
(
Gitlab
::
Ci
::
Artifacts
::
Metrics
)
do
|
metrics
|
expect
(
metrics
).
to
receive
(
:increment_destroyed_artifacts_count
).
with
(
1
).
and_call_original
expect
(
metrics
).
to
receive
(
:increment_destroyed_artifacts_bytes
).
with
(
107464
).
and_call_original
end
execute
end
context
'ProjectStatistics'
do
it
'resets project statistics'
do
expect
(
ProjectStatistics
).
to
receive
(
:increment_statistic
).
once
.
with
(
artifact_with_file
.
project
,
:build_artifacts_size
,
-
artifact_with_file
.
file
.
size
)
.
and_call_original
expect
(
ProjectStatistics
).
to
receive
(
:increment_statistic
).
once
.
with
(
artifact_without_file
.
project
,
:build_artifacts_size
,
0
)
.
and_call_original
execute
end
context
'ProjectStatistics'
do
it
'resets project statistics'
do
expect
(
ProjectStatistics
).
to
receive
(
:increment_statistic
).
once
.
with
(
artifact
.
project
,
:build_artifacts_size
,
-
artifact
.
file
.
size
)
.
and_call_original
context
'with update_stats: false'
do
it
'does not update project statistics'
do
expect
(
ProjectStatistics
).
not_to
receive
(
:increment_statistic
)
execute
service
.
execute
(
update_stats:
false
)
end
context
'with update_stats: false'
do
it
'does not update project statistics'
do
expect
(
ProjectStatistics
).
not_to
receive
(
:increment_statistic
)
service
.
execute
(
update_stats:
false
)
end
it
'returns size statistics'
do
expected_updates
=
{
statistics_updates:
{
artifact_with_file
.
project
=>
-
artifact_with_file
.
file
.
size
,
artifact_without_file
.
project
=>
0
}
}
it
'returns size statistics'
do
expect
(
service
.
execute
(
update_stats:
false
)).
to
match
(
a_hash_including
(
statistics_updates:
{
artifact
.
project
=>
-
artifact
.
file
.
size
}))
end
expect
(
service
.
execute
(
update_stats:
false
)).
to
match
(
a_hash_including
(
expected_updates
))
end
end
end
...
...
@@ -71,7 +86,7 @@ RSpec.describe Ci::JobArtifacts::DestroyBatchService do
it
'raises an exception and stop destroying'
do
expect
{
execute
}.
to
raise_error
(
ActiveRecord
::
RecordNotDestroyed
)
.
and
not_change
{
Ci
::
JobArtifact
.
count
}
.
from
(
1
)
.
and
not_change
{
Ci
::
JobArtifact
.
count
}
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