2015-08-25 21:42:46 -04:00
|
|
|
require 'spec_helper'
|
|
|
|
|
2015-12-09 04:50:51 -05:00
|
|
|
describe Ci::Build, models: true do
|
2016-05-30 06:00:42 -04:00
|
|
|
let(:project) { create(:project) }
|
2016-06-21 08:26:57 -04:00
|
|
|
|
|
|
|
let(:pipeline) do
|
|
|
|
create(:ci_pipeline, project: project,
|
2016-07-18 08:10:50 -04:00
|
|
|
sha: project.commit.id,
|
2016-07-19 13:29:44 -04:00
|
|
|
ref: project.default_branch,
|
2016-07-18 09:11:53 -04:00
|
|
|
status: 'success')
|
2016-06-21 08:26:57 -04:00
|
|
|
end
|
|
|
|
|
2016-06-03 10:22:26 -04:00
|
|
|
let(:build) { create(:ci_build, pipeline: pipeline) }
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2015-10-05 06:02:26 -04:00
|
|
|
it { is_expected.to validate_presence_of :ref }
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2015-09-10 09:52:52 -04:00
|
|
|
it { is_expected.to respond_to :trace_html }
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2016-03-14 08:33:26 -04:00
|
|
|
describe '#first_pending' do
|
2016-06-03 10:22:26 -04:00
|
|
|
let!(:first) { create(:ci_build, pipeline: pipeline, status: 'pending', created_at: Date.yesterday) }
|
|
|
|
let!(:second) { create(:ci_build, pipeline: pipeline, status: 'pending') }
|
2015-09-09 08:17:16 -04:00
|
|
|
subject { Ci::Build.first_pending }
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2015-09-10 09:52:52 -04:00
|
|
|
it { is_expected.to be_a(Ci::Build) }
|
|
|
|
it('returns with the first pending build') { is_expected.to eq(first) }
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
|
2016-03-14 08:33:26 -04:00
|
|
|
describe '#create_from' do
|
2015-08-25 21:42:46 -04:00
|
|
|
before do
|
|
|
|
build.status = 'success'
|
|
|
|
build.save
|
|
|
|
end
|
2015-09-09 08:17:16 -04:00
|
|
|
let(:create_from_build) { Ci::Build.create_from build }
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2016-08-01 11:00:44 -04:00
|
|
|
it 'exists a pending task' do
|
2015-09-09 08:17:16 -04:00
|
|
|
expect(Ci::Build.pending.count(:all)).to eq 0
|
2015-08-25 21:42:46 -04:00
|
|
|
create_from_build
|
2015-09-09 08:17:16 -04:00
|
|
|
expect(Ci::Build.pending.count(:all)).to be > 0
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-10-03 06:22:51 -04:00
|
|
|
describe '#failed_but_allowed?' do
|
|
|
|
subject { build.failed_but_allowed? }
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2016-08-12 03:19:28 -04:00
|
|
|
context 'when build is not allowed to fail' do
|
2016-06-07 03:00:41 -04:00
|
|
|
before do
|
|
|
|
build.allow_failure = false
|
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
|
|
|
context 'and build.status is success' do
|
2016-06-07 03:00:41 -04:00
|
|
|
before do
|
|
|
|
build.status = 'success'
|
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2015-09-10 09:52:52 -04:00
|
|
|
it { is_expected.to be_falsey }
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
context 'and build.status is failed' do
|
2016-06-07 03:00:41 -04:00
|
|
|
before do
|
|
|
|
build.status = 'failed'
|
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2015-09-10 09:52:52 -04:00
|
|
|
it { is_expected.to be_falsey }
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-08-12 03:19:28 -04:00
|
|
|
context 'when build is allowed to fail' do
|
2016-06-07 03:00:41 -04:00
|
|
|
before do
|
|
|
|
build.allow_failure = true
|
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
|
|
|
context 'and build.status is success' do
|
2016-06-07 03:00:41 -04:00
|
|
|
before do
|
|
|
|
build.status = 'success'
|
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2015-09-10 09:52:52 -04:00
|
|
|
it { is_expected.to be_falsey }
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
context 'and build.status is failed' do
|
2016-06-07 03:00:41 -04:00
|
|
|
before do
|
|
|
|
build.status = 'failed'
|
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2015-09-10 09:52:52 -04:00
|
|
|
it { is_expected.to be_truthy }
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-12-08 11:21:16 -05:00
|
|
|
describe '#persisted_environment' do
|
|
|
|
before do
|
|
|
|
@environment = create(:environment, project: project, name: "foo-#{project.default_branch}")
|
|
|
|
end
|
|
|
|
|
|
|
|
subject { build.persisted_environment }
|
|
|
|
|
|
|
|
context 'referenced literally' do
|
|
|
|
let(:build) { create(:ci_build, pipeline: pipeline, environment: "foo-#{project.default_branch}") }
|
|
|
|
|
|
|
|
it { is_expected.to eq(@environment) }
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'referenced with a variable' do
|
|
|
|
let(:build) { create(:ci_build, pipeline: pipeline, environment: "foo-$CI_BUILD_REF_NAME") }
|
|
|
|
|
|
|
|
it { is_expected.to eq(@environment) }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-03-14 08:33:26 -04:00
|
|
|
describe '#trace' do
|
2016-09-19 06:38:03 -04:00
|
|
|
it { expect(build.trace).to be_nil }
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2016-08-12 03:19:28 -04:00
|
|
|
context 'when build.trace contains text' do
|
2015-08-25 21:42:46 -04:00
|
|
|
let(:text) { 'example output' }
|
2016-06-07 03:00:41 -04:00
|
|
|
before do
|
|
|
|
build.trace = text
|
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2016-09-19 06:38:03 -04:00
|
|
|
it { expect(build.trace).to eq(text) }
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when build.trace hides runners token' do
|
|
|
|
let(:token) { 'my_secret_token' }
|
|
|
|
|
|
|
|
before do
|
|
|
|
build.update(trace: token)
|
|
|
|
build.project.update(runners_token: token)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { expect(build.trace).not_to include(token) }
|
|
|
|
it { expect(build.raw_trace).to include(token) }
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when build.trace hides build token' do
|
|
|
|
let(:token) { 'my_secret_token' }
|
|
|
|
|
|
|
|
before do
|
|
|
|
build.update(trace: token)
|
|
|
|
build.update(token: token)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { expect(build.trace).not_to include(token) }
|
|
|
|
it { expect(build.raw_trace).to include(token) }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#raw_trace' do
|
|
|
|
subject { build.raw_trace }
|
|
|
|
|
|
|
|
context 'when build.trace hides runners token' do
|
|
|
|
let(:token) { 'my_secret_token' }
|
|
|
|
|
|
|
|
before do
|
|
|
|
build.project.update(runners_token: token)
|
|
|
|
build.update(trace: token)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.not_to include(token) }
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when build.trace hides build token' do
|
|
|
|
let(:token) { 'my_secret_token' }
|
|
|
|
|
|
|
|
before do
|
|
|
|
build.update(token: token)
|
|
|
|
build.update(trace: token)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.not_to include(token) }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context '#append_trace' do
|
|
|
|
subject { build.trace_html }
|
|
|
|
|
|
|
|
context 'when build.trace hides runners token' do
|
|
|
|
let(:token) { 'my_secret_token' }
|
|
|
|
|
|
|
|
before do
|
|
|
|
build.project.update(runners_token: token)
|
|
|
|
build.append_trace(token, 0)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.not_to include(token) }
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
2015-10-01 05:43:06 -04:00
|
|
|
|
2016-09-19 06:38:03 -04:00
|
|
|
context 'when build.trace hides build token' do
|
2015-10-01 05:43:06 -04:00
|
|
|
let(:token) { 'my_secret_token' }
|
|
|
|
|
|
|
|
before do
|
2016-09-19 06:38:03 -04:00
|
|
|
build.update(token: token)
|
|
|
|
build.append_trace(token, 0)
|
2015-10-01 05:43:06 -04:00
|
|
|
end
|
|
|
|
|
2016-05-23 19:37:59 -04:00
|
|
|
it { is_expected.not_to include(token) }
|
2015-10-01 05:43:06 -04:00
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
|
2015-12-04 06:55:23 -05:00
|
|
|
# TODO: build timeout
|
|
|
|
# describe :timeout do
|
|
|
|
# subject { build.timeout }
|
|
|
|
#
|
2016-06-03 10:22:26 -04:00
|
|
|
# it { is_expected.to eq(pipeline.project.timeout) }
|
2015-12-04 06:55:23 -05:00
|
|
|
# end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2016-03-14 08:33:26 -04:00
|
|
|
describe '#options' do
|
2015-09-14 07:37:18 -04:00
|
|
|
let(:options) do
|
2015-08-25 21:42:46 -04:00
|
|
|
{
|
2015-09-14 07:37:18 -04:00
|
|
|
image: "ruby:2.1",
|
|
|
|
services: [
|
2015-08-25 21:42:46 -04:00
|
|
|
"postgres"
|
|
|
|
]
|
|
|
|
}
|
2015-09-14 07:37:18 -04:00
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
|
|
|
subject { build.options }
|
2015-09-10 09:52:52 -04:00
|
|
|
it { is_expected.to eq(options) }
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
|
2015-12-04 06:55:23 -05:00
|
|
|
# TODO: allow_git_fetch
|
|
|
|
# describe :allow_git_fetch do
|
|
|
|
# subject { build.allow_git_fetch }
|
|
|
|
#
|
|
|
|
# it { is_expected.to eq(project.allow_git_fetch) }
|
|
|
|
# end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2016-03-14 08:33:26 -04:00
|
|
|
describe '#project' do
|
2015-08-25 21:42:46 -04:00
|
|
|
subject { build.project }
|
|
|
|
|
2016-06-03 10:22:26 -04:00
|
|
|
it { is_expected.to eq(pipeline.project) }
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
|
2016-03-14 08:33:26 -04:00
|
|
|
describe '#project_id' do
|
2015-08-25 21:42:46 -04:00
|
|
|
subject { build.project_id }
|
|
|
|
|
2016-06-03 10:22:26 -04:00
|
|
|
it { is_expected.to eq(pipeline.project_id) }
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
|
2016-03-14 08:33:26 -04:00
|
|
|
describe '#project_name' do
|
2015-08-25 21:42:46 -04:00
|
|
|
subject { build.project_name }
|
|
|
|
|
2015-09-10 09:52:52 -04:00
|
|
|
it { is_expected.to eq(project.name) }
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
|
2016-03-14 08:33:26 -04:00
|
|
|
describe '#extract_coverage' do
|
2015-08-25 21:42:46 -04:00
|
|
|
context 'valid content & regex' do
|
|
|
|
subject { build.extract_coverage('Coverage 1033 / 1051 LOC (98.29%) covered', '\(\d+.\d+\%\) covered') }
|
|
|
|
|
2015-09-10 09:52:52 -04:00
|
|
|
it { is_expected.to eq(98.29) }
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
context 'valid content & bad regex' do
|
|
|
|
subject { build.extract_coverage('Coverage 1033 / 1051 LOC (98.29%) covered', 'very covered') }
|
|
|
|
|
2015-09-10 09:52:52 -04:00
|
|
|
it { is_expected.to be_nil }
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
context 'no coverage content & regex' do
|
|
|
|
subject { build.extract_coverage('No coverage for today :sad:', '\(\d+.\d+\%\) covered') }
|
|
|
|
|
2015-09-10 09:52:52 -04:00
|
|
|
it { is_expected.to be_nil }
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
context 'multiple results in content & regex' do
|
|
|
|
subject { build.extract_coverage(' (98.39%) covered. (98.29%) covered', '\(\d+.\d+\%\) covered') }
|
|
|
|
|
2015-09-10 09:52:52 -04:00
|
|
|
it { is_expected.to eq(98.29) }
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
2015-12-20 13:18:14 -05:00
|
|
|
|
|
|
|
context 'using a regex capture' do
|
|
|
|
subject { build.extract_coverage('TOTAL 9926 3489 65%', 'TOTAL\s+\d+\s+\d+\s+(\d{1,3}\%)') }
|
|
|
|
|
|
|
|
it { is_expected.to eq(65) }
|
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
|
2016-12-13 14:21:30 -05:00
|
|
|
describe '#ref_slug' do
|
|
|
|
{
|
|
|
|
'master' => 'master',
|
|
|
|
'1-foo' => '1-foo',
|
|
|
|
'fix/1-foo' => 'fix-1-foo',
|
|
|
|
'fix-1-foo' => 'fix-1-foo',
|
|
|
|
'a' * 63 => 'a' * 63,
|
|
|
|
'a' * 64 => 'a' * 63,
|
|
|
|
'FOO' => 'foo',
|
|
|
|
}.each do |ref, slug|
|
|
|
|
it "transforms #{ref} to #{slug}" do
|
|
|
|
build.ref = ref
|
|
|
|
|
|
|
|
expect(build.ref_slug).to eq(slug)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-03-14 08:33:26 -04:00
|
|
|
describe '#variables' do
|
2016-07-19 18:08:54 -04:00
|
|
|
let(:container_registry_enabled) { false }
|
2016-07-19 07:23:14 -04:00
|
|
|
let(:predefined_variables) do
|
|
|
|
[
|
2016-07-19 18:08:54 -04:00
|
|
|
{ key: 'CI', value: 'true', public: true },
|
|
|
|
{ key: 'GITLAB_CI', value: 'true', public: true },
|
|
|
|
{ key: 'CI_BUILD_ID', value: build.id.to_s, public: true },
|
|
|
|
{ key: 'CI_BUILD_TOKEN', value: build.token, public: false },
|
|
|
|
{ key: 'CI_BUILD_REF', value: build.sha, public: true },
|
|
|
|
{ key: 'CI_BUILD_BEFORE_SHA', value: build.before_sha, public: true },
|
|
|
|
{ key: 'CI_BUILD_REF_NAME', value: 'master', public: true },
|
2016-12-13 14:21:30 -05:00
|
|
|
{ key: 'CI_BUILD_REF_SLUG', value: 'master', public: true },
|
2016-07-19 18:08:54 -04:00
|
|
|
{ key: 'CI_BUILD_NAME', value: 'test', public: true },
|
|
|
|
{ key: 'CI_BUILD_STAGE', value: 'test', public: true },
|
2016-07-20 07:17:21 -04:00
|
|
|
{ key: 'CI_SERVER_NAME', value: 'GitLab', public: true },
|
|
|
|
{ key: 'CI_SERVER_VERSION', value: Gitlab::VERSION, public: true },
|
|
|
|
{ key: 'CI_SERVER_REVISION', value: Gitlab::REVISION, public: true },
|
2016-07-19 18:08:54 -04:00
|
|
|
{ key: 'CI_PROJECT_ID', value: project.id.to_s, public: true },
|
|
|
|
{ key: 'CI_PROJECT_NAME', value: project.path, public: true },
|
|
|
|
{ key: 'CI_PROJECT_PATH', value: project.path_with_namespace, public: true },
|
|
|
|
{ key: 'CI_PROJECT_NAMESPACE', value: project.namespace.path, public: true },
|
|
|
|
{ key: 'CI_PROJECT_URL', value: project.web_url, public: true },
|
2016-07-20 07:17:21 -04:00
|
|
|
{ key: 'CI_PIPELINE_ID', value: pipeline.id.to_s, public: true }
|
2016-07-19 07:23:14 -04:00
|
|
|
]
|
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2016-07-19 18:08:54 -04:00
|
|
|
before do
|
|
|
|
stub_container_registry_config(enabled: container_registry_enabled, host_port: 'registry.example.com')
|
|
|
|
end
|
|
|
|
|
2016-07-19 07:23:14 -04:00
|
|
|
subject { build.variables }
|
2015-10-12 11:06:53 -04:00
|
|
|
|
2015-08-25 21:42:46 -04:00
|
|
|
context 'returns variables' do
|
2016-06-07 03:00:41 -04:00
|
|
|
before do
|
2016-07-19 18:08:54 -04:00
|
|
|
build.yaml_variables = []
|
2016-06-07 03:00:41 -04:00
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2016-07-19 18:08:54 -04:00
|
|
|
it { is_expected.to eq(predefined_variables) }
|
|
|
|
end
|
2015-10-12 11:06:53 -04:00
|
|
|
|
2016-09-05 05:42:59 -04:00
|
|
|
context 'when build has user' do
|
|
|
|
let(:user) { create(:user, username: 'starter') }
|
|
|
|
let(:user_variables) do
|
|
|
|
[
|
|
|
|
{ key: 'GITLAB_USER_ID', value: user.id.to_s, public: true },
|
|
|
|
{ key: 'GITLAB_USER_EMAIL', value: user.email, public: true }
|
|
|
|
]
|
|
|
|
end
|
|
|
|
|
|
|
|
before do
|
|
|
|
build.update_attributes(user: user)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { user_variables.each { |v| is_expected.to include(v) } }
|
|
|
|
end
|
|
|
|
|
2016-12-08 11:21:16 -05:00
|
|
|
context 'when build has an environment' do
|
|
|
|
before do
|
|
|
|
build.update(environment: 'production')
|
|
|
|
create(:environment, project: build.project, name: 'production', slug: 'prod-slug')
|
|
|
|
end
|
|
|
|
|
|
|
|
let(:environment_variables) do
|
|
|
|
[
|
|
|
|
{ key: 'CI_ENVIRONMENT_NAME', value: 'production', public: true },
|
|
|
|
{ key: 'CI_ENVIRONMENT_SLUG', value: 'prod-slug', public: true }
|
|
|
|
]
|
|
|
|
end
|
|
|
|
|
|
|
|
it { environment_variables.each { |v| is_expected.to include(v) } }
|
|
|
|
end
|
|
|
|
|
2016-09-05 05:42:59 -04:00
|
|
|
context 'when build started manually' do
|
|
|
|
before do
|
|
|
|
build.update_attributes(when: :manual)
|
|
|
|
end
|
|
|
|
|
|
|
|
let(:manual_variable) do
|
|
|
|
{ key: 'CI_BUILD_MANUAL', value: 'true', public: true }
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to include(manual_variable) }
|
|
|
|
end
|
|
|
|
|
2016-07-19 18:08:54 -04:00
|
|
|
context 'when build is for tag' do
|
|
|
|
let(:tag_variable) do
|
|
|
|
{ key: 'CI_BUILD_TAG', value: 'master', public: true }
|
2015-10-12 11:06:53 -04:00
|
|
|
end
|
|
|
|
|
2016-07-19 18:08:54 -04:00
|
|
|
before do
|
|
|
|
build.update_attributes(tag: true)
|
2015-10-12 11:06:53 -04:00
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2016-07-19 18:08:54 -04:00
|
|
|
it { is_expected.to include(tag_variable) }
|
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2016-07-19 18:08:54 -04:00
|
|
|
context 'when secure variable is defined' do
|
|
|
|
let(:secure_variable) do
|
|
|
|
{ key: 'SECRET_KEY', value: 'secret_value', public: false }
|
2015-09-14 07:37:18 -04:00
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2016-06-07 03:00:41 -04:00
|
|
|
before do
|
2016-07-19 18:08:54 -04:00
|
|
|
build.project.variables << Ci::Variable.new(key: 'SECRET_KEY', value: 'secret_value')
|
2016-06-07 03:00:41 -04:00
|
|
|
end
|
2015-10-12 11:06:53 -04:00
|
|
|
|
2016-07-19 18:08:54 -04:00
|
|
|
it { is_expected.to include(secure_variable) }
|
|
|
|
end
|
2015-10-12 11:06:53 -04:00
|
|
|
|
2016-07-19 18:08:54 -04:00
|
|
|
context 'when build is for triggers' do
|
|
|
|
let(:trigger) { create(:ci_trigger, project: project) }
|
|
|
|
let(:trigger_request) { create(:ci_trigger_request_with_variables, pipeline: pipeline, trigger: trigger) }
|
|
|
|
let(:user_trigger_variable) do
|
2016-07-25 05:21:20 -04:00
|
|
|
{ key: :TRIGGER_KEY_1, value: 'TRIGGER_VALUE_1', public: false }
|
2016-07-19 18:08:54 -04:00
|
|
|
end
|
|
|
|
let(:predefined_trigger_variable) do
|
|
|
|
{ key: 'CI_BUILD_TRIGGERED', value: 'true', public: true }
|
|
|
|
end
|
2015-10-12 11:06:53 -04:00
|
|
|
|
2016-07-19 18:08:54 -04:00
|
|
|
before do
|
|
|
|
build.trigger_request = trigger_request
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
2015-10-12 11:06:53 -04:00
|
|
|
|
2016-07-19 18:08:54 -04:00
|
|
|
it { is_expected.to include(user_trigger_variable) }
|
|
|
|
it { is_expected.to include(predefined_trigger_variable) }
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
2015-10-12 11:06:53 -04:00
|
|
|
|
2016-07-19 18:08:54 -04:00
|
|
|
context 'when yaml_variables are undefined' do
|
2016-07-19 07:23:14 -04:00
|
|
|
before do
|
|
|
|
build.yaml_variables = nil
|
2015-10-12 11:06:53 -04:00
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2016-07-19 07:23:14 -04:00
|
|
|
context 'use from gitlab-ci.yml' do
|
|
|
|
before do
|
|
|
|
stub_ci_pipeline_yaml_file(config)
|
2015-09-14 07:37:18 -04:00
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2016-08-12 03:19:28 -04:00
|
|
|
context 'when config is not found' do
|
2016-07-19 07:23:14 -04:00
|
|
|
let(:config) { nil }
|
|
|
|
|
|
|
|
it { is_expected.to eq(predefined_variables) }
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
|
2016-08-12 03:19:28 -04:00
|
|
|
context 'when config does not have a questioned job' do
|
2016-07-19 07:23:14 -04:00
|
|
|
let(:config) do
|
|
|
|
YAML.dump({
|
2016-07-19 18:08:54 -04:00
|
|
|
test_other: {
|
|
|
|
script: 'Hello World'
|
|
|
|
}
|
|
|
|
})
|
2016-07-19 07:23:14 -04:00
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2016-07-19 07:23:14 -04:00
|
|
|
it { is_expected.to eq(predefined_variables) }
|
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2016-08-12 03:19:28 -04:00
|
|
|
context 'when config has variables' do
|
2016-07-19 07:23:14 -04:00
|
|
|
let(:config) do
|
|
|
|
YAML.dump({
|
2016-07-19 18:08:54 -04:00
|
|
|
test: {
|
|
|
|
script: 'Hello World',
|
|
|
|
variables: {
|
|
|
|
KEY: 'value'
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
2015-09-14 07:37:18 -04:00
|
|
|
end
|
2016-07-19 07:23:14 -04:00
|
|
|
let(:variables) do
|
2016-12-14 09:38:40 -05:00
|
|
|
[{ key: 'KEY', value: 'value', public: true }]
|
2015-10-12 11:06:53 -04:00
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2016-07-19 07:23:14 -04:00
|
|
|
it { is_expected.to eq(predefined_variables + variables) }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2016-07-19 18:08:54 -04:00
|
|
|
context 'when container registry is enabled' do
|
|
|
|
let(:container_registry_enabled) { true }
|
|
|
|
let(:ci_registry) do
|
2016-07-20 04:29:11 -04:00
|
|
|
{ key: 'CI_REGISTRY', value: 'registry.example.com', public: true }
|
2016-07-19 18:08:54 -04:00
|
|
|
end
|
|
|
|
let(:ci_registry_image) do
|
2016-07-20 04:29:11 -04:00
|
|
|
{ key: 'CI_REGISTRY_IMAGE', value: project.container_registry_repository_url, public: true }
|
2016-07-19 18:08:54 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
context 'and is disabled for project' do
|
|
|
|
before do
|
|
|
|
project.update(container_registry_enabled: false)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to include(ci_registry) }
|
|
|
|
it { is_expected.not_to include(ci_registry_image) }
|
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2016-07-19 18:08:54 -04:00
|
|
|
context 'and is enabled for project' do
|
|
|
|
before do
|
|
|
|
project.update(container_registry_enabled: true)
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
2016-07-19 18:08:54 -04:00
|
|
|
|
|
|
|
it { is_expected.to include(ci_registry) }
|
|
|
|
it { is_expected.to include(ci_registry_image) }
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
end
|
2016-07-19 18:08:54 -04:00
|
|
|
|
|
|
|
context 'when runner is assigned to build' do
|
|
|
|
let(:runner) { create(:ci_runner, description: 'description', tag_list: ['docker', 'linux']) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
build.update(runner: runner)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to include({ key: 'CI_RUNNER_ID', value: runner.id.to_s, public: true }) }
|
|
|
|
it { is_expected.to include({ key: 'CI_RUNNER_DESCRIPTION', value: 'description', public: true }) }
|
|
|
|
it { is_expected.to include({ key: 'CI_RUNNER_TAGS', value: 'docker, linux', public: true }) }
|
|
|
|
end
|
|
|
|
|
2016-12-16 07:24:03 -05:00
|
|
|
context 'when build is for a deployment' do
|
|
|
|
let(:deployment_variable) { { key: 'KUBERNETES_TOKEN', value: 'TOKEN', public: false } }
|
|
|
|
|
|
|
|
before do
|
|
|
|
build.environment = 'production'
|
|
|
|
allow(project).to receive(:deployment_variables).and_return([deployment_variable])
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to include(deployment_variable) }
|
|
|
|
end
|
|
|
|
|
2016-07-19 18:08:54 -04:00
|
|
|
context 'returns variables in valid order' do
|
|
|
|
before do
|
|
|
|
allow(build).to receive(:predefined_variables) { ['predefined'] }
|
2016-07-20 07:17:21 -04:00
|
|
|
allow(project).to receive(:predefined_variables) { ['project'] }
|
|
|
|
allow(pipeline).to receive(:predefined_variables) { ['pipeline'] }
|
|
|
|
allow(build).to receive(:yaml_variables) { ['yaml'] }
|
|
|
|
allow(project).to receive(:secret_variables) { ['secret'] }
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
2016-07-19 18:08:54 -04:00
|
|
|
|
2016-07-20 07:17:21 -04:00
|
|
|
it { is_expected.to eq(%w[predefined project pipeline yaml secret]) }
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
end
|
2015-10-05 06:02:26 -04:00
|
|
|
|
2016-05-06 03:17:27 -04:00
|
|
|
describe '#has_tags?' do
|
|
|
|
context 'when build has tags' do
|
|
|
|
subject { create(:ci_build, tag_list: ['tag']) }
|
|
|
|
it { is_expected.to have_tags }
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when build does not have tags' do
|
|
|
|
subject { create(:ci_build, tag_list: []) }
|
2016-05-23 19:37:59 -04:00
|
|
|
it { is_expected.not_to have_tags }
|
2016-05-06 03:17:27 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-03-14 08:33:26 -04:00
|
|
|
describe '#any_runners_online?' do
|
2015-10-12 15:12:31 -04:00
|
|
|
subject { build.any_runners_online? }
|
|
|
|
|
|
|
|
context 'when no runners' do
|
|
|
|
it { is_expected.to be_falsey }
|
|
|
|
end
|
|
|
|
|
2016-08-12 03:19:28 -04:00
|
|
|
context 'when there are runners' do
|
2016-05-30 06:00:42 -04:00
|
|
|
let(:runner) { create(:ci_runner) }
|
2015-10-12 15:12:31 -04:00
|
|
|
|
|
|
|
before do
|
2015-12-10 11:44:06 -05:00
|
|
|
build.project.runners << runner
|
2015-10-12 15:12:31 -04:00
|
|
|
runner.update_attributes(contacted_at: 1.second.ago)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to be_truthy }
|
|
|
|
|
|
|
|
it 'that is inactive' do
|
|
|
|
runner.update_attributes(active: false)
|
|
|
|
is_expected.to be_falsey
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'that is not online' do
|
|
|
|
runner.update_attributes(contacted_at: nil)
|
|
|
|
is_expected.to be_falsey
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'that cannot handle build' do
|
2016-06-08 03:19:49 -04:00
|
|
|
expect_any_instance_of(Ci::Runner).to receive(:can_pick?).and_return(false)
|
2015-10-12 15:12:31 -04:00
|
|
|
is_expected.to be_falsey
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-03-14 08:33:26 -04:00
|
|
|
describe '#stuck?' do
|
2016-03-09 10:24:02 -05:00
|
|
|
subject { build.stuck? }
|
2015-10-12 15:12:31 -04:00
|
|
|
|
2016-08-16 04:00:13 -04:00
|
|
|
context "when commit_status.status is pending" do
|
|
|
|
before do
|
|
|
|
build.status = 'pending'
|
|
|
|
end
|
2015-10-12 15:12:31 -04:00
|
|
|
|
2016-08-16 04:00:13 -04:00
|
|
|
it { is_expected.to be_truthy }
|
2015-10-12 15:12:31 -04:00
|
|
|
|
2016-08-16 04:00:13 -04:00
|
|
|
context "and there are specific runner" do
|
|
|
|
let(:runner) { create(:ci_runner, contacted_at: 1.second.ago) }
|
2015-10-12 15:12:31 -04:00
|
|
|
|
2016-08-16 04:00:13 -04:00
|
|
|
before do
|
|
|
|
build.project.runners << runner
|
|
|
|
runner.save
|
2015-10-12 15:12:31 -04:00
|
|
|
end
|
2016-08-16 04:00:13 -04:00
|
|
|
|
|
|
|
it { is_expected.to be_falsey }
|
2015-10-12 15:12:31 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-08-12 03:19:28 -04:00
|
|
|
%w[success failed canceled running].each do |state|
|
|
|
|
context "when commit_status.status is #{state}" do
|
2016-06-07 03:00:41 -04:00
|
|
|
before do
|
|
|
|
build.status = state
|
|
|
|
end
|
2015-10-12 15:12:31 -04:00
|
|
|
|
|
|
|
it { is_expected.to be_falsey }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2015-10-12 17:47:32 -04:00
|
|
|
|
2016-03-14 08:33:26 -04:00
|
|
|
describe '#artifacts?' do
|
2015-12-28 05:43:15 -05:00
|
|
|
subject { build.artifacts? }
|
|
|
|
|
|
|
|
context 'artifacts archive does not exist' do
|
2016-06-07 03:00:41 -04:00
|
|
|
before do
|
|
|
|
build.update_attributes(artifacts_file: nil)
|
|
|
|
end
|
|
|
|
|
2015-12-28 05:43:15 -05:00
|
|
|
it { is_expected.to be_falsy }
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'artifacts archive exists' do
|
2016-02-01 05:31:33 -05:00
|
|
|
let(:build) { create(:ci_build, :artifacts) }
|
2015-12-28 05:43:15 -05:00
|
|
|
it { is_expected.to be_truthy }
|
2016-06-10 10:20:11 -04:00
|
|
|
|
|
|
|
context 'is expired' do
|
|
|
|
before { build.update(artifacts_expire_at: Time.now - 7.days) }
|
|
|
|
it { is_expected.to be_falsy }
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'is not expired' do
|
|
|
|
before { build.update(artifacts_expire_at: Time.now + 7.days) }
|
|
|
|
it { is_expected.to be_truthy }
|
|
|
|
end
|
2015-12-28 05:43:15 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-06-10 10:20:11 -04:00
|
|
|
describe '#artifacts_expired?' do
|
|
|
|
subject { build.artifacts_expired? }
|
|
|
|
|
|
|
|
context 'is expired' do
|
|
|
|
before { build.update(artifacts_expire_at: Time.now - 7.days) }
|
2016-06-14 05:38:34 -04:00
|
|
|
|
|
|
|
it { is_expected.to be_truthy }
|
2016-06-10 10:20:11 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
context 'is not expired' do
|
|
|
|
before { build.update(artifacts_expire_at: Time.now + 7.days) }
|
2016-06-14 05:38:34 -04:00
|
|
|
|
|
|
|
it { is_expected.to be_falsey }
|
2016-06-10 10:20:11 -04:00
|
|
|
end
|
|
|
|
end
|
2015-12-28 06:06:27 -05:00
|
|
|
|
2016-03-14 08:33:26 -04:00
|
|
|
describe '#artifacts_metadata?' do
|
2016-01-20 15:48:22 -05:00
|
|
|
subject { build.artifacts_metadata? }
|
2016-01-05 08:05:44 -05:00
|
|
|
context 'artifacts metadata does not exist' do
|
2015-12-28 06:06:27 -05:00
|
|
|
it { is_expected.to be_falsy }
|
|
|
|
end
|
|
|
|
|
2016-01-05 08:05:44 -05:00
|
|
|
context 'artifacts archive is a zip file and metadata exists' do
|
2016-02-01 05:31:33 -05:00
|
|
|
let(:build) { create(:ci_build, :artifacts) }
|
2015-12-28 06:06:27 -05:00
|
|
|
it { is_expected.to be_truthy }
|
|
|
|
end
|
|
|
|
end
|
2016-03-14 08:33:26 -04:00
|
|
|
describe '#repo_url' do
|
2016-05-30 06:00:42 -04:00
|
|
|
let(:build) { create(:ci_build) }
|
2015-12-04 06:55:23 -05:00
|
|
|
let(:project) { build.project }
|
|
|
|
|
|
|
|
subject { build.repo_url }
|
|
|
|
|
|
|
|
it { is_expected.to be_a(String) }
|
|
|
|
it { is_expected.to end_with(".git") }
|
|
|
|
it { is_expected.to start_with(project.web_url[0..6]) }
|
2015-12-11 08:37:16 -05:00
|
|
|
it { is_expected.to include(build.token) }
|
2015-12-04 06:55:23 -05:00
|
|
|
it { is_expected.to include('gitlab-ci-token') }
|
|
|
|
it { is_expected.to include(project.web_url[7..-1]) }
|
|
|
|
end
|
2015-12-08 00:10:17 -05:00
|
|
|
|
2016-06-10 10:20:11 -04:00
|
|
|
describe '#artifacts_expire_in' do
|
|
|
|
subject { build.artifacts_expire_in }
|
|
|
|
it { is_expected.to be_nil }
|
|
|
|
|
|
|
|
context 'when artifacts_expire_at is specified' do
|
|
|
|
let(:expire_at) { Time.now + 7.days }
|
|
|
|
|
|
|
|
before { build.artifacts_expire_at = expire_at }
|
|
|
|
|
|
|
|
it { is_expected.to be_within(5).of(expire_at - Time.now) }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#artifacts_expire_in=' do
|
|
|
|
subject { build.artifacts_expire_in }
|
|
|
|
|
|
|
|
it 'when assigning valid duration' do
|
|
|
|
build.artifacts_expire_in = '7 days'
|
2016-06-10 15:45:06 -04:00
|
|
|
|
2016-06-10 10:20:11 -04:00
|
|
|
is_expected.to be_within(10).of(7.days.to_i)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'when assigning invalid duration' do
|
2016-06-14 08:51:09 -04:00
|
|
|
expect { build.artifacts_expire_in = '7 elephants' }.to raise_error(ChronicDuration::DurationParseError)
|
2016-06-10 10:20:11 -04:00
|
|
|
is_expected.to be_nil
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'when resseting value' do
|
|
|
|
build.artifacts_expire_in = nil
|
2016-06-10 15:45:06 -04:00
|
|
|
|
2016-06-10 10:20:11 -04:00
|
|
|
is_expected.to be_nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#keep_artifacts!' do
|
|
|
|
let(:build) { create(:ci_build, artifacts_expire_at: Time.now + 7.days) }
|
|
|
|
|
|
|
|
it 'to reset expire_at' do
|
|
|
|
build.keep_artifacts!
|
2016-06-10 15:45:06 -04:00
|
|
|
|
2016-06-10 10:20:11 -04:00
|
|
|
expect(build.artifacts_expire_at).to be_nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-03-14 08:33:26 -04:00
|
|
|
describe '#depends_on_builds' do
|
2016-06-03 10:22:26 -04:00
|
|
|
let!(:build) { create(:ci_build, pipeline: pipeline, name: 'build', stage_idx: 0, stage: 'build') }
|
|
|
|
let!(:rspec_test) { create(:ci_build, pipeline: pipeline, name: 'rspec', stage_idx: 1, stage: 'test') }
|
|
|
|
let!(:rubocop_test) { create(:ci_build, pipeline: pipeline, name: 'rubocop', stage_idx: 1, stage: 'test') }
|
|
|
|
let!(:staging) { create(:ci_build, pipeline: pipeline, name: 'staging', stage_idx: 2, stage: 'deploy') }
|
2016-01-14 13:45:43 -05:00
|
|
|
|
2016-08-01 11:00:44 -04:00
|
|
|
it 'expects to have no dependents if this is first build' do
|
2016-01-14 13:45:43 -05:00
|
|
|
expect(build.depends_on_builds).to be_empty
|
|
|
|
end
|
|
|
|
|
2016-08-01 11:00:44 -04:00
|
|
|
it 'expects to have one dependent if this is test' do
|
2016-01-14 13:45:43 -05:00
|
|
|
expect(rspec_test.depends_on_builds.map(&:id)).to contain_exactly(build.id)
|
|
|
|
end
|
|
|
|
|
2016-08-01 11:00:44 -04:00
|
|
|
it 'expects to have all builds from build and test stage if this is last' do
|
2016-01-14 13:45:43 -05:00
|
|
|
expect(staging.depends_on_builds.map(&:id)).to contain_exactly(build.id, rspec_test.id, rubocop_test.id)
|
|
|
|
end
|
|
|
|
|
2016-08-01 11:00:44 -04:00
|
|
|
it 'expects to have retried builds instead the original ones' do
|
2016-01-14 13:45:43 -05:00
|
|
|
retried_rspec = Ci::Build.retry(rspec_test)
|
|
|
|
expect(staging.depends_on_builds.map(&:id)).to contain_exactly(build.id, retried_rspec.id, rubocop_test.id)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-06-03 10:22:26 -04:00
|
|
|
def create_mr(build, pipeline, factory: :merge_request, created_at: Time.now)
|
|
|
|
create(factory, source_project_id: pipeline.gl_project_id,
|
|
|
|
target_project_id: pipeline.gl_project_id,
|
2016-05-30 06:00:42 -04:00
|
|
|
source_branch: build.ref,
|
|
|
|
created_at: created_at)
|
2015-12-08 00:10:17 -05:00
|
|
|
end
|
|
|
|
|
2016-03-14 08:33:26 -04:00
|
|
|
describe '#merge_request' do
|
2016-06-03 10:22:26 -04:00
|
|
|
context 'when a MR has a reference to the pipeline' do
|
2015-12-08 00:10:17 -05:00
|
|
|
before do
|
2016-06-03 10:22:26 -04:00
|
|
|
@merge_request = create_mr(build, pipeline, factory: :merge_request)
|
2015-12-08 00:10:17 -05:00
|
|
|
|
2016-06-03 10:22:26 -04:00
|
|
|
commits = [double(id: pipeline.sha)]
|
2015-12-08 00:10:17 -05:00
|
|
|
allow(@merge_request).to receive(:commits).and_return(commits)
|
|
|
|
allow(MergeRequest).to receive_message_chain(:includes, :where, :reorder).and_return([@merge_request])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns the single associated MR' do
|
|
|
|
expect(build.merge_request.id).to eq(@merge_request.id)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-06-03 10:22:26 -04:00
|
|
|
context 'when there is not a MR referencing the pipeline' do
|
2015-12-08 00:10:17 -05:00
|
|
|
it 'returns nil' do
|
|
|
|
expect(build.merge_request).to be_nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-06-03 10:22:26 -04:00
|
|
|
context 'when more than one MR have a reference to the pipeline' do
|
2015-12-08 00:10:17 -05:00
|
|
|
before do
|
2016-06-03 10:22:26 -04:00
|
|
|
@merge_request = create_mr(build, pipeline, factory: :merge_request)
|
2015-12-08 00:10:17 -05:00
|
|
|
@merge_request.close!
|
2016-06-03 10:22:26 -04:00
|
|
|
@merge_request2 = create_mr(build, pipeline, factory: :merge_request)
|
2015-12-08 00:10:17 -05:00
|
|
|
|
2016-06-03 10:22:26 -04:00
|
|
|
commits = [double(id: pipeline.sha)]
|
2015-12-08 00:10:17 -05:00
|
|
|
allow(@merge_request).to receive(:commits).and_return(commits)
|
|
|
|
allow(@merge_request2).to receive(:commits).and_return(commits)
|
|
|
|
allow(MergeRequest).to receive_message_chain(:includes, :where, :reorder).and_return([@merge_request, @merge_request2])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns the first MR' do
|
|
|
|
expect(build.merge_request.id).to eq(@merge_request.id)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when a Build is created after the MR' do
|
|
|
|
before do
|
2016-06-03 10:22:26 -04:00
|
|
|
@merge_request = create_mr(build, pipeline, factory: :merge_request_with_diffs)
|
|
|
|
pipeline2 = create(:ci_pipeline, project: project)
|
|
|
|
@build2 = create(:ci_build, pipeline: pipeline2)
|
2015-12-08 00:10:17 -05:00
|
|
|
|
2016-12-01 06:17:30 -05:00
|
|
|
allow(@merge_request).to receive(:commits_sha).
|
|
|
|
and_return([pipeline.sha, pipeline2.sha])
|
2015-12-08 00:10:17 -05:00
|
|
|
allow(MergeRequest).to receive_message_chain(:includes, :where, :reorder).and_return([@merge_request])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns the current MR' do
|
|
|
|
expect(@build2.merge_request.id).to eq(@merge_request.id)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2016-02-16 02:39:20 -05:00
|
|
|
|
|
|
|
describe 'build erasable' do
|
|
|
|
shared_examples 'erasable' do
|
2016-08-01 11:00:44 -04:00
|
|
|
it 'removes artifact file' do
|
2016-02-16 02:39:20 -05:00
|
|
|
expect(build.artifacts_file.exists?).to be_falsy
|
|
|
|
end
|
|
|
|
|
2016-08-01 11:00:44 -04:00
|
|
|
it 'removes artifact metadata file' do
|
2016-02-16 02:39:20 -05:00
|
|
|
expect(build.artifacts_metadata.exists?).to be_falsy
|
|
|
|
end
|
|
|
|
|
2016-08-01 11:00:44 -04:00
|
|
|
it 'erases build trace in trace file' do
|
2016-02-16 02:39:20 -05:00
|
|
|
expect(build.trace).to be_empty
|
|
|
|
end
|
|
|
|
|
2016-08-01 11:00:44 -04:00
|
|
|
it 'sets erased to true' do
|
2016-02-16 02:39:20 -05:00
|
|
|
expect(build.erased?).to be true
|
|
|
|
end
|
|
|
|
|
2016-08-01 11:00:44 -04:00
|
|
|
it 'sets erase date' do
|
2016-05-23 19:37:59 -04:00
|
|
|
expect(build.erased_at).not_to be_falsy
|
2016-02-16 02:39:20 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'build is not erasable' do
|
|
|
|
let!(:build) { create(:ci_build) }
|
|
|
|
|
|
|
|
describe '#erase' do
|
|
|
|
subject { build.erase }
|
|
|
|
|
|
|
|
it { is_expected.to be false }
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#erasable?' do
|
|
|
|
subject { build.erasable? }
|
|
|
|
it { is_expected.to eq false }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'build is erasable' do
|
2016-02-19 13:31:57 -05:00
|
|
|
let!(:build) { create(:ci_build, :trace, :success, :artifacts) }
|
2016-02-16 02:39:20 -05:00
|
|
|
|
|
|
|
describe '#erase' do
|
2016-06-07 03:00:41 -04:00
|
|
|
before do
|
|
|
|
build.erase(erased_by: user)
|
|
|
|
end
|
2016-02-16 02:39:20 -05:00
|
|
|
|
|
|
|
context 'erased by user' do
|
|
|
|
let!(:user) { create(:user, username: 'eraser') }
|
|
|
|
|
|
|
|
include_examples 'erasable'
|
|
|
|
|
2016-08-01 11:00:44 -04:00
|
|
|
it 'records user who erased a build' do
|
2016-02-16 02:39:20 -05:00
|
|
|
expect(build.erased_by).to eq user
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'erased by system' do
|
|
|
|
let(:user) { nil }
|
|
|
|
|
|
|
|
include_examples 'erasable'
|
|
|
|
|
2016-08-01 11:00:44 -04:00
|
|
|
it 'does not set user who erased a build' do
|
2016-02-16 02:39:20 -05:00
|
|
|
expect(build.erased_by).to be_nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#erasable?' do
|
|
|
|
subject { build.erasable? }
|
2016-07-18 01:37:00 -04:00
|
|
|
it { is_expected.to be_truthy }
|
2016-02-16 02:39:20 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
describe '#erased?' do
|
2016-02-19 13:31:57 -05:00
|
|
|
let!(:build) { create(:ci_build, :trace, :success, :artifacts) }
|
2016-02-16 02:39:20 -05:00
|
|
|
subject { build.erased? }
|
|
|
|
|
|
|
|
context 'build has not been erased' do
|
2016-07-18 01:37:00 -04:00
|
|
|
it { is_expected.to be_falsey }
|
2016-02-16 02:39:20 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
context 'build has been erased' do
|
2016-06-07 03:00:41 -04:00
|
|
|
before do
|
|
|
|
build.erase
|
|
|
|
end
|
2016-02-16 02:39:20 -05:00
|
|
|
|
2016-07-18 01:37:00 -04:00
|
|
|
it { is_expected.to be_truthy }
|
2016-02-16 02:39:20 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'metadata and build trace are not available' do
|
|
|
|
let!(:build) { create(:ci_build, :success, :artifacts) }
|
2016-07-18 01:37:00 -04:00
|
|
|
|
2016-06-07 03:00:41 -04:00
|
|
|
before do
|
|
|
|
build.remove_artifacts_metadata!
|
|
|
|
end
|
2016-02-16 02:39:20 -05:00
|
|
|
|
|
|
|
describe '#erase' do
|
2016-08-01 11:00:44 -04:00
|
|
|
it 'does not raise error' do
|
2016-05-23 19:37:59 -04:00
|
|
|
expect { build.erase }.not_to raise_error
|
2016-02-16 02:39:20 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2016-06-21 08:26:57 -04:00
|
|
|
|
|
|
|
describe '#commit' do
|
|
|
|
it 'returns commit pipeline has been created for' do
|
|
|
|
expect(build.commit).to eq project.commit
|
|
|
|
end
|
|
|
|
end
|
2016-06-28 08:24:43 -04:00
|
|
|
|
2016-08-11 09:22:35 -04:00
|
|
|
describe '#when' do
|
|
|
|
subject { build.when }
|
|
|
|
|
2016-08-12 03:19:28 -04:00
|
|
|
context 'when `when` is undefined' do
|
2016-08-11 09:22:35 -04:00
|
|
|
before do
|
|
|
|
build.when = nil
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'use from gitlab-ci.yml' do
|
|
|
|
before do
|
|
|
|
stub_ci_pipeline_yaml_file(config)
|
|
|
|
end
|
|
|
|
|
2016-08-12 03:19:28 -04:00
|
|
|
context 'when config is not found' do
|
2016-08-11 09:22:35 -04:00
|
|
|
let(:config) { nil }
|
|
|
|
|
|
|
|
it { is_expected.to eq('on_success') }
|
|
|
|
end
|
|
|
|
|
2016-08-12 03:19:28 -04:00
|
|
|
context 'when config does not have a questioned job' do
|
2016-08-11 09:22:35 -04:00
|
|
|
let(:config) do
|
|
|
|
YAML.dump({
|
|
|
|
test_other: {
|
|
|
|
script: 'Hello World'
|
|
|
|
}
|
|
|
|
})
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to eq('on_success') }
|
|
|
|
end
|
|
|
|
|
2016-08-12 03:19:28 -04:00
|
|
|
context 'when config has `when`' do
|
2016-08-11 09:22:35 -04:00
|
|
|
let(:config) do
|
|
|
|
YAML.dump({
|
|
|
|
test: {
|
|
|
|
script: 'Hello World',
|
|
|
|
when: 'always'
|
|
|
|
}
|
|
|
|
})
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to eq('always') }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-12-13 08:18:34 -05:00
|
|
|
describe '#cancelable?' do
|
|
|
|
subject { build }
|
|
|
|
|
|
|
|
context 'when build is cancelable' do
|
|
|
|
context 'when build is pending' do
|
|
|
|
it { is_expected.to be_cancelable }
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when build is running' do
|
|
|
|
before do
|
|
|
|
build.run!
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to be_cancelable }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when build is not cancelable' do
|
|
|
|
context 'when build is successful' do
|
|
|
|
before do
|
|
|
|
build.success!
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.not_to be_cancelable }
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when build is failed' do
|
|
|
|
before do
|
|
|
|
build.drop!
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.not_to be_cancelable }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-06-28 08:24:43 -04:00
|
|
|
describe '#retryable?' do
|
2016-12-13 07:57:18 -05:00
|
|
|
subject { build }
|
|
|
|
|
|
|
|
context 'when build is retryable' do
|
|
|
|
context 'when build is successful' do
|
|
|
|
before do
|
|
|
|
build.success!
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to be_retryable }
|
2016-07-07 09:18:54 -04:00
|
|
|
end
|
2016-06-28 08:24:43 -04:00
|
|
|
|
2016-12-13 07:57:18 -05:00
|
|
|
context 'when build is failed' do
|
|
|
|
before do
|
|
|
|
build.drop!
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to be_retryable }
|
|
|
|
end
|
2016-12-13 08:01:17 -05:00
|
|
|
|
|
|
|
context 'when build is canceled' do
|
|
|
|
before do
|
|
|
|
build.cancel!
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to be_retryable }
|
|
|
|
end
|
2016-06-28 08:24:43 -04:00
|
|
|
end
|
|
|
|
|
2016-12-13 07:57:18 -05:00
|
|
|
context 'when build is not retryable' do
|
|
|
|
context 'when build is running' do
|
|
|
|
before do
|
|
|
|
build.run!
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.not_to be_retryable }
|
2016-07-19 08:59:38 -04:00
|
|
|
end
|
|
|
|
|
2016-12-13 07:57:18 -05:00
|
|
|
context 'when build is skipped' do
|
|
|
|
before do
|
|
|
|
build.skip!
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.not_to be_retryable }
|
|
|
|
end
|
2016-07-19 08:59:38 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-07-16 19:48:51 -04:00
|
|
|
describe '#manual?' do
|
|
|
|
before do
|
|
|
|
build.update(when: value)
|
|
|
|
end
|
|
|
|
|
|
|
|
subject { build.manual? }
|
|
|
|
|
|
|
|
context 'when is set to manual' do
|
|
|
|
let(:value) { 'manual' }
|
|
|
|
|
|
|
|
it { is_expected.to be_truthy }
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when set to something else' do
|
|
|
|
let(:value) { 'something else' }
|
|
|
|
|
|
|
|
it { is_expected.to be_falsey }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#other_actions' do
|
|
|
|
let(:build) { create(:ci_build, :manual, pipeline: pipeline) }
|
|
|
|
let!(:other_build) { create(:ci_build, :manual, pipeline: pipeline, name: 'other action') }
|
|
|
|
|
|
|
|
subject { build.other_actions }
|
|
|
|
|
|
|
|
it 'returns other actions' do
|
|
|
|
is_expected.to contain_exactly(other_build)
|
|
|
|
end
|
2016-07-20 10:38:38 -04:00
|
|
|
|
|
|
|
context 'when build is retried' do
|
|
|
|
let!(:new_build) { Ci::Build.retry(build) }
|
|
|
|
|
|
|
|
it 'does not return any of them' do
|
|
|
|
is_expected.not_to include(build, new_build)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when other build is retried' do
|
|
|
|
let!(:retried_build) { Ci::Build.retry(other_build) }
|
|
|
|
|
|
|
|
it 'returns a retried build' do
|
|
|
|
is_expected.to contain_exactly(retried_build)
|
|
|
|
end
|
|
|
|
end
|
2016-07-16 19:48:51 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
describe '#play' do
|
|
|
|
let(:build) { create(:ci_build, :manual, pipeline: pipeline) }
|
|
|
|
|
|
|
|
subject { build.play }
|
|
|
|
|
2016-08-12 03:19:28 -04:00
|
|
|
it 'enqueues a build' do
|
2016-07-16 19:48:51 -04:00
|
|
|
is_expected.to be_pending
|
|
|
|
is_expected.to eq(build)
|
|
|
|
end
|
|
|
|
|
2016-08-12 07:57:58 -04:00
|
|
|
context 'for successful build' do
|
|
|
|
before do
|
2016-08-12 09:09:35 -04:00
|
|
|
build.update(status: 'success')
|
2016-08-12 07:57:58 -04:00
|
|
|
end
|
2016-07-16 19:48:51 -04:00
|
|
|
|
|
|
|
it 'creates a new build' do
|
|
|
|
is_expected.to be_pending
|
|
|
|
is_expected.not_to eq(build)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2016-07-19 07:23:14 -04:00
|
|
|
|
|
|
|
describe '#when' do
|
|
|
|
subject { build.when }
|
|
|
|
|
2016-08-03 05:24:59 -04:00
|
|
|
context 'when `when` is undefined' do
|
2016-07-19 07:23:14 -04:00
|
|
|
before do
|
|
|
|
build.when = nil
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'use from gitlab-ci.yml' do
|
|
|
|
before do
|
|
|
|
stub_ci_pipeline_yaml_file(config)
|
|
|
|
end
|
|
|
|
|
2016-08-03 05:24:59 -04:00
|
|
|
context 'when config is not found' do
|
2016-07-19 07:23:14 -04:00
|
|
|
let(:config) { nil }
|
|
|
|
|
|
|
|
it { is_expected.to eq('on_success') }
|
|
|
|
end
|
|
|
|
|
2016-08-03 05:24:59 -04:00
|
|
|
context 'when config does not have a questioned job' do
|
2016-07-19 07:23:14 -04:00
|
|
|
let(:config) do
|
|
|
|
YAML.dump({
|
2016-07-19 08:30:09 -04:00
|
|
|
test_other: {
|
|
|
|
script: 'Hello World'
|
|
|
|
}
|
|
|
|
})
|
2016-07-19 07:23:14 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to eq('on_success') }
|
|
|
|
end
|
|
|
|
|
2016-08-03 05:24:59 -04:00
|
|
|
context 'when config has when' do
|
2016-07-19 07:23:14 -04:00
|
|
|
let(:config) do
|
|
|
|
YAML.dump({
|
2016-07-19 08:30:09 -04:00
|
|
|
test: {
|
|
|
|
script: 'Hello World',
|
|
|
|
when: 'always'
|
|
|
|
}
|
|
|
|
})
|
2016-07-19 07:23:14 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to eq('always') }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2016-07-19 08:30:09 -04:00
|
|
|
|
|
|
|
describe '#retryable?' do
|
|
|
|
context 'when build is running' do
|
|
|
|
before { build.run! }
|
|
|
|
|
2016-07-07 09:03:21 -04:00
|
|
|
it 'returns false' do
|
2016-07-18 01:37:00 -04:00
|
|
|
expect(build).not_to be_retryable
|
2016-06-28 08:24:43 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when build is finished' do
|
2016-07-07 09:18:54 -04:00
|
|
|
before do
|
|
|
|
build.success!
|
|
|
|
end
|
2016-06-28 08:24:43 -04:00
|
|
|
|
2016-07-07 09:03:21 -04:00
|
|
|
it 'returns true' do
|
2016-07-18 01:37:00 -04:00
|
|
|
expect(build).to be_retryable
|
2016-06-28 08:24:43 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2016-11-16 18:23:05 -05:00
|
|
|
|
2016-11-17 06:08:28 -05:00
|
|
|
describe '#has_environment?' do
|
|
|
|
subject { build.has_environment? }
|
|
|
|
|
|
|
|
context 'when environment is defined' do
|
|
|
|
before do
|
|
|
|
build.update(environment: 'review')
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to be_truthy }
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when environment is not defined' do
|
|
|
|
before do
|
|
|
|
build.update(environment: nil)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to be_falsey }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-11-16 18:23:05 -05:00
|
|
|
describe '#starts_environment?' do
|
|
|
|
subject { build.starts_environment? }
|
|
|
|
|
|
|
|
context 'when environment is defined' do
|
|
|
|
before do
|
|
|
|
build.update(environment: 'review')
|
|
|
|
end
|
|
|
|
|
2016-11-17 06:08:28 -05:00
|
|
|
context 'no action is defined' do
|
|
|
|
it { is_expected.to be_truthy }
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'and start action is defined' do
|
|
|
|
before do
|
|
|
|
build.update(options: { environment: { action: 'start' } } )
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to be_truthy }
|
|
|
|
end
|
2016-11-16 18:23:05 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
context 'when environment is not defined' do
|
|
|
|
before do
|
|
|
|
build.update(environment: nil)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to be_falsey }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#stops_environment?' do
|
|
|
|
subject { build.stops_environment? }
|
|
|
|
|
|
|
|
context 'when environment is defined' do
|
|
|
|
before do
|
|
|
|
build.update(environment: 'review')
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'no action is defined' do
|
|
|
|
it { is_expected.to be_falsey }
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'and stop action is defined' do
|
|
|
|
before do
|
|
|
|
build.update(options: { environment: { action: 'stop' } } )
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to be_truthy }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when environment is not defined' do
|
|
|
|
before do
|
|
|
|
build.update(environment: nil)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to be_falsey }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#last_deployment' do
|
|
|
|
subject { build.last_deployment }
|
|
|
|
|
2016-11-17 06:08:28 -05:00
|
|
|
context 'when multiple deployments are created' do
|
2016-11-16 18:23:05 -05:00
|
|
|
let!(:deployment1) { create(:deployment, deployable: build) }
|
|
|
|
let!(:deployment2) { create(:deployment, deployable: build) }
|
|
|
|
|
2016-11-17 06:08:28 -05:00
|
|
|
it 'returns the latest one' do
|
|
|
|
is_expected.to eq(deployment2)
|
|
|
|
end
|
2016-11-16 18:23:05 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#outdated_deployment?' do
|
|
|
|
subject { build.outdated_deployment? }
|
|
|
|
|
|
|
|
context 'when build succeeded' do
|
|
|
|
let(:build) { create(:ci_build, :success) }
|
|
|
|
let!(:deployment) { create(:deployment, deployable: build) }
|
|
|
|
|
|
|
|
context 'current deployment is latest' do
|
|
|
|
it { is_expected.to be_falsey }
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'current deployment is not latest on environment' do
|
|
|
|
let!(:deployment2) { create(:deployment, environment: deployment.environment) }
|
|
|
|
|
|
|
|
it { is_expected.to be_truthy }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when build failed' do
|
|
|
|
let(:build) { create(:ci_build, :failed) }
|
|
|
|
|
|
|
|
it { is_expected.to be_falsey }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#expanded_environment_name' do
|
|
|
|
subject { build.expanded_environment_name }
|
|
|
|
|
2016-12-19 08:09:40 -05:00
|
|
|
context 'when environment uses $CI_BUILD_REF_NAME' do
|
|
|
|
let(:build) do
|
|
|
|
create(:ci_build,
|
|
|
|
ref: 'master',
|
|
|
|
environment: 'review/$CI_BUILD_REF_NAME')
|
|
|
|
end
|
2016-11-16 18:23:05 -05:00
|
|
|
|
|
|
|
it { is_expected.to eq('review/master') }
|
|
|
|
end
|
2016-12-19 08:09:40 -05:00
|
|
|
|
|
|
|
context 'when environment uses yaml_variables containing symbol keys' do
|
|
|
|
let(:build) do
|
|
|
|
create(:ci_build,
|
2016-12-19 09:22:57 -05:00
|
|
|
yaml_variables: [{ key: :APP_HOST, value: 'host' }],
|
2016-12-19 08:09:40 -05:00
|
|
|
environment: 'review/$APP_HOST')
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to eq('review/host') }
|
|
|
|
end
|
2016-11-16 18:23:05 -05:00
|
|
|
end
|
2016-12-14 04:21:16 -05:00
|
|
|
|
|
|
|
describe '#detailed_status' do
|
|
|
|
let(:user) { create(:user) }
|
|
|
|
|
|
|
|
it 'returns a detailed status' do
|
|
|
|
expect(build.detailed_status(user))
|
|
|
|
.to be_a Gitlab::Ci::Status::Build::Cancelable
|
|
|
|
end
|
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|