# frozen_string_literal: true require 'spec_helper' RSpec.describe 'Pipelines', :js, feature_category: :projects do include ListboxHelpers include ProjectForksHelper include Spec::Support::Helpers::ModalHelpers let(:project) { create(:project) } let(:expected_detached_mr_tag) { 'merge request' } context 'when user is logged in' do let(:user) { create(:user) } before do sign_in(user) project.add_developer(user) project.update!(auto_devops_attributes: { enabled: false }) end describe 'GET /:project/-/pipelines' do let(:project) { create(:project, :repository) } let!(:pipeline) do create( :ci_empty_pipeline, project: project, ref: 'master', status: 'running', sha: project.commit.id ) end context 'scope' do before do create(:ci_empty_pipeline, status: 'pending', project: project, sha: project.commit.id, ref: 'master') create(:ci_empty_pipeline, status: 'running', project: project, sha: project.commit.id, ref: 'master') create(:ci_empty_pipeline, status: 'created', project: project, sha: project.commit.id, ref: 'master') create(:ci_empty_pipeline, status: 'success', project: project, sha: project.commit.id, ref: 'master') end [:all, :running, :pending, :finished, :branches].each do |scope| context "when displaying #{scope}" do before do visit_project_pipelines(scope: scope) end it 'contains pipeline commit short SHA' do expect(page).to have_content(pipeline.short_sha) end it 'contains branch name' do expect(page).to have_content(pipeline.ref) end end end end context 'header tabs' do before do visit project_pipelines_path(project) wait_for_requests end it 'shows a tab for All pipelines and count' do expect(page.find('.js-pipelines-tab-all').text).to include('All') expect(page.find('.js-pipelines-tab-all .badge').text).to include('1') end it 'shows a tab for Finished pipelines and count' do expect(page.find('.js-pipelines-tab-finished').text).to include('Finished') end it 'shows a tab for Branches' do expect(page.find('.js-pipelines-tab-branches').text).to include('Branches') end it 'shows a tab for Tags' do expect(page.find('.js-pipelines-tab-tags').text).to include('Tags') end it 'updates content when tab is clicked' do page.find('.js-pipelines-tab-finished').click wait_for_requests expect(page).to have_content('There are currently no finished pipelines.') end end context 'navigation links' do before do visit project_pipelines_path(project) wait_for_requests end it 'renders "CI lint" link' do expect(page).to have_link('CI lint') end it 'renders "Run pipeline" link' do expect(page).to have_link('Run pipeline') end end context 'when pipeline is cancelable' do let!(:build) do create(:ci_build, pipeline: pipeline, stage: 'test') end before do build.run visit_project_pipelines end it 'indicates that pipeline can be canceled' do expect(page).to have_selector('.js-pipelines-cancel-button') expect(page).to have_selector('.ci-running') end context 'when canceling' do before do find('.js-pipelines-cancel-button').click click_button 'Stop pipeline' wait_for_requests end it 'indicated that pipelines was canceled', :sidekiq_might_not_need_inline do expect(page).not_to have_selector('.js-pipelines-cancel-button') expect(page).to have_selector('.ci-canceled') end end end context 'when pipeline is retryable', :sidekiq_might_not_need_inline do let!(:build) do create(:ci_build, pipeline: pipeline, stage: 'test') end before do build.drop visit_project_pipelines end it 'indicates that pipeline can be retried' do expect(page).to have_selector('.js-pipelines-retry-button') expect(page).to have_selector('.ci-failed') end context 'when retrying' do before do find('.js-pipelines-retry-button').click wait_for_requests end it 'shows running pipeline that is not retryable' do expect(page).not_to have_selector('.js-pipelines-retry-button') expect(page).to have_selector('.ci-running') end end end context 'when pipeline is detached merge request pipeline' do let(:merge_request) do create(:merge_request, :with_detached_merge_request_pipeline, source_project: source_project, target_project: target_project) end let!(:pipeline) { merge_request.all_pipelines.first } let(:source_project) { project } let(:target_project) { project } before do visit project_pipelines_path(source_project) end shared_examples_for 'detached merge request pipeline' do it 'shows pipeline information without pipeline ref', :sidekiq_might_not_need_inline do within '.pipeline-tags' do expect(page).to have_content(expected_detached_mr_tag) expect(page).to have_link(merge_request.iid, href: project_merge_request_path(project, merge_request)) expect(page).not_to have_link(pipeline.ref) end end end it_behaves_like 'detached merge request pipeline' context 'when source project is a forked project' do let(:source_project) { fork_project(project, user, repository: true) } it_behaves_like 'detached merge request pipeline' end end context 'when pipeline is merge request pipeline' do let(:merge_request) do create(:merge_request, :with_merge_request_pipeline, source_project: source_project, target_project: target_project, merge_sha: target_project.commit.sha) end let!(:pipeline) { merge_request.all_pipelines.first } let(:source_project) { project } let(:target_project) { project } before do visit project_pipelines_path(source_project) end shared_examples_for 'Correct merge request pipeline information' do it 'does not show detached tag for the pipeline, and shows the link of the merge request, and does not show the ref of the pipeline', :sidekiq_might_not_need_inline do within '.pipeline-tags' do expect(page).not_to have_content(expected_detached_mr_tag) expect(page).to have_link(merge_request.iid, href: project_merge_request_path(project, merge_request)) expect(page).not_to have_link(pipeline.ref) end end end it_behaves_like 'Correct merge request pipeline information' context 'when source project is a forked project' do let(:source_project) { fork_project(project, user, repository: true) } it_behaves_like 'Correct merge request pipeline information' end end context 'when pipeline has configuration errors' do let(:pipeline) do create(:ci_pipeline, :invalid, project: project) end before do visit_project_pipelines end it 'contains badge that indicates errors' do expect(page).to have_content 'yaml invalid' end it 'contains badge with tooltip which contains error' do expect(pipeline).to have_yaml_errors expect(page).to have_selector( %Q{span[title="#{pipeline.yaml_errors}"]}) end it 'contains badge that indicates failure reason' do expect(page).to have_content 'error' end it 'contains badge with tooltip which contains failure reason' do expect(pipeline.failure_reason?).to eq true expect(page).to have_selector( %Q{span[title="#{pipeline.present.failure_reason}"]}) end end context 'with manual actions' do let!(:manual) do create(:ci_build, :manual, pipeline: pipeline, name: 'manual build', stage: 'test') end before do visit_project_pipelines end it 'has a dropdown with play button' do expect(page).to have_selector('[data-testid="pipelines-manual-actions-dropdown"] [data-testid="play-icon"]') end it 'has link to the manual action' do find('[data-testid="pipelines-manual-actions-dropdown"]').click expect(page).to have_button('manual build') end context 'when manual action was played' do before do find('[data-testid="pipelines-manual-actions-dropdown"]').click click_button('manual build') end it 'enqueues manual action job' do expect(page).to have_selector('[data-testid="pipelines-manual-actions-dropdown"] .gl-dropdown-toggle:disabled') end end end context 'when there is a delayed job' do let!(:delayed_job) do create(:ci_build, :scheduled, pipeline: pipeline, name: 'delayed job 1', stage: 'test') end before do visit_project_pipelines end it 'has a dropdown for actionable jobs' do expect(page).to have_selector('[data-testid="pipelines-manual-actions-dropdown"] [data-testid="play-icon"]') end it "has link to the delayed job's action" do find('[data-testid="pipelines-manual-actions-dropdown"]').click time_diff = [0, delayed_job.scheduled_at - Time.zone.now].max expect(page).to have_button('delayed job 1') expect(page).to have_content(Time.at(time_diff).utc.strftime("%H:%M:%S")) end context 'when delayed job is expired already' do let!(:delayed_job) do create(:ci_build, :expired_scheduled, pipeline: pipeline, name: 'delayed job 1', stage: 'test') end it "shows 00:00:00 as the remaining time" do find('[data-testid="pipelines-manual-actions-dropdown"]').click expect(page).to have_content("00:00:00") end end context 'when user played a delayed job immediately' do let(:manual_action_selector) { '[data-testid="pipelines-manual-actions-dropdown"]' } before do find(manual_action_selector).click accept_gl_confirm do click_button 'delayed job 1' end # Wait for UI to transition to ensure a request has been made within(manual_action_selector) { find('.gl-spinner') } within(manual_action_selector) { find('[data-testid="play-icon"]') } wait_for_requests end it 'enqueues the delayed job', :js do expect(delayed_job.reload).to be_pending end end end context 'for generic statuses' do context 'when preparing' do let!(:pipeline) do create(:ci_empty_pipeline, status: 'preparing', project: project) end let!(:status) do create(:generic_commit_status, :preparing, pipeline: pipeline) end before do visit_project_pipelines end it 'is cancelable' do expect(page).to have_selector('.js-pipelines-cancel-button') end it 'shows the pipeline as preparing' do expect(page).to have_selector('.ci-preparing') end end context 'when running' do let!(:running) do create(:generic_commit_status, status: 'running', pipeline: pipeline, stage: 'test') end before do visit_project_pipelines end it 'is cancelable' do expect(page).to have_selector('.js-pipelines-cancel-button') end it 'has pipeline running' do expect(page).to have_selector('.ci-running') end context 'when canceling' do before do find('.js-pipelines-cancel-button').click click_button 'Stop pipeline' end it 'indicates that pipeline was canceled', :sidekiq_might_not_need_inline do expect(page).not_to have_selector('.js-pipelines-cancel-button') expect(page).to have_selector('.ci-canceled') end end end context 'when failed' do let!(:status) do create(:generic_commit_status, :pending, pipeline: pipeline, stage: 'test') end before do status.drop visit_project_pipelines end it 'is not retryable' do expect(page).not_to have_selector('.js-pipelines-retry-button') end it 'has failed pipeline', :sidekiq_might_not_need_inline do expect(page).to have_selector('.ci-failed') end end end context 'downloadable pipelines' do context 'with artifacts' do let!(:with_artifacts) do build = create(:ci_build, :success, pipeline: pipeline, name: 'rspec tests', stage: 'test') create(:ci_job_artifact, :codequality, job: build) end before do visit_project_pipelines end it 'has artifacts dropdown' do expect(page).to have_selector('[data-testid="pipeline-multi-actions-dropdown"]') end end context 'with artifacts expired' do let!(:with_artifacts_expired) do create(:ci_build, :expired, :success, pipeline: pipeline, name: 'rspec', stage: 'test') end before do visit_project_pipelines end it { expect(page).not_to have_selector('[data-testid="artifact-item"]') } end context 'without artifacts' do let!(:without_artifacts) do create(:ci_build, :success, pipeline: pipeline, name: 'rspec', stage: 'test') end before do visit_project_pipelines end it { expect(page).not_to have_selector('[data-testid="artifact-item"]') } end context 'with trace artifact' do before do create(:ci_build, :success, :trace_artifact, pipeline: pipeline) visit_project_pipelines end it 'does not show trace artifact as artifacts' do expect(page).not_to have_selector('[data-testid="artifact-item"]') end end end context 'mini pipeline graph' do let!(:build) do create(:ci_build, :pending, pipeline: pipeline, stage: 'build', name: 'build') end dropdown_selector = '[data-testid="mini-pipeline-graph-dropdown"]' before do visit_project_pipelines end it 'renders a mini pipeline graph' do expect(page).to have_selector('[data-testid="pipeline-mini-graph"]') expect(page).to have_selector(dropdown_selector) end context 'when clicking a stage badge' do it 'opens a dropdown' do find(dropdown_selector).click expect(page).to have_link build.name end it 'is possible to cancel pending build' do find(dropdown_selector).click find('.js-ci-action').click wait_for_requests expect(build.reload).to be_canceled end end context 'for a failed pipeline' do let!(:build) do create(:ci_build, :failed, pipeline: pipeline, stage: 'build', name: 'build') end it 'displays the failure reason' do find(dropdown_selector).click within('.js-builds-dropdown-list') do build_element = page.find('.mini-pipeline-graph-dropdown-item') expect(build_element['title']).to eq('build - failed - (unknown failure)') end end end end context 'with pagination' do before do allow(Ci::Pipeline).to receive(:default_per_page).and_return(1) create(:ci_empty_pipeline, project: project) end it 'renders pagination' do visit project_pipelines_path(project) wait_for_requests expect(page).to have_selector('.gl-pagination') end it 'renders second page of pipelines' do visit project_pipelines_path(project, page: '2') wait_for_requests expect(page).to have_selector('.gl-pagination .page-link', count: 4) end it 'shows updated content' do visit project_pipelines_path(project) wait_for_requests page.find('.page-link.next-page-item').click expect(page).to have_selector('.gl-pagination .page-link', count: 4) end end context 'with pipeline key selection' do before do visit project_pipelines_path(project) wait_for_requests end it 'changes the Pipeline ID column for Pipeline IID' do page.find('[data-testid="pipeline-key-collapsible-box"]').click within '.gl-new-dropdown-contents' do dropdown_options = page.find_all '.gl-new-dropdown-item' dropdown_options[1].click end expect(page.find('[data-testid="pipeline-th"]')).to have_content 'Pipeline' expect(page.find('[data-testid="pipeline-url-link"]')).to have_content "##{pipeline.iid}" end end end describe 'GET /:project/-/pipelines/show' do let(:project) { create(:project, :repository) } let(:pipeline) do create(:ci_empty_pipeline, project: project, sha: project.commit.id, user: user) end let(:external_stage) { create(:ci_stage, name: 'external', pipeline: pipeline) } before do create_build('build', 0, 'build', :success) create_build('test', 1, 'rspec 0:2', :pending) create_build('test', 1, 'rspec 1:2', :running) create_build('test', 1, 'spinach 0:2', :created) create_build('test', 1, 'spinach 1:2', :created) create_build('test', 1, 'audit', :created) create_build('deploy', 2, 'production', :created) create(:generic_commit_status, pipeline: pipeline, ci_stage: external_stage, name: 'jenkins', ref: 'master') visit project_pipeline_path(project, pipeline) wait_for_requests end it 'shows a graph with grouped stages' do expect(page).to have_css('.js-pipeline-graph') # header expect(page).to have_text("##{pipeline.id}") expect(page).to have_selector(%Q(img[src="#{pipeline.user.avatar_url}"])) expect(page).to have_link(pipeline.user.name, href: user_path(pipeline.user)) # stages expect(page).to have_text('build') expect(page).to have_text('test') expect(page).to have_text('deploy') expect(page).to have_text('external') # builds expect(page).to have_text('rspec') expect(page).to have_text('spinach') expect(page).to have_text('rspec') expect(page).to have_text('production') expect(page).to have_text('jenkins') end def create_build(stage, stage_idx, name, status) create(:ci_build, pipeline: pipeline, stage: stage, stage_idx: stage_idx, name: name, status: status) end end describe 'POST /:project/-/pipelines' do let(:project) { create(:project, :repository) } before do visit new_project_pipeline_path(project) end context 'for valid commit', :js do before do click_button project.default_branch wait_for_requests find('.gl-new-dropdown-item', text: 'master').click wait_for_requests end context 'with gitlab-ci.yml', :js do before do stub_ci_pipeline_to_return_yaml_file end it 'creates a new pipeline' do expect do click_on 'Run pipeline' wait_for_requests end .to change { Ci::Pipeline.count }.by(1) expect(Ci::Pipeline.last).to be_web end context 'when variables are specified' do it 'creates a new pipeline with variables', quarantine: 'https://gitlab.com/gitlab-org/gitlab/-/issues/375552' do page.within(find("[data-testid='ci-variable-row']")) do find("[data-testid='pipeline-form-ci-variable-key']").set('key_name') find("[data-testid='pipeline-form-ci-variable-value']").set('value') end expect do click_on 'Run pipeline' wait_for_requests end .to change { Ci::Pipeline.count }.by(1) expect(Ci::Pipeline.last.variables.map { |var| var.slice(:key, :secret_value) }) .to eq [{ key: "key_name", secret_value: "value" }.with_indifferent_access] end end end context 'without gitlab-ci.yml' do before do click_on 'Run pipeline' wait_for_requests end it { expect(page).to have_content('Missing CI config file') } it 'creates a pipeline after first request failed and a valid gitlab-ci.yml file is available when trying again', quarantine: 'https://gitlab.com/gitlab-org/gitlab/-/issues/375552' do stub_ci_pipeline_to_return_yaml_file expect do click_on 'Run pipeline' wait_for_requests end .to change { Ci::Pipeline.count }.by(1) end end end end describe 'Reset runner caches' do let(:project) { create(:project, :repository) } before do create(:ci_empty_pipeline, status: 'success', project: project, sha: project.commit.id, ref: 'master') project.add_maintainer(user) visit project_pipelines_path(project) end it 'has a clear caches button' do expect(page).to have_button 'Clear runner caches' end describe 'user clicks the button' do context 'when project already has jobs_cache_index' do before do project.update!(jobs_cache_index: 1) end it 'increments jobs_cache_index' do click_button 'Clear runner caches' wait_for_requests expect(page.find('[data-testid="alert-info"]')).to have_content 'Project cache successfully reset.' end end context 'when project does not have jobs_cache_index' do it 'sets jobs_cache_index to 1' do click_button 'Clear runner caches' wait_for_requests expect(page.find('[data-testid="alert-info"]')).to have_content 'Project cache successfully reset.' end end end end describe 'Run Pipelines' do let(:project) { create(:project, :repository) } before do visit new_project_pipeline_path(project) end describe 'new pipeline page' do it 'has field to add a new pipeline' do expect(page).to have_button project.default_branch expect(page).to have_content('Run for') end end describe 'find pipelines' do it 'shows filtered pipelines', :js do click_button project.default_branch send_keys('fix') expect_listbox_item('fix') end end end describe 'Empty State' do let(:project) { create(:project, :repository) } context 'when `ios_specific_templates` is not enabled' do before do visit project_pipelines_path(project) end it 'renders empty state' do expect(page).to have_content 'Try test template' end end describe 'when the `ios_specific_templates` experiment is enabled and the "Set up a runner" button is clicked' do before do stub_experiments(ios_specific_templates: :candidate) create(:project_setting, project: project, target_platforms: %w(ios)) visit project_pipelines_path(project) click_button 'Set up a runner' end it 'displays a modal with the macOS platform selected and an explanation popover' do expect(page).to have_button 'macOS', class: 'selected' expect(page).to have_selector('#runner-instructions-modal___BV_modal_content_') expect(page).to have_selector('.popover') end end end end context 'when user is not logged in' do before do project.update!(auto_devops_attributes: { enabled: false }) visit project_pipelines_path(project) end context 'when project is public' do let(:project) { create(:project, :public, :repository) } context 'without pipelines' do it { expect(page).to have_content 'This project is not currently set up to run pipelines.' } end end context 'when project is private' do let(:project) { create(:project, :private, :repository) } it 'redirects the user to sign_in and displays the flash alert' do expect(page).to have_content 'You need to sign in' expect(page).to have_current_path("/users/sign_in") end end end def visit_project_pipelines(**query) visit project_pipelines_path(project, query) wait_for_requests end end