2015-08-25 21:42:46 -04:00
|
|
|
module Ci
|
2015-10-06 06:01:16 -04:00
|
|
|
class Build < CommitStatus
|
2016-08-08 06:01:25 -04:00
|
|
|
include TokenAuthenticatable
|
|
|
|
|
2015-08-25 21:42:46 -04:00
|
|
|
belongs_to :runner, class_name: 'Ci::Runner'
|
|
|
|
belongs_to :trigger_request, class_name: 'Ci::TriggerRequest'
|
2016-02-16 02:39:20 -05:00
|
|
|
belongs_to :erased_by, class_name: 'User'
|
2015-08-25 21:42:46 -04:00
|
|
|
|
|
|
|
serialize :options
|
2016-07-16 14:10:22 -04:00
|
|
|
serialize :yaml_variables
|
2015-08-25 21:42:46 -04:00
|
|
|
|
|
|
|
validates :coverage, numericality: true, allow_blank: true
|
2015-10-02 07:46:38 -04:00
|
|
|
validates_presence_of :ref
|
2015-08-25 21:42:46 -04:00
|
|
|
|
|
|
|
scope :unstarted, ->() { where(runner_id: nil) }
|
2015-10-02 07:46:38 -04:00
|
|
|
scope :ignore_failures, ->() { where(allow_failure: false) }
|
2016-07-20 11:42:07 -04:00
|
|
|
scope :with_artifacts, ->() { where.not(artifacts_file: [nil, '']) }
|
2016-07-28 01:09:40 -04:00
|
|
|
scope :with_artifacts_not_expired, ->() { with_artifacts.where('artifacts_expire_at IS NULL OR artifacts_expire_at > ?', Time.now) }
|
2016-06-08 11:18:54 -04:00
|
|
|
scope :with_expired_artifacts, ->() { with_artifacts.where('artifacts_expire_at < ?', Time.now) }
|
2016-07-07 06:56:02 -04:00
|
|
|
scope :last_month, ->() { where('created_at > ?', Date.today - 1.month) }
|
2016-08-11 09:22:35 -04:00
|
|
|
scope :manual_actions, ->() { where(when: :manual).relevant }
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2015-10-12 17:47:32 -04:00
|
|
|
mount_uploader :artifacts_file, ArtifactUploader
|
2015-12-30 08:41:44 -05:00
|
|
|
mount_uploader :artifacts_metadata, ArtifactUploader
|
2015-10-12 17:47:32 -04:00
|
|
|
|
2015-08-25 21:42:46 -04:00
|
|
|
acts_as_taggable
|
|
|
|
|
2016-08-08 06:01:25 -04:00
|
|
|
add_authentication_token_field :token
|
|
|
|
|
2016-06-29 05:44:39 -04:00
|
|
|
before_save :update_artifacts_size, if: :artifacts_file_changed?
|
2016-08-08 06:01:25 -04:00
|
|
|
before_save :ensure_token
|
2016-01-05 14:38:35 -05:00
|
|
|
before_destroy { project }
|
|
|
|
|
2016-03-31 13:51:28 -04:00
|
|
|
after_create :execute_hooks
|
2015-08-25 21:42:46 -04:00
|
|
|
|
|
|
|
class << self
|
|
|
|
def first_pending
|
|
|
|
pending.unstarted.order('created_at ASC').first
|
|
|
|
end
|
|
|
|
|
|
|
|
def create_from(build)
|
|
|
|
new_build = build.dup
|
2015-10-06 06:01:16 -04:00
|
|
|
new_build.status = 'pending'
|
2015-08-25 21:42:46 -04:00
|
|
|
new_build.runner_id = nil
|
2015-10-06 06:01:16 -04:00
|
|
|
new_build.trigger_request_id = nil
|
2016-09-15 07:49:11 -04:00
|
|
|
new_build.token = nil
|
2015-08-25 21:42:46 -04:00
|
|
|
new_build.save
|
|
|
|
end
|
|
|
|
|
2016-06-10 17:36:54 -04:00
|
|
|
def retry(build, user = nil)
|
2016-08-11 16:54:25 -04:00
|
|
|
new_build = Ci::Build.create(
|
|
|
|
ref: build.ref,
|
|
|
|
tag: build.tag,
|
|
|
|
options: build.options,
|
|
|
|
commands: build.commands,
|
|
|
|
tag_list: build.tag_list,
|
|
|
|
project: build.project,
|
|
|
|
pipeline: build.pipeline,
|
|
|
|
name: build.name,
|
|
|
|
allow_failure: build.allow_failure,
|
|
|
|
stage: build.stage,
|
|
|
|
stage_idx: build.stage_idx,
|
|
|
|
trigger_request: build.trigger_request,
|
|
|
|
yaml_variables: build.yaml_variables,
|
|
|
|
when: build.when,
|
|
|
|
user: user,
|
|
|
|
environment: build.environment,
|
2016-08-12 07:57:58 -04:00
|
|
|
status_event: 'enqueue'
|
2016-08-11 16:54:25 -04:00
|
|
|
)
|
2016-03-08 13:22:50 -05:00
|
|
|
MergeRequests::AddTodoWhenBuildFailsService.new(build.project, nil).close(new_build)
|
2016-08-18 17:36:54 -04:00
|
|
|
build.pipeline.mark_as_processable_after_stage(build.stage_idx)
|
2015-08-25 21:42:46 -04:00
|
|
|
new_build
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-08-11 09:22:35 -04:00
|
|
|
state_machine :status do
|
2016-02-19 11:38:47 -05:00
|
|
|
after_transition pending: :running do |build|
|
2015-12-07 07:23:23 -05:00
|
|
|
build.execute_hooks
|
|
|
|
end
|
|
|
|
|
2016-02-19 11:38:47 -05:00
|
|
|
after_transition any => [:success, :failed, :canceled] do |build|
|
2015-12-04 06:55:23 -05:00
|
|
|
build.update_coverage
|
2015-12-07 07:23:23 -05:00
|
|
|
build.execute_hooks
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
2016-06-10 17:36:54 -04:00
|
|
|
|
2016-06-14 08:44:09 -04:00
|
|
|
after_transition any => [:success] do |build|
|
2016-06-10 17:36:54 -04:00
|
|
|
if build.environment.present?
|
2016-09-14 16:33:56 -04:00
|
|
|
service = CreateDeploymentService.new(
|
|
|
|
build.project, build.user,
|
|
|
|
environment: build.environment,
|
|
|
|
sha: build.sha,
|
|
|
|
ref: build.ref,
|
|
|
|
tag: build.tag,
|
2016-09-22 07:45:25 -04:00
|
|
|
options: build.options.to_h[:environment],
|
2016-09-14 17:00:15 -04:00
|
|
|
variables: build.variables)
|
2016-06-14 08:44:09 -04:00
|
|
|
service.execute(build)
|
2016-06-10 17:36:54 -04:00
|
|
|
end
|
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
|
2016-07-16 12:39:58 -04:00
|
|
|
def manual?
|
|
|
|
self.when == 'manual'
|
|
|
|
end
|
|
|
|
|
2016-07-16 19:48:51 -04:00
|
|
|
def other_actions
|
2016-07-20 10:38:38 -04:00
|
|
|
pipeline.manual_actions.where.not(name: name)
|
2016-07-16 19:48:51 -04:00
|
|
|
end
|
|
|
|
|
2016-07-16 12:39:58 -04:00
|
|
|
def playable?
|
2016-08-17 04:46:34 -04:00
|
|
|
project.builds_enabled? && commands.present? && manual? && skipped?
|
2016-07-16 12:39:58 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def play(current_user = nil)
|
2016-07-16 19:48:51 -04:00
|
|
|
# Try to queue a current build
|
2016-08-12 07:57:58 -04:00
|
|
|
if self.enqueue
|
2016-07-17 09:59:07 -04:00
|
|
|
self.update(user: current_user)
|
|
|
|
self
|
2016-07-16 12:39:58 -04:00
|
|
|
else
|
|
|
|
# Otherwise we need to create a duplicate
|
|
|
|
Ci::Build.retry(self, current_user)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-11-03 05:44:07 -05:00
|
|
|
def retryable?
|
2016-06-28 08:24:43 -04:00
|
|
|
project.builds_enabled? && commands.present? && complete?
|
2015-11-03 05:44:07 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def retried?
|
2016-06-02 10:19:18 -04:00
|
|
|
!self.pipeline.statuses.latest.include?(self)
|
2016-03-31 13:51:28 -04:00
|
|
|
end
|
|
|
|
|
2016-01-14 13:45:43 -05:00
|
|
|
def depends_on_builds
|
|
|
|
# Get builds of the same type
|
2016-06-02 10:19:18 -04:00
|
|
|
latest_builds = self.pipeline.builds.latest
|
2016-01-14 13:45:43 -05:00
|
|
|
|
|
|
|
# Return builds from previous stages
|
|
|
|
latest_builds.where('stage_idx < ?', stage_idx)
|
|
|
|
end
|
|
|
|
|
2015-08-25 21:42:46 -04:00
|
|
|
def trace_html
|
2016-05-12 10:27:58 -04:00
|
|
|
trace_with_state[:html] || ''
|
2016-05-09 12:59:45 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def trace_with_state(state = nil)
|
|
|
|
trace_with_state = Ci::Ansi2html::convert(trace, state) if trace.present?
|
|
|
|
trace_with_state || {}
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def timeout
|
2015-12-04 06:55:23 -05:00
|
|
|
project.build_timeout
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def variables
|
2016-07-20 07:17:21 -04:00
|
|
|
variables = predefined_variables
|
|
|
|
variables += project.predefined_variables
|
|
|
|
variables += pipeline.predefined_variables
|
|
|
|
variables += runner.predefined_variables if runner
|
|
|
|
variables += project.container_registry_variables
|
|
|
|
variables += yaml_variables
|
2016-09-05 05:42:59 -04:00
|
|
|
variables += user_variables
|
2016-07-20 07:17:21 -04:00
|
|
|
variables += project.secret_variables
|
|
|
|
variables += trigger_request.user_variables if trigger_request
|
2016-07-16 14:10:22 -04:00
|
|
|
variables
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
|
2015-12-08 00:10:17 -05:00
|
|
|
def merge_request
|
|
|
|
merge_requests = MergeRequest.includes(:merge_request_diff)
|
2016-06-02 10:19:18 -04:00
|
|
|
.where(source_branch: ref, source_project_id: pipeline.gl_project_id)
|
2015-12-08 00:10:17 -05:00
|
|
|
.reorder(iid: :asc)
|
|
|
|
|
|
|
|
merge_requests.find do |merge_request|
|
2016-06-03 07:09:49 -04:00
|
|
|
merge_request.commits.any? { |ci| ci.id == pipeline.sha }
|
2015-12-08 00:10:17 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-08-25 21:42:46 -04:00
|
|
|
def project_id
|
2016-06-03 07:09:49 -04:00
|
|
|
pipeline.project_id
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def project_name
|
|
|
|
project.name
|
|
|
|
end
|
|
|
|
|
|
|
|
def repo_url
|
2016-09-16 06:46:33 -04:00
|
|
|
auth = "gitlab-ci-token:#{ensure_token!}@"
|
2015-12-04 06:55:23 -05:00
|
|
|
project.http_url_to_repo.sub(/^https?:\/\//) do |prefix|
|
|
|
|
prefix + auth
|
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def allow_git_fetch
|
2015-12-04 06:55:23 -05:00
|
|
|
project.build_allow_git_fetch
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def update_coverage
|
2016-02-19 11:38:47 -05:00
|
|
|
return unless project
|
2015-12-04 06:55:23 -05:00
|
|
|
coverage_regex = project.build_coverage_regex
|
|
|
|
return unless coverage_regex
|
|
|
|
coverage = extract_coverage(trace, coverage_regex)
|
2015-08-25 21:42:46 -04:00
|
|
|
|
|
|
|
if coverage.is_a? Numeric
|
|
|
|
update_attributes(coverage: coverage)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def extract_coverage(text, regex)
|
|
|
|
begin
|
2015-12-20 13:18:14 -05:00
|
|
|
matches = text.scan(Regexp.new(regex)).last
|
|
|
|
matches = matches.last if matches.kind_of?(Array)
|
2015-08-25 21:42:46 -04:00
|
|
|
coverage = matches.gsub(/\d+(\.\d+)?/).first
|
|
|
|
|
|
|
|
if coverage.present?
|
|
|
|
coverage.to_f
|
|
|
|
end
|
2015-10-03 01:56:37 -04:00
|
|
|
rescue
|
2015-08-25 21:42:46 -04:00
|
|
|
# if bad regex or something goes wrong we dont want to interrupt transition
|
|
|
|
# so we just silentrly ignore error for now
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-06-21 07:12:02 -04:00
|
|
|
def has_trace_file?
|
2016-08-25 07:53:20 -04:00
|
|
|
File.exist?(path_to_trace) || has_old_trace_file?
|
2016-06-21 07:12:02 -04:00
|
|
|
end
|
|
|
|
|
2016-02-02 06:12:03 -05:00
|
|
|
def has_trace?
|
|
|
|
raw_trace.present?
|
2016-02-01 07:46:02 -05:00
|
|
|
end
|
|
|
|
|
2015-10-01 05:43:06 -04:00
|
|
|
def raw_trace
|
2016-09-01 16:17:05 -04:00
|
|
|
if File.exist?(trace_file_path)
|
|
|
|
File.read(trace_file_path)
|
2015-08-25 21:42:46 -04:00
|
|
|
else
|
|
|
|
# backward compatibility
|
|
|
|
read_attribute :trace
|
|
|
|
end
|
|
|
|
end
|
2015-10-01 05:43:06 -04:00
|
|
|
|
2016-08-25 07:53:20 -04:00
|
|
|
##
|
|
|
|
# Deprecated
|
|
|
|
#
|
|
|
|
# This is a hotfix for CI build data integrity, see #4246
|
|
|
|
def has_old_trace_file?
|
|
|
|
project.ci_id && File.exist?(old_path_to_trace)
|
|
|
|
end
|
|
|
|
|
2015-10-01 05:43:06 -04:00
|
|
|
def trace
|
2016-09-19 06:38:03 -04:00
|
|
|
hide_secrets(raw_trace)
|
2015-10-01 05:43:06 -04:00
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2016-03-31 07:24:14 -04:00
|
|
|
def trace_length
|
2016-04-08 19:21:18 -04:00
|
|
|
if raw_trace
|
2016-06-08 13:39:33 -04:00
|
|
|
raw_trace.bytesize
|
2016-03-31 07:24:14 -04:00
|
|
|
else
|
2016-04-08 19:21:18 -04:00
|
|
|
0
|
2016-03-31 07:24:14 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-08-25 21:42:46 -04:00
|
|
|
def trace=(trace)
|
2016-03-29 09:34:18 -04:00
|
|
|
recreate_trace_dir
|
2016-09-19 06:38:03 -04:00
|
|
|
trace = hide_secrets(trace)
|
2016-03-29 09:34:18 -04:00
|
|
|
File.write(path_to_trace, trace)
|
|
|
|
end
|
|
|
|
|
|
|
|
def recreate_trace_dir
|
2016-05-08 17:33:34 -04:00
|
|
|
unless Dir.exist?(dir_to_trace)
|
2015-12-29 02:52:06 -05:00
|
|
|
FileUtils.mkdir_p(dir_to_trace)
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
2016-03-29 09:34:18 -04:00
|
|
|
end
|
|
|
|
private :recreate_trace_dir
|
2015-08-25 21:42:46 -04:00
|
|
|
|
2016-04-08 19:21:18 -04:00
|
|
|
def append_trace(trace_part, offset)
|
2016-03-29 09:34:18 -04:00
|
|
|
recreate_trace_dir
|
|
|
|
|
2016-09-19 06:38:03 -04:00
|
|
|
trace_part = hide_secrets(trace_part)
|
|
|
|
|
2016-04-19 09:48:16 -04:00
|
|
|
File.truncate(path_to_trace, offset) if File.exist?(path_to_trace)
|
2016-06-08 13:39:33 -04:00
|
|
|
File.open(path_to_trace, 'ab') do |f|
|
2016-03-29 09:34:18 -04:00
|
|
|
f.write(trace_part)
|
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
|
2016-08-30 06:47:31 -04:00
|
|
|
def trace_file_path
|
|
|
|
if has_old_trace_file?
|
|
|
|
old_path_to_trace
|
|
|
|
else
|
|
|
|
path_to_trace
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-08-25 21:42:46 -04:00
|
|
|
def dir_to_trace
|
|
|
|
File.join(
|
2015-09-10 06:42:41 -04:00
|
|
|
Settings.gitlab_ci.builds_path,
|
2015-08-25 21:42:46 -04:00
|
|
|
created_at.utc.strftime("%Y_%m"),
|
|
|
|
project.id.to_s
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
def path_to_trace
|
|
|
|
"#{dir_to_trace}/#{id}.log"
|
|
|
|
end
|
|
|
|
|
2015-12-28 09:28:39 -05:00
|
|
|
##
|
|
|
|
# Deprecated
|
|
|
|
#
|
2015-12-29 02:52:06 -05:00
|
|
|
# This is a hotfix for CI build data integrity, see #4246
|
|
|
|
# Should be removed in 8.4, after CI files migration has been done.
|
|
|
|
#
|
2015-12-28 09:28:39 -05:00
|
|
|
def old_dir_to_trace
|
|
|
|
File.join(
|
|
|
|
Settings.gitlab_ci.builds_path,
|
|
|
|
created_at.utc.strftime("%Y_%m"),
|
|
|
|
project.ci_id.to_s
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
##
|
|
|
|
# Deprecated
|
|
|
|
#
|
2015-12-29 02:52:06 -05:00
|
|
|
# This is a hotfix for CI build data integrity, see #4246
|
|
|
|
# Should be removed in 8.4, after CI files migration has been done.
|
|
|
|
#
|
2015-12-28 09:28:39 -05:00
|
|
|
def old_path_to_trace
|
|
|
|
"#{old_dir_to_trace}/#{id}.log"
|
|
|
|
end
|
|
|
|
|
2015-12-29 02:52:06 -05:00
|
|
|
##
|
|
|
|
# Deprecated
|
|
|
|
#
|
|
|
|
# This contains a hotfix for CI build data integrity, see #4246
|
|
|
|
#
|
|
|
|
# This method is used by `ArtifactUploader` to create a store_dir.
|
|
|
|
# Warning: Uploader uses it after AND before file has been stored.
|
|
|
|
#
|
|
|
|
# This method returns old path to artifacts only if it already exists.
|
|
|
|
#
|
|
|
|
def artifacts_path
|
|
|
|
old = File.join(created_at.utc.strftime('%Y_%m'),
|
|
|
|
project.ci_id.to_s,
|
|
|
|
id.to_s)
|
|
|
|
|
|
|
|
old_store = File.join(ArtifactUploader.artifacts_path, old)
|
|
|
|
return old if project.ci_id && File.directory?(old_store)
|
|
|
|
|
|
|
|
File.join(
|
|
|
|
created_at.utc.strftime('%Y_%m'),
|
|
|
|
project.id.to_s,
|
|
|
|
id.to_s
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
2016-05-18 18:43:00 -04:00
|
|
|
def valid_token?(token)
|
2016-08-08 06:01:25 -04:00
|
|
|
self.token && ActiveSupport::SecurityUtils.variable_size_secure_compare(token, self.token)
|
2015-10-12 17:47:32 -04:00
|
|
|
end
|
|
|
|
|
2016-05-06 03:17:27 -04:00
|
|
|
def has_tags?
|
|
|
|
tag_list.any?
|
|
|
|
end
|
|
|
|
|
2015-10-12 15:12:31 -04:00
|
|
|
def any_runners_online?
|
2016-06-08 03:19:49 -04:00
|
|
|
project.any_runners? { |runner| runner.active? && runner.online? && runner.can_pick?(self) }
|
2015-10-12 15:12:31 -04:00
|
|
|
end
|
|
|
|
|
2016-03-09 10:24:02 -05:00
|
|
|
def stuck?
|
2015-10-12 15:12:31 -04:00
|
|
|
pending? && !any_runners_online?
|
|
|
|
end
|
|
|
|
|
2015-12-07 07:23:23 -05:00
|
|
|
def execute_hooks
|
2016-02-19 11:38:47 -05:00
|
|
|
return unless project
|
2016-08-12 04:09:29 -04:00
|
|
|
build_data = Gitlab::DataBuilder::Build.build(self)
|
2015-12-04 06:55:23 -05:00
|
|
|
project.execute_hooks(build_data.dup, :build_hooks)
|
|
|
|
project.execute_services(build_data.dup, :build_hooks)
|
2016-06-01 10:50:32 -04:00
|
|
|
project.running_or_pending_build_count(force: true)
|
2015-12-07 07:23:23 -05:00
|
|
|
end
|
|
|
|
|
2015-12-28 05:43:15 -05:00
|
|
|
def artifacts?
|
2016-09-29 06:59:31 -04:00
|
|
|
!artifacts_expired? && artifacts_file.exists?
|
2015-12-28 05:43:15 -05:00
|
|
|
end
|
|
|
|
|
2016-01-20 15:48:22 -05:00
|
|
|
def artifacts_metadata?
|
2016-01-12 05:02:15 -05:00
|
|
|
artifacts? && artifacts_metadata.exists?
|
2015-12-28 06:06:27 -05:00
|
|
|
end
|
|
|
|
|
2016-01-20 15:48:22 -05:00
|
|
|
def artifacts_metadata_entry(path, **options)
|
2016-06-28 07:16:48 -04:00
|
|
|
metadata = Gitlab::Ci::Build::Artifacts::Metadata.new(
|
|
|
|
artifacts_metadata.path,
|
|
|
|
path,
|
|
|
|
**options)
|
|
|
|
|
|
|
|
metadata.to_entry
|
2015-12-28 04:35:51 -05:00
|
|
|
end
|
|
|
|
|
2016-05-18 16:21:51 -04:00
|
|
|
def erase_artifacts!
|
|
|
|
remove_artifacts_file!
|
|
|
|
remove_artifacts_metadata!
|
2016-06-17 06:58:26 -04:00
|
|
|
save
|
2016-05-18 16:21:51 -04:00
|
|
|
end
|
|
|
|
|
2016-02-16 02:39:20 -05:00
|
|
|
def erase(opts = {})
|
|
|
|
return false unless erasable?
|
|
|
|
|
2016-05-18 16:21:51 -04:00
|
|
|
erase_artifacts!
|
2016-02-16 02:39:20 -05:00
|
|
|
erase_trace!
|
|
|
|
update_erased!(opts[:erased_by])
|
|
|
|
end
|
|
|
|
|
|
|
|
def erasable?
|
|
|
|
complete? && (artifacts? || has_trace?)
|
|
|
|
end
|
|
|
|
|
|
|
|
def erased?
|
|
|
|
!self.erased_at.nil?
|
|
|
|
end
|
|
|
|
|
2016-05-18 16:21:51 -04:00
|
|
|
def artifacts_expired?
|
2016-06-10 08:40:25 -04:00
|
|
|
artifacts_expire_at && artifacts_expire_at < Time.now
|
2016-05-18 16:21:51 -04:00
|
|
|
end
|
|
|
|
|
2016-06-10 08:25:54 -04:00
|
|
|
def artifacts_expire_in
|
|
|
|
artifacts_expire_at - Time.now if artifacts_expire_at
|
|
|
|
end
|
|
|
|
|
|
|
|
def artifacts_expire_in=(value)
|
2016-06-10 15:45:06 -04:00
|
|
|
self.artifacts_expire_at =
|
|
|
|
if value
|
|
|
|
Time.now + ChronicDuration.parse(value)
|
|
|
|
end
|
2016-06-10 08:25:54 -04:00
|
|
|
end
|
|
|
|
|
2016-06-08 11:18:54 -04:00
|
|
|
def keep_artifacts!
|
2016-05-18 16:21:51 -04:00
|
|
|
self.update(artifacts_expire_at: nil)
|
|
|
|
end
|
|
|
|
|
2016-07-19 07:23:14 -04:00
|
|
|
def when
|
|
|
|
read_attribute(:when) || build_attributes_from_config[:when] || 'on_success'
|
2016-02-16 02:39:20 -05:00
|
|
|
end
|
|
|
|
|
2016-07-19 07:23:14 -04:00
|
|
|
def yaml_variables
|
|
|
|
read_attribute(:yaml_variables) || build_attributes_from_config[:yaml_variables] || []
|
2016-02-16 02:39:20 -05:00
|
|
|
end
|
|
|
|
|
2016-09-05 05:42:59 -04:00
|
|
|
def user_variables
|
|
|
|
return [] if user.blank?
|
|
|
|
|
|
|
|
[
|
|
|
|
{ key: 'GITLAB_USER_ID', value: user.id.to_s, public: true },
|
|
|
|
{ key: 'GITLAB_USER_EMAIL', value: user.email, public: true }
|
|
|
|
]
|
|
|
|
end
|
|
|
|
|
2016-02-16 02:39:20 -05:00
|
|
|
private
|
|
|
|
|
2016-06-29 05:44:39 -04:00
|
|
|
def update_artifacts_size
|
2016-06-30 12:00:35 -04:00
|
|
|
self.artifacts_size = if artifacts_file.exists?
|
|
|
|
artifacts_file.size
|
2016-07-04 10:56:23 -04:00
|
|
|
else
|
|
|
|
nil
|
2016-06-30 12:00:35 -04:00
|
|
|
end
|
2016-06-29 05:44:39 -04:00
|
|
|
end
|
|
|
|
|
2016-02-16 02:39:20 -05:00
|
|
|
def erase_trace!
|
|
|
|
self.trace = nil
|
|
|
|
end
|
|
|
|
|
|
|
|
def update_erased!(user = nil)
|
2016-06-08 11:18:54 -04:00
|
|
|
self.update(erased_by: user, erased_at: Time.now, artifacts_expire_at: nil)
|
2016-02-16 02:39:20 -05:00
|
|
|
end
|
|
|
|
|
2015-10-12 11:06:53 -04:00
|
|
|
def predefined_variables
|
2016-07-20 07:17:21 -04:00
|
|
|
variables = [
|
|
|
|
{ key: 'CI', value: 'true', public: true },
|
|
|
|
{ key: 'GITLAB_CI', value: 'true', public: true },
|
|
|
|
{ key: 'CI_BUILD_ID', value: id.to_s, public: true },
|
|
|
|
{ key: 'CI_BUILD_TOKEN', value: token, public: false },
|
|
|
|
{ key: 'CI_BUILD_REF', value: sha, public: true },
|
|
|
|
{ key: 'CI_BUILD_BEFORE_SHA', value: before_sha, public: true },
|
|
|
|
{ key: 'CI_BUILD_REF_NAME', value: ref, public: true },
|
|
|
|
{ key: 'CI_BUILD_NAME', value: name, public: true },
|
|
|
|
{ key: 'CI_BUILD_STAGE', value: stage, public: true },
|
|
|
|
{ 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-06-21 07:55:33 -04:00
|
|
|
variables << { key: 'CI_BUILD_TAG', value: ref, public: true } if tag?
|
|
|
|
variables << { key: 'CI_BUILD_TRIGGERED', value: 'true', public: true } if trigger_request
|
2016-09-05 05:42:59 -04:00
|
|
|
variables << { key: 'CI_BUILD_MANUAL', value: 'true', public: true } if manual?
|
2015-10-12 11:06:53 -04:00
|
|
|
variables
|
|
|
|
end
|
2016-07-19 07:23:14 -04:00
|
|
|
|
|
|
|
def build_attributes_from_config
|
|
|
|
return {} unless pipeline.config_processor
|
2016-08-04 11:44:27 -04:00
|
|
|
|
2016-07-19 07:23:14 -04:00
|
|
|
pipeline.config_processor.build_attributes(name)
|
|
|
|
end
|
2016-09-19 06:38:03 -04:00
|
|
|
|
|
|
|
def hide_secrets(trace)
|
2016-09-20 09:41:41 -04:00
|
|
|
return unless trace
|
|
|
|
|
|
|
|
trace = trace.dup
|
|
|
|
Ci::MaskSecret.mask!(trace, project.runners_token) if project
|
|
|
|
Ci::MaskSecret.mask!(trace, token)
|
2016-09-19 06:38:03 -04:00
|
|
|
trace
|
|
|
|
end
|
2015-08-25 21:42:46 -04:00
|
|
|
end
|
|
|
|
end
|