2016-05-13 09:36:44 -04:00
|
|
|
require 'spec_helper'
|
|
|
|
|
2017-06-29 00:13:10 -04:00
|
|
|
describe 'Pipelines', :js do
|
2019-03-20 11:39:27 -04:00
|
|
|
include ProjectForksHelper
|
|
|
|
|
2017-08-02 15:55:11 -04:00
|
|
|
let(:project) { create(:project) }
|
2016-05-13 09:36:44 -04:00
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
context 'when user is logged in' do
|
|
|
|
let(:user) { create(:user) }
|
2016-05-21 18:58:11 -04:00
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
before do
|
2017-06-21 19:44:10 -04:00
|
|
|
sign_in(user)
|
2017-12-22 03:18:28 -05:00
|
|
|
project.add_developer(user)
|
2018-09-18 15:20:43 -04:00
|
|
|
project.update!(auto_devops_attributes: { enabled: false })
|
2016-05-21 18:58:11 -04:00
|
|
|
end
|
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
describe 'GET /:project/pipelines' do
|
2017-07-24 18:51:14 -04:00
|
|
|
let(:project) { create(:project, :repository) }
|
2017-01-04 06:20:12 -05:00
|
|
|
|
|
|
|
let!(:pipeline) do
|
|
|
|
create(
|
|
|
|
:ci_empty_pipeline,
|
|
|
|
project: project,
|
|
|
|
ref: 'master',
|
|
|
|
status: 'running',
|
2017-05-03 07:27:17 -04:00
|
|
|
sha: project.commit.id
|
2017-01-04 06:20:12 -05:00
|
|
|
)
|
2016-11-15 15:45:05 -05:00
|
|
|
end
|
|
|
|
|
2017-02-23 08:44:58 -05:00
|
|
|
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
|
2016-12-08 13:05:20 -05:00
|
|
|
|
2017-02-23 08:44:58 -05:00
|
|
|
[: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
|
2017-02-08 12:48:41 -05:00
|
|
|
|
2017-02-23 08:44:58 -05:00
|
|
|
it 'contains branch name' do
|
|
|
|
expect(page).to have_content(pipeline.ref)
|
|
|
|
end
|
2017-02-08 12:48:41 -05:00
|
|
|
end
|
2017-01-04 06:20:12 -05:00
|
|
|
end
|
2016-11-17 13:55:10 -05:00
|
|
|
end
|
2016-05-13 14:17:15 -04:00
|
|
|
|
2017-02-15 09:06:00 -05:00
|
|
|
context 'header tabs' do
|
|
|
|
before do
|
2017-07-06 12:20:50 -04:00
|
|
|
visit project_pipelines_path(project)
|
2017-05-17 14:25:13 -04:00
|
|
|
wait_for_requests
|
2017-02-15 09:06:00 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'shows a tab for All pipelines and count' do
|
2017-11-14 10:35:29 -05:00
|
|
|
expect(page.find('.js-pipelines-tab-all').text).to include('All')
|
2017-02-15 09:06:00 -05:00
|
|
|
expect(page.find('.js-pipelines-tab-all .badge').text).to include('1')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'shows a tab for Pending pipelines and count' do
|
2017-11-14 10:35:29 -05:00
|
|
|
expect(page.find('.js-pipelines-tab-pending').text).to include('Pending')
|
2017-02-15 09:06:00 -05:00
|
|
|
expect(page.find('.js-pipelines-tab-pending .badge').text).to include('0')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'shows a tab for Running pipelines and count' do
|
2017-11-14 10:35:29 -05:00
|
|
|
expect(page.find('.js-pipelines-tab-running').text).to include('Running')
|
2017-02-15 09:06:00 -05:00
|
|
|
expect(page.find('.js-pipelines-tab-running .badge').text).to include('1')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'shows a tab for Finished pipelines and count' do
|
2017-11-14 10:35:29 -05:00
|
|
|
expect(page.find('.js-pipelines-tab-finished').text).to include('Finished')
|
2017-02-15 09:06:00 -05:00
|
|
|
expect(page.find('.js-pipelines-tab-finished .badge').text).to include('0')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'shows a tab for Branches' do
|
2017-11-14 10:35:29 -05:00
|
|
|
expect(page.find('.js-pipelines-tab-branches').text).to include('Branches')
|
2017-02-15 09:06:00 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'shows a tab for Tags' do
|
2017-11-14 10:35:29 -05:00
|
|
|
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-pending').click
|
|
|
|
wait_for_requests
|
2018-03-01 12:40:00 -05:00
|
|
|
expect(page).to have_content('There are currently no pending pipelines.')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'navigation links' do
|
|
|
|
before do
|
|
|
|
visit project_pipelines_path(project)
|
|
|
|
wait_for_requests
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'renders run pipeline link' do
|
|
|
|
expect(page).to have_link('Run Pipeline')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'renders ci lint link' do
|
|
|
|
expect(page).to have_link('CI Lint')
|
2017-02-15 09:06:00 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
context 'when pipeline is cancelable' do
|
|
|
|
let!(:build) do
|
|
|
|
create(:ci_build, pipeline: pipeline,
|
2018-09-02 10:35:15 -04:00
|
|
|
stage: 'test')
|
2017-01-04 06:20:12 -05:00
|
|
|
end
|
2016-05-13 14:17:15 -04:00
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
before do
|
|
|
|
build.run
|
|
|
|
visit_project_pipelines
|
|
|
|
end
|
2016-05-13 14:17:15 -04:00
|
|
|
|
2017-01-05 09:14:06 -05:00
|
|
|
it 'indicates that pipeline can be canceled' do
|
2017-03-17 13:30:32 -04:00
|
|
|
expect(page).to have_selector('.js-pipelines-cancel-button')
|
2017-01-05 09:14:06 -05:00
|
|
|
expect(page).to have_selector('.ci-running')
|
|
|
|
end
|
2016-07-16 19:48:51 -04:00
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
context 'when canceling' do
|
2017-03-17 13:30:32 -04:00
|
|
|
before do
|
2018-02-06 22:05:54 -05:00
|
|
|
find('.js-pipelines-cancel-button').click
|
2018-05-11 10:03:27 -04:00
|
|
|
find('.js-modal-primary-action').click
|
2017-05-17 14:25:13 -04:00
|
|
|
wait_for_requests
|
2017-03-17 13:30:32 -04:00
|
|
|
end
|
2016-07-16 19:48:51 -04:00
|
|
|
|
2017-01-05 09:14:06 -05:00
|
|
|
it 'indicated that pipelines was canceled' do
|
2017-03-17 13:30:32 -04:00
|
|
|
expect(page).not_to have_selector('.js-pipelines-cancel-button')
|
2017-01-05 09:14:06 -05:00
|
|
|
expect(page).to have_selector('.ci-canceled')
|
|
|
|
end
|
2017-01-04 06:20:12 -05:00
|
|
|
end
|
2016-12-20 14:07:34 -05:00
|
|
|
end
|
2016-07-16 19:48:51 -04:00
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
context 'when pipeline is retryable' do
|
|
|
|
let!(:build) do
|
|
|
|
create(:ci_build, pipeline: pipeline,
|
2018-09-02 10:35:15 -04:00
|
|
|
stage: 'test')
|
2017-01-04 06:20:12 -05:00
|
|
|
end
|
2016-12-08 13:22:05 -05:00
|
|
|
|
2016-11-15 15:45:05 -05:00
|
|
|
before do
|
2017-01-04 06:20:12 -05:00
|
|
|
build.drop
|
|
|
|
visit_project_pipelines
|
2016-11-15 15:45:05 -05:00
|
|
|
end
|
2016-07-16 19:48:51 -04:00
|
|
|
|
2017-01-05 09:14:06 -05:00
|
|
|
it 'indicates that pipeline can be retried' do
|
2017-03-17 13:30:32 -04:00
|
|
|
expect(page).to have_selector('.js-pipelines-retry-button')
|
2017-01-05 09:14:06 -05:00
|
|
|
expect(page).to have_selector('.ci-failed')
|
|
|
|
end
|
2017-01-04 06:20:12 -05:00
|
|
|
|
|
|
|
context 'when retrying' do
|
2017-03-17 13:30:32 -04:00
|
|
|
before do
|
|
|
|
find('.js-pipelines-retry-button').click
|
2017-05-17 14:25:13 -04:00
|
|
|
wait_for_requests
|
2017-03-17 13:30:32 -04:00
|
|
|
end
|
2017-01-04 06:20:12 -05:00
|
|
|
|
2017-01-05 09:14:06 -05:00
|
|
|
it 'shows running pipeline that is not retryable' do
|
2017-03-17 13:30:32 -04:00
|
|
|
expect(page).not_to have_selector('.js-pipelines-retry-button')
|
2017-01-05 09:14:06 -05:00
|
|
|
expect(page).to have_selector('.ci-running')
|
|
|
|
end
|
2016-12-20 14:07:34 -05:00
|
|
|
end
|
2016-07-16 19:48:51 -04:00
|
|
|
end
|
|
|
|
|
2019-03-20 11:39:27 -04:00
|
|
|
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 'showing detached merge request pipeline information' do
|
|
|
|
it 'shows detached tag for the pipeline' do
|
|
|
|
within '.pipeline-tags' do
|
|
|
|
expect(page).to have_content('detached')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'shows the link of the merge request' do
|
|
|
|
within '.branch-commit' do
|
|
|
|
expect(page).to have_link(merge_request.iid,
|
|
|
|
href: project_merge_request_path(project, merge_request))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not show the ref of the pipeline' do
|
|
|
|
within '.branch-commit' do
|
|
|
|
expect(page).not_to have_link(pipeline.ref)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it_behaves_like 'showing detached 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 'showing detached merge request pipeline information'
|
|
|
|
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' do
|
|
|
|
within '.pipeline-tags' do
|
|
|
|
expect(page).not_to have_content('detached')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'shows the link of the merge request' do
|
|
|
|
within '.branch-commit' do
|
|
|
|
expect(page).to have_link(merge_request.iid,
|
|
|
|
href: project_merge_request_path(project, merge_request))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not show the ref of the pipeline' do
|
|
|
|
within '.branch-commit' do
|
|
|
|
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
|
|
|
|
|
2017-01-04 06:48:24 -05:00
|
|
|
context 'when pipeline has configuration errors' do
|
|
|
|
let(:pipeline) do
|
|
|
|
create(:ci_pipeline, :invalid, project: project)
|
|
|
|
end
|
|
|
|
|
2017-06-14 14:18:56 -04:00
|
|
|
before do
|
|
|
|
visit_project_pipelines
|
|
|
|
end
|
2017-01-04 06:48:24 -05:00
|
|
|
|
2017-01-04 07:01:36 -05:00
|
|
|
it 'contains badge that indicates errors' do
|
2017-01-04 06:48:24 -05:00
|
|
|
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[data-original-title="#{pipeline.yaml_errors}"]})
|
|
|
|
end
|
2017-10-03 07:24:56 -04:00
|
|
|
|
|
|
|
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[data-original-title="#{pipeline.present.failure_reason}"]})
|
|
|
|
end
|
2017-01-04 06:48:24 -05:00
|
|
|
end
|
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
context 'with manual actions' do
|
|
|
|
let!(:manual) do
|
|
|
|
create(:ci_build, :manual,
|
2016-11-17 13:55:10 -05:00
|
|
|
pipeline: pipeline,
|
2017-01-04 06:20:12 -05:00
|
|
|
name: 'manual build',
|
2018-09-02 10:35:15 -04:00
|
|
|
stage: 'test')
|
2016-11-17 13:55:10 -05:00
|
|
|
end
|
2016-05-31 09:31:06 -04:00
|
|
|
|
2017-06-14 14:18:56 -04:00
|
|
|
before do
|
|
|
|
visit_project_pipelines
|
|
|
|
end
|
2016-05-31 09:31:06 -04:00
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
it 'has a dropdown with play button' do
|
2017-06-16 08:00:28 -04:00
|
|
|
expect(page).to have_selector('.dropdown-new.btn.btn-default .icon-play')
|
2016-05-31 09:31:06 -04:00
|
|
|
end
|
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
it 'has link to the manual action' do
|
|
|
|
find('.js-pipeline-dropdown-manual-actions').click
|
|
|
|
|
2017-03-17 13:30:32 -04:00
|
|
|
expect(page).to have_button('manual build')
|
2016-05-31 09:31:06 -04:00
|
|
|
end
|
2016-11-18 10:16:51 -05:00
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
context 'when manual action was played' do
|
|
|
|
before do
|
|
|
|
find('.js-pipeline-dropdown-manual-actions').click
|
2017-03-17 13:30:32 -04:00
|
|
|
click_button('manual build')
|
2017-01-04 06:20:12 -05:00
|
|
|
end
|
2016-11-18 10:16:51 -05:00
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
it 'enqueues manual action job' do
|
2017-03-17 13:30:32 -04:00
|
|
|
expect(page).to have_selector('.js-pipeline-dropdown-manual-actions:disabled')
|
2017-01-04 06:20:12 -05:00
|
|
|
end
|
2016-11-18 10:16:51 -05:00
|
|
|
end
|
2016-05-31 09:31:06 -04:00
|
|
|
end
|
|
|
|
|
2018-10-04 07:52:14 -04:00
|
|
|
context 'when there is a delayed job' do
|
2018-10-03 03:11:08 -04:00
|
|
|
let!(:delayed_job) do
|
|
|
|
create(:ci_build, :scheduled,
|
|
|
|
pipeline: pipeline,
|
|
|
|
name: 'delayed job',
|
2018-09-02 10:35:15 -04:00
|
|
|
stage: 'test')
|
2018-10-03 03:11:08 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
before do
|
|
|
|
visit_project_pipelines
|
|
|
|
end
|
|
|
|
|
2018-10-04 07:52:14 -04:00
|
|
|
it 'has a dropdown for actionable jobs' do
|
2018-10-03 03:11:08 -04:00
|
|
|
expect(page).to have_selector('.dropdown-new.btn.btn-default .icon-play')
|
|
|
|
end
|
|
|
|
|
2018-10-04 07:52:14 -04:00
|
|
|
it "has link to the delayed job's action" do
|
2018-10-03 03:11:08 -04:00
|
|
|
find('.js-pipeline-dropdown-manual-actions').click
|
|
|
|
|
2018-10-05 03:07:28 -04:00
|
|
|
time_diff = [0, delayed_job.scheduled_at - Time.now].max
|
2018-10-03 03:11:08 -04:00
|
|
|
expect(page).to have_button('delayed job')
|
2018-10-05 03:07:28 -04:00
|
|
|
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',
|
2018-09-02 10:35:15 -04:00
|
|
|
stage: 'test')
|
2018-10-05 03:07:28 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it "shows 00:00:00 as the remaining time" do
|
|
|
|
find('.js-pipeline-dropdown-manual-actions').click
|
|
|
|
|
2018-10-05 04:11:37 -04:00
|
|
|
expect(page).to have_content("00:00:00")
|
2018-10-05 03:07:28 -04:00
|
|
|
end
|
2018-10-03 03:11:08 -04:00
|
|
|
end
|
|
|
|
|
2018-10-04 07:52:14 -04:00
|
|
|
context 'when user played a delayed job immediately' do
|
2018-10-03 03:11:08 -04:00
|
|
|
before do
|
2018-10-04 07:52:14 -04:00
|
|
|
find('.js-pipeline-dropdown-manual-actions').click
|
|
|
|
page.accept_confirm { click_button('delayed job') }
|
|
|
|
wait_for_requests
|
2018-10-03 03:11:08 -04:00
|
|
|
end
|
|
|
|
|
2018-10-04 07:52:14 -04:00
|
|
|
it 'enqueues the delayed job', :js do
|
|
|
|
expect(delayed_job.reload).to be_pending
|
2018-10-03 03:11:08 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
context 'for generic statuses' do
|
2019-02-26 21:13:06 -05:00
|
|
|
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
|
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
context 'when running' do
|
|
|
|
let!(:running) do
|
|
|
|
create(:generic_commit_status,
|
|
|
|
status: 'running',
|
|
|
|
pipeline: pipeline,
|
2017-01-04 07:37:13 -05:00
|
|
|
stage: 'test')
|
2017-01-04 06:20:12 -05:00
|
|
|
end
|
2016-05-31 09:31:06 -04:00
|
|
|
|
2017-06-14 14:18:56 -04:00
|
|
|
before do
|
|
|
|
visit_project_pipelines
|
|
|
|
end
|
2017-01-04 06:20:12 -05:00
|
|
|
|
|
|
|
it 'is cancelable' do
|
2017-03-17 13:30:32 -04:00
|
|
|
expect(page).to have_selector('.js-pipelines-cancel-button')
|
2017-01-04 06:20:12 -05:00
|
|
|
end
|
2016-05-31 09:31:06 -04:00
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
it 'has pipeline running' do
|
|
|
|
expect(page).to have_selector('.ci-running')
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when canceling' do
|
2017-06-14 14:18:56 -04:00
|
|
|
before do
|
2018-02-06 22:05:54 -05:00
|
|
|
find('.js-pipelines-cancel-button').click
|
2018-05-11 10:03:27 -04:00
|
|
|
find('.js-modal-primary-action').click
|
2017-06-14 14:18:56 -04:00
|
|
|
end
|
2017-01-04 06:20:12 -05:00
|
|
|
|
2017-01-05 09:14:06 -05:00
|
|
|
it 'indicates that pipeline was canceled' do
|
2017-03-17 13:30:32 -04:00
|
|
|
expect(page).not_to have_selector('.js-pipelines-cancel-button')
|
2017-01-05 09:14:06 -05:00
|
|
|
expect(page).to have_selector('.ci-canceled')
|
|
|
|
end
|
2017-01-04 06:20:12 -05:00
|
|
|
end
|
2016-05-31 09:31:06 -04:00
|
|
|
end
|
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
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
|
2017-03-17 13:30:32 -04:00
|
|
|
expect(page).not_to have_selector('.js-pipelines-retry-button')
|
2017-01-04 06:20:12 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'has failed pipeline' do
|
|
|
|
expect(page).to have_selector('.ci-failed')
|
|
|
|
end
|
2016-05-31 09:31:06 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
context 'downloadable pipelines' do
|
|
|
|
context 'with artifacts' do
|
|
|
|
let!(:with_artifacts) do
|
|
|
|
create(:ci_build, :artifacts, :success,
|
|
|
|
pipeline: pipeline,
|
|
|
|
name: 'rspec tests',
|
|
|
|
stage: 'test')
|
|
|
|
end
|
|
|
|
|
2017-06-14 14:18:56 -04:00
|
|
|
before do
|
|
|
|
visit_project_pipelines
|
|
|
|
end
|
2017-01-04 06:20:12 -05:00
|
|
|
|
|
|
|
it 'has artifats' do
|
|
|
|
expect(page).to have_selector('.build-artifacts')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'has artifacts download dropdown' do
|
|
|
|
find('.js-pipeline-dropdown-download').click
|
|
|
|
|
|
|
|
expect(page).to have_link(with_artifacts.name)
|
|
|
|
end
|
2017-02-15 19:47:26 -05:00
|
|
|
|
|
|
|
it 'has download attribute on download links' do
|
|
|
|
find('.js-pipeline-dropdown-download').click
|
|
|
|
expect(page).to have_selector('a', text: 'Download')
|
|
|
|
page.all('.build-artifacts a', text: 'Download').each do |link|
|
|
|
|
expect(link[:download]).to eq ''
|
|
|
|
end
|
|
|
|
end
|
2016-11-17 13:55:10 -05:00
|
|
|
end
|
2016-05-13 18:11:57 -04:00
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
context 'with artifacts expired' do
|
|
|
|
let!(:with_artifacts_expired) do
|
2017-11-02 14:38:25 -04:00
|
|
|
create(:ci_build, :expired, :success,
|
2017-01-04 06:20:12 -05:00
|
|
|
pipeline: pipeline,
|
|
|
|
name: 'rspec',
|
|
|
|
stage: 'test')
|
|
|
|
end
|
2016-12-21 06:09:06 -05:00
|
|
|
|
2017-06-14 14:18:56 -04:00
|
|
|
before do
|
|
|
|
visit_project_pipelines
|
|
|
|
end
|
2017-01-04 06:20:12 -05:00
|
|
|
|
|
|
|
it { expect(page).not_to have_selector('.build-artifacts') }
|
2016-11-15 15:45:05 -05:00
|
|
|
end
|
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
context 'without artifacts' do
|
|
|
|
let!(:without_artifacts) do
|
|
|
|
create(:ci_build, :success,
|
|
|
|
pipeline: pipeline,
|
|
|
|
name: 'rspec',
|
|
|
|
stage: 'test')
|
|
|
|
end
|
|
|
|
|
2017-06-14 14:18:56 -04:00
|
|
|
before do
|
|
|
|
visit_project_pipelines
|
|
|
|
end
|
2016-12-21 06:09:06 -05:00
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
it { expect(page).not_to have_selector('.build-artifacts') }
|
2016-11-15 15:45:05 -05:00
|
|
|
end
|
2018-03-13 11:52:34 -04:00
|
|
|
|
|
|
|
context 'with trace artifact' do
|
|
|
|
before do
|
|
|
|
create(:ci_build, :success, :trace_artifact, pipeline: pipeline)
|
|
|
|
|
|
|
|
visit_project_pipelines
|
|
|
|
end
|
|
|
|
|
2018-03-14 03:54:15 -04:00
|
|
|
it 'does not show trace artifact as artifacts' do
|
2018-03-13 11:52:34 -04:00
|
|
|
expect(page).not_to have_selector('.build-artifacts')
|
|
|
|
end
|
|
|
|
end
|
2016-05-13 14:17:15 -04:00
|
|
|
end
|
2016-05-13 09:36:44 -04:00
|
|
|
|
2017-01-05 16:18:44 -05:00
|
|
|
context 'mini pipeline graph' do
|
2017-01-04 06:20:12 -05:00
|
|
|
let!(:build) do
|
2017-01-05 16:25:29 -05:00
|
|
|
create(:ci_build, :pending, pipeline: pipeline,
|
|
|
|
stage: 'build',
|
|
|
|
name: 'build')
|
2016-11-17 13:55:10 -05:00
|
|
|
end
|
2016-07-28 01:09:40 -04:00
|
|
|
|
2017-06-14 14:18:56 -04:00
|
|
|
before do
|
|
|
|
visit_project_pipelines
|
|
|
|
end
|
2016-07-28 01:09:40 -04:00
|
|
|
|
2017-01-05 16:18:44 -05:00
|
|
|
it 'should render a mini pipeline graph' do
|
|
|
|
expect(page).to have_selector('.js-mini-pipeline-graph')
|
|
|
|
expect(page).to have_selector('.js-builds-dropdown-button')
|
|
|
|
end
|
|
|
|
|
2017-01-05 16:25:29 -05:00
|
|
|
context 'when clicking a stage badge' do
|
2017-01-04 06:20:12 -05:00
|
|
|
it 'should open a dropdown' do
|
2017-06-17 02:08:27 -04:00
|
|
|
find('.js-builds-dropdown-button').click
|
2016-07-28 01:09:40 -04:00
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
expect(page).to have_link build.name
|
|
|
|
end
|
2016-05-13 14:17:15 -04:00
|
|
|
|
2017-01-05 16:25:29 -05:00
|
|
|
it 'should be possible to cancel pending build' do
|
2017-06-17 02:08:27 -04:00
|
|
|
find('.js-builds-dropdown-button').click
|
2018-05-03 16:46:35 -04:00
|
|
|
find('.js-ci-action').click
|
|
|
|
wait_for_requests
|
2016-07-28 01:09:40 -04:00
|
|
|
|
2017-01-05 16:25:29 -05:00
|
|
|
expect(build.reload).to be_canceled
|
2017-01-04 06:20:12 -05:00
|
|
|
end
|
2018-04-05 17:04:42 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
context 'for a failed pipeline' do
|
|
|
|
let!(:build) do
|
|
|
|
create(:ci_build, :failed, pipeline: pipeline,
|
|
|
|
stage: 'build',
|
|
|
|
name: 'build')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'should display the failure reason' do
|
|
|
|
find('.js-builds-dropdown-button').click
|
|
|
|
|
|
|
|
within('.js-builds-dropdown-list') do
|
|
|
|
build_element = page.find('.mini-pipeline-graph-dropdown-item')
|
2018-08-09 07:05:13 -04:00
|
|
|
expect(build_element['data-original-title']).to eq('build - failed - (unknown failure)')
|
2018-04-05 17:04:42 -04:00
|
|
|
end
|
|
|
|
end
|
2017-01-04 06:20:12 -05:00
|
|
|
end
|
2016-05-13 14:17:15 -04:00
|
|
|
end
|
2017-02-15 11:36:13 -05:00
|
|
|
|
|
|
|
context 'with pagination' do
|
|
|
|
before do
|
2017-02-16 15:17:12 -05:00
|
|
|
allow(Ci::Pipeline).to receive(:default_per_page).and_return(1)
|
2019-01-16 07:09:29 -05:00
|
|
|
create(:ci_empty_pipeline, project: project)
|
2017-02-15 11:36:13 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'should render pagination' do
|
2017-07-06 12:20:50 -04:00
|
|
|
visit project_pipelines_path(project)
|
2017-05-17 14:25:13 -04:00
|
|
|
wait_for_requests
|
2017-02-15 11:36:13 -05:00
|
|
|
|
2017-02-16 15:17:12 -05:00
|
|
|
expect(page).to have_selector('.gl-pagination')
|
2017-02-15 11:36:13 -05:00
|
|
|
end
|
|
|
|
|
2017-02-16 15:17:12 -05:00
|
|
|
it 'should render second page of pipelines' do
|
2017-07-06 12:20:50 -04:00
|
|
|
visit project_pipelines_path(project, page: '2')
|
2017-05-17 14:25:13 -04:00
|
|
|
wait_for_requests
|
2017-02-15 11:36:13 -05:00
|
|
|
|
2017-02-16 15:17:12 -05:00
|
|
|
expect(page).to have_selector('.gl-pagination .page', count: 2)
|
2017-02-15 11:36:13 -05:00
|
|
|
end
|
2017-11-14 10:35:29 -05:00
|
|
|
|
|
|
|
it 'should show updated content' do
|
|
|
|
visit project_pipelines_path(project)
|
|
|
|
wait_for_requests
|
2019-02-27 05:16:46 -05:00
|
|
|
page.find('.js-next-button .page-link').click
|
2017-11-14 10:35:29 -05:00
|
|
|
|
|
|
|
expect(page).to have_selector('.gl-pagination .page', count: 2)
|
|
|
|
end
|
2017-02-15 11:36:13 -05:00
|
|
|
end
|
2016-05-13 14:17:15 -04:00
|
|
|
end
|
2016-12-20 08:22:31 -05:00
|
|
|
|
2017-05-07 08:09:21 -04:00
|
|
|
describe 'GET /:project/pipelines/show' do
|
2017-07-24 18:51:14 -04:00
|
|
|
let(:project) { create(:project, :repository) }
|
2017-05-07 08:09:21 -04:00
|
|
|
|
|
|
|
let(:pipeline) do
|
|
|
|
create(:ci_empty_pipeline,
|
|
|
|
project: project,
|
|
|
|
sha: project.commit.id,
|
|
|
|
user: user)
|
|
|
|
end
|
|
|
|
|
|
|
|
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, stage: 'external', name: 'jenkins', stage_idx: 3)
|
|
|
|
|
2017-07-06 12:20:50 -04:00
|
|
|
visit project_pipeline_path(project, pipeline)
|
2017-05-17 14:25:13 -04:00
|
|
|
wait_for_requests
|
2017-05-07 08:09:21 -04:00
|
|
|
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[alt$="#{pipeline.user.name}'s avatar"]))
|
|
|
|
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')
|
2017-05-07 12:26:39 -04:00
|
|
|
expect(page).to have_text('rspec')
|
2017-05-07 08:09:21 -04:00
|
|
|
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
|
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
describe 'POST /:project/pipelines' do
|
2017-07-24 18:51:14 -04:00
|
|
|
let(:project) { create(:project, :repository) }
|
2017-01-04 06:20:12 -05:00
|
|
|
|
|
|
|
before do
|
2017-07-06 12:20:50 -04:00
|
|
|
visit new_project_pipeline_path(project)
|
2016-12-20 08:22:31 -05:00
|
|
|
end
|
|
|
|
|
2017-10-03 04:35:01 -04:00
|
|
|
context 'for valid commit', :js do
|
2017-01-03 11:31:43 -05:00
|
|
|
before do
|
|
|
|
click_button project.default_branch
|
|
|
|
|
|
|
|
page.within '.dropdown-menu' do
|
|
|
|
click_link 'master'
|
|
|
|
end
|
|
|
|
end
|
2016-12-20 08:22:31 -05:00
|
|
|
|
2017-01-05 16:18:44 -05:00
|
|
|
context 'with gitlab-ci.yml' do
|
2017-06-14 14:18:56 -04:00
|
|
|
before do
|
|
|
|
stub_ci_pipeline_to_return_yaml_file
|
|
|
|
end
|
2016-12-20 08:22:31 -05:00
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
it 'creates a new pipeline' do
|
2018-05-02 10:06:47 -04:00
|
|
|
expect { click_on 'Create pipeline' }
|
2017-01-04 06:20:12 -05:00
|
|
|
.to change { Ci::Pipeline.count }.by(1)
|
2017-05-24 09:13:51 -04:00
|
|
|
|
|
|
|
expect(Ci::Pipeline.last).to be_web
|
2017-01-04 06:20:12 -05:00
|
|
|
end
|
2018-05-04 07:31:20 -04:00
|
|
|
|
|
|
|
context 'when variables are specified' do
|
|
|
|
it 'creates a new pipeline with variables' do
|
|
|
|
page.within '.ci-variable-row-body' do
|
|
|
|
fill_in "Input variable key", with: "key_name"
|
|
|
|
fill_in "Input variable value", with: "value"
|
|
|
|
end
|
|
|
|
|
|
|
|
expect { click_on 'Create pipeline' }
|
|
|
|
.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
|
2016-12-20 08:22:31 -05:00
|
|
|
end
|
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
context 'without gitlab-ci.yml' do
|
2017-06-14 14:18:56 -04:00
|
|
|
before do
|
2018-05-02 10:06:47 -04:00
|
|
|
click_on 'Create pipeline'
|
2017-06-14 14:18:56 -04:00
|
|
|
end
|
2016-12-20 08:22:31 -05:00
|
|
|
|
2017-01-05 16:18:44 -05:00
|
|
|
it { expect(page).to have_content('Missing .gitlab-ci.yml file') }
|
2017-11-23 11:34:52 -05:00
|
|
|
it 'creates a pipeline after first request failed and a valid gitlab-ci.yml file is available when trying again' do
|
|
|
|
click_button project.default_branch
|
|
|
|
|
|
|
|
stub_ci_pipeline_to_return_yaml_file
|
|
|
|
|
|
|
|
page.within '.dropdown-menu' do
|
|
|
|
click_link 'master'
|
|
|
|
end
|
|
|
|
|
2018-05-02 10:06:47 -04:00
|
|
|
expect { click_on 'Create pipeline' }
|
2017-11-23 11:34:52 -05:00
|
|
|
.to change { Ci::Pipeline.count }.by(1)
|
|
|
|
end
|
2016-12-20 08:22:31 -05:00
|
|
|
end
|
|
|
|
end
|
2016-11-17 13:55:10 -05:00
|
|
|
end
|
2016-05-13 14:17:15 -04:00
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
describe 'Create pipelines' do
|
2017-07-24 18:51:14 -04:00
|
|
|
let(:project) { create(:project, :repository) }
|
2016-05-18 14:02:10 -04:00
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
before do
|
2017-07-06 12:20:50 -04:00
|
|
|
visit new_project_pipeline_path(project)
|
2017-01-04 06:20:12 -05:00
|
|
|
end
|
2016-05-18 14:02:10 -04:00
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
describe 'new pipeline page' do
|
|
|
|
it 'has field to add a new pipeline' do
|
2017-01-03 11:31:43 -05:00
|
|
|
expect(page).to have_selector('.js-branch-select')
|
|
|
|
expect(find('.js-branch-select')).to have_content project.default_branch
|
2018-05-02 10:26:01 -04:00
|
|
|
expect(page).to have_content('Create for')
|
2016-11-17 13:55:10 -05:00
|
|
|
end
|
2016-05-13 18:11:57 -04:00
|
|
|
end
|
2016-05-13 14:17:15 -04:00
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
describe 'find pipelines' do
|
2017-10-03 04:35:01 -04:00
|
|
|
it 'shows filtered pipelines', :js do
|
2017-01-03 11:31:43 -05:00
|
|
|
click_button project.default_branch
|
2016-05-18 14:02:10 -04:00
|
|
|
|
2017-01-03 11:31:43 -05:00
|
|
|
page.within '.dropdown-menu' do
|
|
|
|
find('.dropdown-input-field').native.send_keys('fix')
|
|
|
|
|
|
|
|
page.within '.dropdown-content' do
|
|
|
|
expect(page).to have_content('fix')
|
|
|
|
end
|
2017-01-04 06:20:12 -05:00
|
|
|
end
|
|
|
|
end
|
2016-05-13 14:17:15 -04:00
|
|
|
end
|
|
|
|
end
|
2018-01-04 17:33:17 -05:00
|
|
|
|
|
|
|
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')
|
2018-07-11 10:36:08 -04:00
|
|
|
project.add_maintainer(user)
|
2018-01-04 17:33:17 -05:00
|
|
|
visit project_pipelines_path(project)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'has a clear caches button' do
|
2018-03-07 07:01:36 -05:00
|
|
|
expect(page).to have_button 'Clear Runner Caches'
|
2018-01-04 17:33:17 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
describe 'user clicks the button' do
|
|
|
|
context 'when project already has jobs_cache_index' do
|
|
|
|
before do
|
2018-07-02 06:43:06 -04:00
|
|
|
project.update(jobs_cache_index: 1)
|
2018-01-04 17:33:17 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'increments jobs_cache_index' do
|
2018-03-07 07:01:36 -05:00
|
|
|
click_button 'Clear Runner Caches'
|
|
|
|
wait_for_requests
|
2018-01-04 17:33:17 -05:00
|
|
|
expect(page.find('.flash-notice')).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
|
2018-03-07 07:01:36 -05:00
|
|
|
click_button 'Clear Runner Caches'
|
|
|
|
wait_for_requests
|
2018-01-04 17:33:17 -05:00
|
|
|
expect(page.find('.flash-notice')).to have_content 'Project cache successfully reset.'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2018-03-01 12:40:00 -05:00
|
|
|
|
|
|
|
describe 'Empty State' do
|
|
|
|
let(:project) { create(:project, :repository) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
visit project_pipelines_path(project)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'renders empty state' do
|
|
|
|
expect(page).to have_content 'Build with confidence'
|
|
|
|
end
|
|
|
|
end
|
2016-05-13 09:36:44 -04:00
|
|
|
end
|
2016-07-07 05:02:57 -04:00
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
context 'when user is not logged in' do
|
2016-07-07 05:02:57 -04:00
|
|
|
before do
|
2018-09-18 15:20:43 -04:00
|
|
|
project.update!(auto_devops_attributes: { enabled: false })
|
2017-07-06 12:20:50 -04:00
|
|
|
visit project_pipelines_path(project)
|
2016-07-07 05:02:57 -04:00
|
|
|
end
|
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
context 'when project is public' do
|
2017-07-24 18:51:14 -04:00
|
|
|
let(:project) { create(:project, :public, :repository) }
|
2017-01-04 06:20:12 -05:00
|
|
|
|
2018-03-01 12:40:00 -05:00
|
|
|
context 'without pipelines' do
|
|
|
|
it { expect(page).to have_content 'This project is not currently set up to run pipelines.' }
|
|
|
|
end
|
2016-07-07 05:02:57 -04:00
|
|
|
end
|
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
context 'when project is private' do
|
2017-07-24 18:51:14 -04:00
|
|
|
let(:project) { create(:project, :private, :repository) }
|
2016-07-07 05:02:57 -04:00
|
|
|
|
2017-01-04 06:20:12 -05:00
|
|
|
it { expect(page).to have_content 'You need to sign in' }
|
2016-07-07 05:02:57 -04:00
|
|
|
end
|
|
|
|
end
|
2017-01-04 06:04:48 -05:00
|
|
|
|
|
|
|
def visit_project_pipelines(**query)
|
2017-07-06 12:20:50 -04:00
|
|
|
visit project_pipelines_path(project, query)
|
2017-05-17 14:25:13 -04:00
|
|
|
wait_for_requests
|
2017-01-04 06:04:48 -05:00
|
|
|
end
|
2016-05-13 09:36:44 -04:00
|
|
|
end
|