2019-04-15 06:17:05 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2016-12-20 09:53:53 -05:00
|
|
|
require 'spec_helper'
|
|
|
|
|
|
|
|
describe Projects::PipelinesController do
|
2017-05-06 12:45:46 -04:00
|
|
|
include ApiHelpers
|
|
|
|
|
2019-10-01 20:06:26 -04:00
|
|
|
let_it_be(:user) { create(:user) }
|
2019-12-24 10:07:44 -05:00
|
|
|
let_it_be(:project) { create(:project, :public, :repository) }
|
2019-01-28 07:12:30 -05:00
|
|
|
let(:feature) { ProjectFeature::ENABLED }
|
2016-12-20 09:53:53 -05:00
|
|
|
|
|
|
|
before do
|
2017-07-18 10:32:34 -04:00
|
|
|
stub_not_protect_default_branch
|
|
|
|
project.add_developer(user)
|
2017-10-12 04:39:11 -04:00
|
|
|
project.project_feature.update(builds_access_level: feature)
|
2017-05-04 15:01:15 -04:00
|
|
|
|
2016-12-20 09:53:53 -05:00
|
|
|
sign_in(user)
|
|
|
|
end
|
|
|
|
|
2016-12-21 09:13:24 -05:00
|
|
|
describe 'GET index.json' do
|
|
|
|
before do
|
2019-12-24 10:07:44 -05:00
|
|
|
create_all_pipeline_types
|
2016-12-21 09:13:24 -05:00
|
|
|
end
|
|
|
|
|
2018-05-29 11:19:39 -04:00
|
|
|
context 'when using persisted stages', :request_store do
|
2019-12-24 10:07:44 -05:00
|
|
|
render_views
|
|
|
|
|
2018-05-29 11:19:39 -04:00
|
|
|
before do
|
|
|
|
stub_feature_flags(ci_pipeline_persisted_stages: true)
|
2018-05-22 06:30:45 -04:00
|
|
|
end
|
2017-10-12 04:39:11 -04:00
|
|
|
|
2020-02-10 07:08:59 -05:00
|
|
|
it 'returns serialized pipelines' do
|
2019-03-29 12:23:51 -04:00
|
|
|
expect(::Gitlab::GitalyClient).to receive(:allow_ref_name_caching).and_call_original
|
|
|
|
|
2019-12-24 10:07:44 -05:00
|
|
|
get_pipelines_index_json
|
2016-12-21 09:13:24 -05:00
|
|
|
|
2018-05-29 11:19:39 -04:00
|
|
|
expect(response).to have_gitlab_http_status(:ok)
|
|
|
|
expect(response).to match_response_schema('pipeline')
|
|
|
|
|
|
|
|
expect(json_response).to include('pipelines')
|
2020-03-09 08:07:45 -04:00
|
|
|
expect(json_response['pipelines'].count).to eq 6
|
|
|
|
expect(json_response['count']['all']).to eq '6'
|
|
|
|
expect(json_response['count']['running']).to eq '2'
|
2018-05-29 11:19:39 -04:00
|
|
|
expect(json_response['count']['pending']).to eq '1'
|
|
|
|
expect(json_response['count']['finished']).to eq '3'
|
|
|
|
|
|
|
|
json_response.dig('pipelines', 0, 'details', 'stages').tap do |stages|
|
|
|
|
expect(stages.count).to eq 3
|
|
|
|
end
|
2019-12-24 10:07:44 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not execute N+1 queries' do
|
|
|
|
get_pipelines_index_json
|
|
|
|
|
|
|
|
control_count = ActiveRecord::QueryRecorder.new do
|
|
|
|
get_pipelines_index_json
|
|
|
|
end.count
|
2018-05-29 11:19:39 -04:00
|
|
|
|
2019-12-24 10:07:44 -05:00
|
|
|
create_all_pipeline_types
|
|
|
|
|
|
|
|
# There appears to be one extra query for Pipelines#has_warnings? for some reason
|
|
|
|
expect { get_pipelines_index_json }.not_to exceed_query_limit(control_count + 1)
|
|
|
|
expect(response).to have_gitlab_http_status(:ok)
|
2020-03-09 08:07:45 -04:00
|
|
|
expect(json_response['pipelines'].count).to eq 12
|
2018-05-29 11:19:39 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-01-16 07:09:29 -05:00
|
|
|
context 'when using legacy stages', :request_store do
|
2018-05-29 11:19:39 -04:00
|
|
|
before do
|
|
|
|
stub_feature_flags(ci_pipeline_persisted_stages: false)
|
|
|
|
end
|
|
|
|
|
2018-07-02 11:09:49 -04:00
|
|
|
it 'returns JSON with serialized pipelines' do
|
|
|
|
get_pipelines_index_json
|
2018-05-29 11:19:39 -04:00
|
|
|
|
|
|
|
expect(response).to have_gitlab_http_status(:ok)
|
|
|
|
expect(response).to match_response_schema('pipeline')
|
|
|
|
|
|
|
|
expect(json_response).to include('pipelines')
|
2020-03-09 08:07:45 -04:00
|
|
|
expect(json_response['pipelines'].count).to eq 6
|
|
|
|
expect(json_response['count']['all']).to eq '6'
|
|
|
|
expect(json_response['count']['running']).to eq '2'
|
2018-05-29 11:19:39 -04:00
|
|
|
expect(json_response['count']['pending']).to eq '1'
|
|
|
|
expect(json_response['count']['finished']).to eq '3'
|
|
|
|
|
|
|
|
json_response.dig('pipelines', 0, 'details', 'stages').tap do |stages|
|
|
|
|
expect(stages.count).to eq 3
|
|
|
|
end
|
2018-07-02 11:09:49 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not execute N+1 queries' do
|
2020-02-10 07:08:59 -05:00
|
|
|
get_pipelines_index_json
|
|
|
|
|
|
|
|
control_count = ActiveRecord::QueryRecorder.new do
|
2018-07-02 11:09:49 -04:00
|
|
|
get_pipelines_index_json
|
2020-02-10 07:08:59 -05:00
|
|
|
end.count
|
2018-05-29 11:19:39 -04:00
|
|
|
|
2020-02-10 07:08:59 -05:00
|
|
|
create_all_pipeline_types
|
|
|
|
|
|
|
|
# There appears to be one extra query for Pipelines#has_warnings? for some reason
|
|
|
|
expect { get_pipelines_index_json }.not_to exceed_query_limit(control_count + 1)
|
2020-03-09 08:07:45 -04:00
|
|
|
|
2020-02-10 07:08:59 -05:00
|
|
|
expect(response).to have_gitlab_http_status(:ok)
|
2020-03-09 08:07:45 -04:00
|
|
|
expect(json_response['pipelines'].count).to eq 12
|
2018-05-29 11:19:39 -04:00
|
|
|
end
|
2016-12-21 09:13:24 -05:00
|
|
|
end
|
2017-10-12 04:39:11 -04:00
|
|
|
|
2018-05-11 10:17:03 -04:00
|
|
|
it 'does not include coverage data for the pipelines' do
|
2018-05-21 05:39:46 -04:00
|
|
|
get_pipelines_index_json
|
2018-05-11 10:17:03 -04:00
|
|
|
|
|
|
|
expect(json_response['pipelines'][0]).not_to include('coverage')
|
|
|
|
end
|
|
|
|
|
2017-10-12 04:39:11 -04:00
|
|
|
context 'when performing gitaly calls', :request_store do
|
2019-12-05 07:07:43 -05:00
|
|
|
it 'limits the Gitaly requests' do
|
2018-09-25 13:12:51 -04:00
|
|
|
# Isolate from test preparation (Repository#exists? is also cached in RequestStore)
|
|
|
|
RequestStore.end!
|
|
|
|
RequestStore.clear!
|
|
|
|
RequestStore.begin!
|
|
|
|
|
2019-04-04 16:22:11 -04:00
|
|
|
expect(::Gitlab::GitalyClient).to receive(:allow_ref_name_caching).and_call_original
|
|
|
|
|
2019-12-05 07:07:43 -05:00
|
|
|
# ListCommitsByOid, RepositoryExists, HasLocalBranches
|
2018-05-21 05:39:46 -04:00
|
|
|
expect { get_pipelines_index_json }
|
2019-12-05 07:07:43 -05:00
|
|
|
.to change { Gitlab::GitalyClient.get_request_count }.by(3)
|
2017-10-12 04:39:11 -04:00
|
|
|
end
|
|
|
|
end
|
2018-05-21 05:39:46 -04:00
|
|
|
|
2018-06-26 12:31:05 -04:00
|
|
|
context 'when the project is private' do
|
|
|
|
let(:project) { create(:project, :private, :repository) }
|
|
|
|
|
|
|
|
it 'returns `not_found` when the user does not have access' do
|
|
|
|
sign_in(create(:user))
|
|
|
|
|
|
|
|
get_pipelines_index_json
|
|
|
|
|
|
|
|
expect(response).to have_gitlab_http_status(:not_found)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns the pipelines when the user has access' do
|
|
|
|
get_pipelines_index_json
|
|
|
|
|
2020-03-09 08:07:45 -04:00
|
|
|
expect(json_response['pipelines'].size).to eq(6)
|
2018-06-26 12:31:05 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-05-21 05:39:46 -04:00
|
|
|
def get_pipelines_index_json
|
2018-12-17 17:52:17 -05:00
|
|
|
get :index, params: {
|
|
|
|
namespace_id: project.namespace,
|
|
|
|
project_id: project
|
|
|
|
},
|
2018-05-21 05:39:46 -04:00
|
|
|
format: :json
|
|
|
|
end
|
|
|
|
|
2019-12-24 10:07:44 -05:00
|
|
|
def create_all_pipeline_types
|
|
|
|
%w(pending running success failed canceled).each_with_index do |status, index|
|
|
|
|
create_pipeline(status, project.commit("HEAD~#{index}"))
|
|
|
|
end
|
2020-03-09 08:07:45 -04:00
|
|
|
|
|
|
|
create_pipeline_with_merge_request
|
2019-12-24 10:07:44 -05:00
|
|
|
end
|
|
|
|
|
2020-03-09 08:07:45 -04:00
|
|
|
def create_pipeline_with_merge_request
|
|
|
|
# New merge requests must be created with different branches, so
|
|
|
|
# let's just create new ones with random names.
|
|
|
|
branch_name = "test-#{SecureRandom.hex}"
|
|
|
|
project.repository.create_branch(branch_name, project.repository.root_ref)
|
|
|
|
mr = create(:merge_request, source_project: project, target_project: project, source_branch: branch_name)
|
|
|
|
create_pipeline(:running, project.commit('HEAD'), merge_request: mr)
|
|
|
|
end
|
|
|
|
|
|
|
|
def create_pipeline(status, sha, merge_request: nil)
|
2019-12-24 10:07:44 -05:00
|
|
|
user = create(:user)
|
2018-05-22 06:30:45 -04:00
|
|
|
pipeline = create(:ci_empty_pipeline, status: status,
|
|
|
|
project: project,
|
2019-12-24 10:07:44 -05:00
|
|
|
sha: sha,
|
2020-03-09 08:07:45 -04:00
|
|
|
user: user,
|
|
|
|
merge_request: merge_request)
|
2018-05-22 06:30:45 -04:00
|
|
|
|
2019-12-24 10:07:44 -05:00
|
|
|
create_build(pipeline, 'build', 1, 'build', user)
|
|
|
|
create_build(pipeline, 'test', 2, 'test', user)
|
|
|
|
create_build(pipeline, 'deploy', 3, 'deploy', user)
|
2020-03-09 08:07:45 -04:00
|
|
|
|
|
|
|
pipeline
|
2018-05-22 06:30:45 -04:00
|
|
|
end
|
|
|
|
|
2019-12-24 10:07:44 -05:00
|
|
|
def create_build(pipeline, stage, stage_idx, name, user = nil)
|
2018-05-22 06:30:45 -04:00
|
|
|
status = %w[created running pending success failed canceled].sample
|
2020-02-18 01:09:06 -05:00
|
|
|
create(
|
|
|
|
:ci_build,
|
|
|
|
:artifacts,
|
|
|
|
artifacts_expire_at: 2.days.from_now,
|
|
|
|
pipeline: pipeline,
|
|
|
|
stage: stage,
|
|
|
|
stage_idx: stage_idx,
|
|
|
|
name: name,
|
|
|
|
status: status,
|
|
|
|
user: user
|
|
|
|
)
|
2018-05-21 05:39:46 -04:00
|
|
|
end
|
2016-12-21 09:13:24 -05:00
|
|
|
end
|
|
|
|
|
2018-05-21 05:39:46 -04:00
|
|
|
describe 'GET show.json' do
|
2019-11-13 07:06:22 -05:00
|
|
|
let(:pipeline) { create(:ci_pipeline, project: project) }
|
2017-05-06 12:45:46 -04:00
|
|
|
|
|
|
|
it 'returns the pipeline' do
|
|
|
|
get_pipeline_json
|
|
|
|
|
2017-10-19 14:28:19 -04:00
|
|
|
expect(response).to have_gitlab_http_status(:ok)
|
2017-05-06 12:45:46 -04:00
|
|
|
expect(json_response).not_to be_an(Array)
|
|
|
|
expect(json_response['id']).to be(pipeline.id)
|
|
|
|
expect(json_response['details']).to have_key 'stages'
|
|
|
|
end
|
|
|
|
|
2017-06-09 13:12:51 -04:00
|
|
|
context 'when the pipeline has multiple stages and groups', :request_store do
|
2018-05-21 05:39:46 -04:00
|
|
|
let(:project) { create(:project, :repository) }
|
|
|
|
|
|
|
|
let(:pipeline) do
|
|
|
|
create(:ci_empty_pipeline, project: project,
|
|
|
|
user: user,
|
|
|
|
sha: project.commit.id)
|
|
|
|
end
|
|
|
|
|
2017-05-10 12:39:17 -04:00
|
|
|
before do
|
|
|
|
create_build('build', 0, 'build')
|
|
|
|
create_build('test', 1, 'rspec 0')
|
|
|
|
create_build('deploy', 2, 'production')
|
|
|
|
create_build('post deploy', 3, 'pages 0')
|
|
|
|
end
|
|
|
|
|
2017-05-06 12:45:46 -04:00
|
|
|
it 'does not perform N + 1 queries' do
|
2019-08-19 11:28:20 -04:00
|
|
|
# Set up all required variables
|
|
|
|
get_pipeline_json
|
|
|
|
|
2017-05-06 12:45:46 -04:00
|
|
|
control_count = ActiveRecord::QueryRecorder.new { get_pipeline_json }.count
|
|
|
|
|
2019-08-19 11:28:20 -04:00
|
|
|
first_build = pipeline.builds.first
|
|
|
|
first_build.tag_list << [:hello, :world]
|
|
|
|
create(:deployment, deployable: first_build)
|
|
|
|
|
|
|
|
second_build = pipeline.builds.second
|
|
|
|
second_build.tag_list << [:docker, :ruby]
|
|
|
|
create(:deployment, deployable: second_build)
|
2017-05-06 12:45:46 -04:00
|
|
|
|
2017-05-10 12:39:17 -04:00
|
|
|
new_count = ActiveRecord::QueryRecorder.new { get_pipeline_json }.count
|
2018-05-21 05:39:46 -04:00
|
|
|
|
2019-08-19 11:28:20 -04:00
|
|
|
expect(new_count).to be_within(1).of(control_count)
|
2017-05-06 12:45:46 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-01-28 07:12:30 -05:00
|
|
|
context 'when builds are disabled' do
|
|
|
|
let(:feature) { ProjectFeature::DISABLED }
|
|
|
|
|
|
|
|
it 'users can not see internal pipelines' do
|
|
|
|
get_pipeline_json
|
|
|
|
|
|
|
|
expect(response).to have_gitlab_http_status(:not_found)
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when pipeline is external' do
|
|
|
|
let(:pipeline) { create(:ci_pipeline, source: :external, project: project) }
|
|
|
|
|
|
|
|
it 'users can see the external pipeline' do
|
|
|
|
get_pipeline_json
|
|
|
|
|
|
|
|
expect(response).to have_gitlab_http_status(:ok)
|
|
|
|
expect(json_response['id']).to be(pipeline.id)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-10-16 05:07:51 -04:00
|
|
|
context 'with triggered pipelines' do
|
|
|
|
let_it_be(:project) { create(:project, :repository) }
|
|
|
|
let_it_be(:source_project) { create(:project, :repository) }
|
|
|
|
let_it_be(:target_project) { create(:project, :repository) }
|
|
|
|
let_it_be(:root_pipeline) { create_pipeline(project) }
|
|
|
|
let_it_be(:source_pipeline) { create_pipeline(source_project) }
|
|
|
|
let_it_be(:source_of_source_pipeline) { create_pipeline(source_project) }
|
|
|
|
let_it_be(:target_pipeline) { create_pipeline(target_project) }
|
|
|
|
let_it_be(:target_of_target_pipeline) { create_pipeline(target_project) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
create_link(source_of_source_pipeline, source_pipeline)
|
|
|
|
create_link(source_pipeline, root_pipeline)
|
|
|
|
create_link(root_pipeline, target_pipeline)
|
|
|
|
create_link(target_pipeline, target_of_target_pipeline)
|
|
|
|
end
|
|
|
|
|
|
|
|
shared_examples 'not expanded' do
|
|
|
|
let(:expected_stages) { be_nil }
|
|
|
|
|
|
|
|
it 'does return base details' do
|
|
|
|
get_pipeline_json(root_pipeline)
|
|
|
|
|
|
|
|
expect(json_response['triggered_by']).to include('id' => source_pipeline.id)
|
|
|
|
expect(json_response['triggered']).to contain_exactly(
|
|
|
|
include('id' => target_pipeline.id))
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not expand triggered_by pipeline' do
|
|
|
|
get_pipeline_json(root_pipeline)
|
|
|
|
|
|
|
|
triggered_by = json_response['triggered_by']
|
|
|
|
expect(triggered_by['triggered_by']).to be_nil
|
|
|
|
expect(triggered_by['triggered']).to be_nil
|
|
|
|
expect(triggered_by['details']['stages']).to expected_stages
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not expand triggered pipelines' do
|
|
|
|
get_pipeline_json(root_pipeline)
|
|
|
|
|
|
|
|
first_triggered = json_response['triggered'].first
|
|
|
|
expect(first_triggered['triggered_by']).to be_nil
|
|
|
|
expect(first_triggered['triggered']).to be_nil
|
|
|
|
expect(first_triggered['details']['stages']).to expected_stages
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
shared_examples 'expanded' do
|
|
|
|
it 'does return base details' do
|
|
|
|
get_pipeline_json(root_pipeline)
|
|
|
|
|
|
|
|
expect(json_response['triggered_by']).to include('id' => source_pipeline.id)
|
|
|
|
expect(json_response['triggered']).to contain_exactly(
|
|
|
|
include('id' => target_pipeline.id))
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'does expand triggered_by pipeline' do
|
|
|
|
get_pipeline_json(root_pipeline)
|
|
|
|
|
|
|
|
triggered_by = json_response['triggered_by']
|
|
|
|
expect(triggered_by['triggered_by']).to include(
|
|
|
|
'id' => source_of_source_pipeline.id)
|
|
|
|
expect(triggered_by['details']['stages']).not_to be_nil
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not recursively expand triggered_by' do
|
|
|
|
get_pipeline_json(root_pipeline)
|
|
|
|
|
|
|
|
triggered_by = json_response['triggered_by']
|
|
|
|
expect(triggered_by['triggered']).to be_nil
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'does expand triggered pipelines' do
|
|
|
|
get_pipeline_json(root_pipeline)
|
|
|
|
|
|
|
|
first_triggered = json_response['triggered'].first
|
|
|
|
expect(first_triggered['triggered']).to contain_exactly(
|
|
|
|
include('id' => target_of_target_pipeline.id))
|
|
|
|
expect(first_triggered['details']['stages']).not_to be_nil
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not recursively expand triggered' do
|
|
|
|
get_pipeline_json(root_pipeline)
|
|
|
|
|
|
|
|
first_triggered = json_response['triggered'].first
|
|
|
|
expect(first_triggered['triggered_by']).to be_nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when it does have permission to read other projects' do
|
|
|
|
before do
|
|
|
|
source_project.add_developer(user)
|
|
|
|
target_project.add_developer(user)
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when not-expanding any pipelines' do
|
|
|
|
let(:expanded) { nil }
|
|
|
|
|
|
|
|
it_behaves_like 'not expanded'
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when expanding non-existing pipeline' do
|
|
|
|
let(:expanded) { [-1] }
|
|
|
|
|
|
|
|
it_behaves_like 'not expanded'
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when expanding pipeline that is not directly expandable' do
|
|
|
|
let(:expanded) { [source_of_source_pipeline.id, target_of_target_pipeline.id] }
|
|
|
|
|
|
|
|
it_behaves_like 'not expanded'
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when expanding self' do
|
|
|
|
let(:expanded) { [root_pipeline.id] }
|
|
|
|
|
|
|
|
context 'it does not recursively expand pipelines' do
|
|
|
|
it_behaves_like 'not expanded'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when expanding source and target pipeline' do
|
|
|
|
let(:expanded) { [source_pipeline.id, target_pipeline.id] }
|
|
|
|
|
|
|
|
it_behaves_like 'expanded'
|
|
|
|
|
|
|
|
context 'when expand depth is limited to 1' do
|
|
|
|
before do
|
|
|
|
stub_const('TriggeredPipelineEntity::MAX_EXPAND_DEPTH', 1)
|
|
|
|
end
|
|
|
|
|
|
|
|
it_behaves_like 'not expanded' do
|
|
|
|
# We expect that triggered/triggered_by is not expanded,
|
|
|
|
# but we still return details.stages for that pipeline
|
|
|
|
let(:expected_stages) { be_a(Array) }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when expanding all' do
|
|
|
|
let(:expanded) do
|
|
|
|
[
|
|
|
|
source_of_source_pipeline.id,
|
|
|
|
source_pipeline.id,
|
|
|
|
root_pipeline.id,
|
|
|
|
target_pipeline.id,
|
|
|
|
target_of_target_pipeline.id
|
|
|
|
]
|
|
|
|
end
|
|
|
|
|
|
|
|
it_behaves_like 'expanded'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when does not have permission to read other projects' do
|
|
|
|
let(:expanded) { [source_pipeline.id, target_pipeline.id] }
|
|
|
|
|
|
|
|
it_behaves_like 'not expanded'
|
|
|
|
end
|
|
|
|
|
|
|
|
def create_pipeline(project)
|
|
|
|
create(:ci_empty_pipeline, project: project).tap do |pipeline|
|
|
|
|
create(:ci_build, pipeline: pipeline, stage: 'test', name: 'rspec')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def create_link(source_pipeline, pipeline)
|
|
|
|
source_pipeline.sourced_pipelines.create!(
|
|
|
|
source_job: source_pipeline.builds.all.sample,
|
|
|
|
source_project: source_pipeline.project,
|
|
|
|
project: pipeline.project,
|
|
|
|
pipeline: pipeline
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
def get_pipeline_json(pipeline)
|
|
|
|
params = {
|
|
|
|
namespace_id: pipeline.project.namespace,
|
|
|
|
project_id: pipeline.project,
|
|
|
|
id: pipeline,
|
|
|
|
expanded: expanded
|
|
|
|
}
|
|
|
|
|
|
|
|
get :show, params: params.compact, format: :json
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-05-06 12:45:46 -04:00
|
|
|
def get_pipeline_json
|
2018-12-17 17:52:17 -05:00
|
|
|
get :show, params: { namespace_id: project.namespace, project_id: project, id: pipeline }, format: :json
|
2017-05-06 12:45:46 -04:00
|
|
|
end
|
2017-05-10 12:39:17 -04:00
|
|
|
|
|
|
|
def create_build(stage, stage_idx, name)
|
|
|
|
create(:ci_build, pipeline: pipeline, stage: stage, stage_idx: stage_idx, name: name)
|
|
|
|
end
|
2017-05-06 12:45:46 -04:00
|
|
|
end
|
|
|
|
|
2016-12-20 09:53:53 -05:00
|
|
|
describe 'GET stages.json' do
|
2016-12-21 09:13:24 -05:00
|
|
|
let(:pipeline) { create(:ci_pipeline, project: project) }
|
|
|
|
|
2016-12-20 09:53:53 -05:00
|
|
|
context 'when accessing existing stage' do
|
|
|
|
before do
|
2018-09-06 06:21:58 -04:00
|
|
|
create(:ci_build, :retried, :failed, pipeline: pipeline, stage: 'build')
|
2016-12-20 09:53:53 -05:00
|
|
|
create(:ci_build, pipeline: pipeline, stage: 'build')
|
2018-09-06 06:21:58 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
context 'without retried' do
|
|
|
|
before do
|
|
|
|
get_stage('build')
|
|
|
|
end
|
2016-12-20 09:53:53 -05:00
|
|
|
|
2018-09-06 06:21:58 -04:00
|
|
|
it 'returns pipeline jobs without the retried builds' do
|
|
|
|
expect(response).to have_gitlab_http_status(:ok)
|
|
|
|
expect(response).to match_response_schema('pipeline_stage')
|
|
|
|
expect(json_response['latest_statuses'].length).to eq 1
|
|
|
|
expect(json_response).not_to have_key('retried')
|
|
|
|
end
|
2016-12-20 09:53:53 -05:00
|
|
|
end
|
|
|
|
|
2018-09-06 06:21:58 -04:00
|
|
|
context 'with retried' do
|
|
|
|
before do
|
|
|
|
get_stage('build', retried: true)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns pipelines jobs with the retried builds' do
|
|
|
|
expect(response).to have_gitlab_http_status(:ok)
|
|
|
|
expect(response).to match_response_schema('pipeline_stage')
|
|
|
|
expect(json_response['latest_statuses'].length).to eq 1
|
|
|
|
expect(json_response['retried'].length).to eq 1
|
|
|
|
end
|
2016-12-20 09:53:53 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when accessing unknown stage' do
|
|
|
|
before do
|
|
|
|
get_stage('test')
|
|
|
|
end
|
|
|
|
|
2016-12-20 14:07:34 -05:00
|
|
|
it 'responds with not found' do
|
2017-10-19 14:28:19 -04:00
|
|
|
expect(response).to have_gitlab_http_status(:not_found)
|
2016-12-20 14:07:34 -05:00
|
|
|
end
|
2016-12-20 09:53:53 -05:00
|
|
|
end
|
|
|
|
|
2018-09-06 06:21:58 -04:00
|
|
|
def get_stage(name, params = {})
|
2018-12-17 17:52:17 -05:00
|
|
|
get :stage, params: {
|
|
|
|
**params.merge(
|
|
|
|
namespace_id: project.namespace,
|
|
|
|
project_id: project,
|
|
|
|
id: pipeline.id,
|
|
|
|
stage: name,
|
|
|
|
format: :json)
|
|
|
|
}
|
2016-12-20 14:07:34 -05:00
|
|
|
end
|
2016-12-20 09:53:53 -05:00
|
|
|
end
|
2017-03-06 07:01:18 -05:00
|
|
|
|
2018-05-01 17:56:33 -04:00
|
|
|
describe 'GET stages_ajax.json' do
|
|
|
|
let(:pipeline) { create(:ci_pipeline, project: project) }
|
|
|
|
|
|
|
|
context 'when accessing existing stage' do
|
|
|
|
before do
|
|
|
|
create(:ci_build, pipeline: pipeline, stage: 'build')
|
|
|
|
|
|
|
|
get_stage_ajax('build')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns html source for stage dropdown' do
|
|
|
|
expect(response).to have_gitlab_http_status(:ok)
|
|
|
|
expect(response).to render_template('projects/pipelines/_stage')
|
|
|
|
expect(json_response).to include('html')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when accessing unknown stage' do
|
|
|
|
before do
|
|
|
|
get_stage_ajax('test')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'responds with not found' do
|
|
|
|
expect(response).to have_gitlab_http_status(:not_found)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def get_stage_ajax(name)
|
2018-12-17 17:52:17 -05:00
|
|
|
get :stage_ajax, params: {
|
|
|
|
namespace_id: project.namespace,
|
|
|
|
project_id: project,
|
|
|
|
id: pipeline.id,
|
|
|
|
stage: name
|
|
|
|
},
|
2018-05-02 12:00:12 -04:00
|
|
|
format: :json
|
2018-05-01 17:56:33 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-03-06 07:01:18 -05:00
|
|
|
describe 'GET status.json' do
|
2017-03-21 09:21:13 -04:00
|
|
|
let(:pipeline) { create(:ci_pipeline, project: project) }
|
|
|
|
let(:status) { pipeline.detailed_status(double('user')) }
|
2017-03-10 12:44:41 -05:00
|
|
|
|
2017-03-21 09:21:13 -04:00
|
|
|
before do
|
2018-12-17 17:52:17 -05:00
|
|
|
get :status, params: {
|
|
|
|
namespace_id: project.namespace,
|
|
|
|
project_id: project,
|
|
|
|
id: pipeline.id
|
|
|
|
},
|
2017-03-21 09:21:13 -04:00
|
|
|
format: :json
|
|
|
|
end
|
2017-03-06 07:01:18 -05:00
|
|
|
|
2017-03-21 09:21:13 -04:00
|
|
|
it 'return a detailed pipeline status in json' do
|
2017-10-19 14:28:19 -04:00
|
|
|
expect(response).to have_gitlab_http_status(:ok)
|
2017-03-21 09:21:13 -04:00
|
|
|
expect(json_response['text']).to eq status.text
|
|
|
|
expect(json_response['label']).to eq status.label
|
|
|
|
expect(json_response['icon']).to eq status.icon
|
2017-12-07 07:15:49 -05:00
|
|
|
expect(json_response['favicon']).to match_asset_path("/assets/ci_favicons/#{status.favicon}.png")
|
2017-03-06 07:01:18 -05:00
|
|
|
end
|
|
|
|
end
|
2017-05-04 15:01:15 -04:00
|
|
|
|
|
|
|
describe 'POST retry.json' do
|
|
|
|
let!(:pipeline) { create(:ci_pipeline, :failed, project: project) }
|
|
|
|
let!(:build) { create(:ci_build, :failed, pipeline: pipeline) }
|
|
|
|
|
|
|
|
before do
|
2018-12-17 17:52:17 -05:00
|
|
|
post :retry, params: {
|
|
|
|
namespace_id: project.namespace,
|
|
|
|
project_id: project,
|
|
|
|
id: pipeline.id
|
|
|
|
},
|
2017-05-04 15:01:15 -04:00
|
|
|
format: :json
|
|
|
|
end
|
|
|
|
|
2019-01-28 07:12:30 -05:00
|
|
|
it 'retries a pipeline without returning any content' do
|
|
|
|
expect(response).to have_gitlab_http_status(:no_content)
|
|
|
|
expect(build.reload).to be_retried
|
2017-06-07 06:32:16 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
context 'when builds are disabled' do
|
2019-01-28 07:12:30 -05:00
|
|
|
let(:feature) { ProjectFeature::DISABLED }
|
|
|
|
|
2017-06-07 06:32:16 -04:00
|
|
|
it 'fails to retry pipeline' do
|
2017-10-19 14:28:19 -04:00
|
|
|
expect(response).to have_gitlab_http_status(:not_found)
|
2017-06-07 06:32:16 -04:00
|
|
|
end
|
2017-05-04 15:01:15 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'POST cancel.json' do
|
|
|
|
let!(:pipeline) { create(:ci_pipeline, project: project) }
|
|
|
|
let!(:build) { create(:ci_build, :running, pipeline: pipeline) }
|
2017-07-03 17:58:28 -04:00
|
|
|
|
2017-05-04 15:01:15 -04:00
|
|
|
before do
|
2018-12-17 17:52:17 -05:00
|
|
|
post :cancel, params: {
|
|
|
|
namespace_id: project.namespace,
|
|
|
|
project_id: project,
|
|
|
|
id: pipeline.id
|
|
|
|
},
|
2017-05-04 15:01:15 -04:00
|
|
|
format: :json
|
|
|
|
end
|
|
|
|
|
2019-10-23 05:06:03 -04:00
|
|
|
it 'cancels a pipeline without returning any content', :sidekiq_might_not_need_inline do
|
2019-01-28 07:12:30 -05:00
|
|
|
expect(response).to have_gitlab_http_status(:no_content)
|
|
|
|
expect(pipeline.reload).to be_canceled
|
2017-06-07 06:32:16 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
context 'when builds are disabled' do
|
2019-01-28 07:12:30 -05:00
|
|
|
let(:feature) { ProjectFeature::DISABLED }
|
|
|
|
|
2017-06-07 06:32:16 -04:00
|
|
|
it 'fails to retry pipeline' do
|
2017-10-19 14:28:19 -04:00
|
|
|
expect(response).to have_gitlab_http_status(:not_found)
|
2017-06-07 06:32:16 -04:00
|
|
|
end
|
2017-05-04 15:01:15 -04:00
|
|
|
end
|
|
|
|
end
|
2018-07-27 00:26:13 -04:00
|
|
|
|
2019-10-08 05:06:09 -04:00
|
|
|
describe 'GET test_report.json' do
|
|
|
|
subject(:get_test_report_json) do
|
2020-02-07 16:08:39 -05:00
|
|
|
get :test_report, params: {
|
2019-10-08 05:06:09 -04:00
|
|
|
namespace_id: project.namespace,
|
|
|
|
project_id: project,
|
|
|
|
id: pipeline.id
|
|
|
|
},
|
|
|
|
format: :json
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when feature is enabled' do
|
|
|
|
before do
|
|
|
|
stub_feature_flags(junit_pipeline_view: true)
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when pipeline does not have a test report' do
|
|
|
|
let(:pipeline) { create(:ci_pipeline, project: project) }
|
|
|
|
|
|
|
|
it 'renders an empty test report' do
|
|
|
|
get_test_report_json
|
|
|
|
|
|
|
|
expect(response).to have_gitlab_http_status(:ok)
|
|
|
|
expect(json_response['total_count']).to eq(0)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when pipeline has a test report' do
|
|
|
|
let(:pipeline) { create(:ci_pipeline, :with_test_reports, project: project) }
|
|
|
|
|
|
|
|
it 'renders the test report' do
|
|
|
|
get_test_report_json
|
|
|
|
|
|
|
|
expect(response).to have_gitlab_http_status(:ok)
|
|
|
|
expect(json_response['total_count']).to eq(4)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when pipeline has corrupt test reports' do
|
|
|
|
let(:pipeline) { create(:ci_pipeline, project: project) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
job = create(:ci_build, pipeline: pipeline)
|
|
|
|
create(:ci_job_artifact, :junit_with_corrupted_data, job: job, project: project)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'renders the test reports' do
|
|
|
|
get_test_report_json
|
|
|
|
|
|
|
|
expect(response).to have_gitlab_http_status(:ok)
|
|
|
|
expect(json_response['status']).to eq('error_parsing_report')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when feature is disabled' do
|
|
|
|
let(:pipeline) { create(:ci_empty_pipeline, project: project) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
stub_feature_flags(junit_pipeline_view: false)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'renders empty response' do
|
|
|
|
get_test_report_json
|
|
|
|
|
|
|
|
expect(response).to have_gitlab_http_status(:no_content)
|
|
|
|
expect(response.body).to be_empty
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-02-11 04:08:39 -05:00
|
|
|
describe 'GET test_report_count.json' do
|
|
|
|
subject(:test_reports_count_json) do
|
|
|
|
get :test_reports_count, params: {
|
|
|
|
namespace_id: project.namespace,
|
|
|
|
project_id: project,
|
|
|
|
id: pipeline.id
|
|
|
|
},
|
|
|
|
format: :json
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when feature is enabled' do
|
|
|
|
before do
|
|
|
|
stub_feature_flags(junit_pipeline_view: true)
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when pipeline does not have a test report' do
|
|
|
|
let(:pipeline) { create(:ci_pipeline, project: project) }
|
|
|
|
|
|
|
|
it 'renders an empty badge counter' do
|
|
|
|
test_reports_count_json
|
|
|
|
|
|
|
|
expect(response).to have_gitlab_http_status(:ok)
|
|
|
|
expect(json_response['total_count']).to eq(0)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when pipeline has a test report' do
|
|
|
|
let(:pipeline) { create(:ci_pipeline, :with_test_reports, project: project) }
|
|
|
|
|
|
|
|
it 'renders the badge counter value' do
|
|
|
|
test_reports_count_json
|
|
|
|
|
|
|
|
expect(response).to have_gitlab_http_status(:ok)
|
|
|
|
expect(json_response['total_count']).to eq(4)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when pipeline has corrupt test reports' do
|
|
|
|
let(:pipeline) { create(:ci_pipeline, project: project) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
job = create(:ci_build, pipeline: pipeline)
|
|
|
|
create(:ci_job_artifact, :junit_with_corrupted_data, job: job, project: project)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'renders 0' do
|
|
|
|
test_reports_count_json
|
|
|
|
|
|
|
|
expect(response).to have_gitlab_http_status(:ok)
|
|
|
|
expect(json_response['total_count']).to eq(0)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when feature is disabled' do
|
|
|
|
let(:pipeline) { create(:ci_empty_pipeline, project: project) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
stub_feature_flags(junit_pipeline_view: false)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'renders empty response' do
|
|
|
|
test_reports_count_json
|
|
|
|
|
|
|
|
expect(response).to have_gitlab_http_status(:no_content)
|
|
|
|
expect(response.body).to be_empty
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-07-27 00:26:13 -04:00
|
|
|
describe 'GET latest' do
|
|
|
|
let(:branch_main) { project.repository.branches[0] }
|
|
|
|
let(:branch_secondary) { project.repository.branches[1] }
|
|
|
|
|
|
|
|
let!(:pipeline_master) do
|
|
|
|
create(:ci_pipeline,
|
|
|
|
ref: branch_main.name,
|
|
|
|
sha: branch_main.target,
|
|
|
|
project: project)
|
|
|
|
end
|
|
|
|
|
|
|
|
let!(:pipeline_secondary) do
|
|
|
|
create(:ci_pipeline,
|
|
|
|
ref: branch_secondary.name,
|
|
|
|
sha: branch_secondary.target,
|
|
|
|
project: project)
|
|
|
|
end
|
|
|
|
|
|
|
|
before do
|
|
|
|
project.change_head(branch_main.name)
|
|
|
|
project.reload_default_branch
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'no ref provided' do
|
|
|
|
it 'shows latest pipeline for the default project branch' do
|
|
|
|
get :show, params: { namespace_id: project.namespace, project_id: project, latest: true, ref: nil }
|
|
|
|
|
2020-01-28 07:08:44 -05:00
|
|
|
expect(response).to have_gitlab_http_status(:ok)
|
2018-07-27 00:26:13 -04:00
|
|
|
expect(assigns(:pipeline)).to have_attributes(id: pipeline_master.id)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'ref provided' do
|
|
|
|
before do
|
|
|
|
create(:ci_pipeline, ref: 'master', project: project)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'shows the latest pipeline for the provided ref' do
|
|
|
|
get :show, params: { namespace_id: project.namespace, project_id: project, latest: true, ref: branch_secondary.name }
|
|
|
|
|
2020-01-28 07:08:44 -05:00
|
|
|
expect(response).to have_gitlab_http_status(:ok)
|
2018-07-27 00:26:13 -04:00
|
|
|
expect(assigns(:pipeline)).to have_attributes(id: pipeline_secondary.id)
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'newer pipeline exists for older sha' do
|
|
|
|
before do
|
|
|
|
create(:ci_pipeline, ref: branch_secondary.name, sha: project.commit(branch_secondary.name).parent, project: project)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'shows the provided ref with the last sha/pipeline combo' do
|
|
|
|
get :show, params: { namespace_id: project.namespace, project_id: project, latest: true, ref: branch_secondary.name }
|
|
|
|
|
2020-01-28 07:08:44 -05:00
|
|
|
expect(response).to have_gitlab_http_status(:ok)
|
2018-07-27 00:26:13 -04:00
|
|
|
expect(assigns(:pipeline)).to have_attributes(id: pipeline_secondary.id)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'renders a 404 if no pipeline is found for the ref' do
|
|
|
|
get :show, params: { namespace_id: project.namespace, project_id: project, ref: 'no-branch' }
|
|
|
|
|
2020-01-28 07:08:44 -05:00
|
|
|
expect(response).to have_gitlab_http_status(:not_found)
|
2018-07-27 00:26:13 -04:00
|
|
|
end
|
|
|
|
end
|
2020-01-08 07:07:59 -05:00
|
|
|
|
|
|
|
describe 'DELETE #destroy' do
|
|
|
|
let!(:project) { create(:project, :private, :repository) }
|
|
|
|
let!(:pipeline) { create(:ci_pipeline, :failed, project: project) }
|
|
|
|
let!(:build) { create(:ci_build, :failed, pipeline: pipeline) }
|
|
|
|
|
|
|
|
context 'when user has ability to delete pipeline' do
|
|
|
|
before do
|
|
|
|
sign_in(project.owner)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'deletes pipeline and redirects' do
|
|
|
|
delete_pipeline
|
|
|
|
|
2020-01-28 07:08:44 -05:00
|
|
|
expect(response).to have_gitlab_http_status(:see_other)
|
2020-01-08 07:07:59 -05:00
|
|
|
|
|
|
|
expect(Ci::Build.exists?(build.id)).to be_falsy
|
|
|
|
expect(Ci::Pipeline.exists?(pipeline.id)).to be_falsy
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'and builds are disabled' do
|
|
|
|
let(:feature) { ProjectFeature::DISABLED }
|
|
|
|
|
|
|
|
it 'fails to delete pipeline' do
|
|
|
|
delete_pipeline
|
|
|
|
|
2020-01-28 07:08:44 -05:00
|
|
|
expect(response).to have_gitlab_http_status(:not_found)
|
2020-01-08 07:07:59 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when user has no privileges' do
|
|
|
|
it 'fails to delete pipeline' do
|
|
|
|
delete_pipeline
|
|
|
|
|
2020-01-28 07:08:44 -05:00
|
|
|
expect(response).to have_gitlab_http_status(:forbidden)
|
2020-01-08 07:07:59 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def delete_pipeline
|
|
|
|
delete :destroy, params: {
|
|
|
|
namespace_id: project.namespace,
|
|
|
|
project_id: project,
|
|
|
|
id: pipeline.id
|
|
|
|
}
|
|
|
|
end
|
|
|
|
end
|
2016-12-20 09:53:53 -05:00
|
|
|
end
|