gitlab-org--gitlab-foss/app/models/ci/build.rb

422 lines
10 KiB
Ruby
Raw Normal View History

2015-08-26 01:42:46 +00:00
module Ci
2015-10-06 10:01:16 +00:00
class Build < CommitStatus
2015-08-26 01:42:46 +00:00
belongs_to :runner, class_name: 'Ci::Runner'
belongs_to :trigger_request, class_name: 'Ci::TriggerRequest'
belongs_to :erased_by, class_name: 'User'
2015-08-26 01:42:46 +00:00
serialize :options
validates :coverage, numericality: true, allow_blank: true
2015-10-02 11:46:38 +00:00
validates_presence_of :ref
2015-08-26 01:42:46 +00:00
scope :unstarted, ->() { where(runner_id: nil) }
2015-10-02 11:46:38 +00:00
scope :ignore_failures, ->() { where(allow_failure: false) }
2016-05-18 20:21:51 +00:00
scope :with_artifacts, ->() { where.not(artifacts_file: nil) }
scope :with_artifacts_expired, ->() { with_artifacts.where('artifacts_expire_at < ?', Time.now) }
2015-08-26 01:42:46 +00:00
mount_uploader :artifacts_file, ArtifactUploader
mount_uploader :artifacts_metadata, ArtifactUploader
2015-08-26 01:42:46 +00:00
acts_as_taggable
before_destroy { project }
2016-03-31 17:51:28 +00:00
after_create :execute_hooks
2015-08-26 01:42:46 +00:00
class << self
def last_month
where('created_at > ?', Date.today - 1.month)
end
def first_pending
pending.unstarted.order('created_at ASC').first
end
def create_from(build)
new_build = build.dup
2015-10-06 10:01:16 +00:00
new_build.status = 'pending'
2015-08-26 01:42:46 +00:00
new_build.runner_id = nil
2015-10-06 10:01:16 +00:00
new_build.trigger_request_id = nil
2015-08-26 01:42:46 +00:00
new_build.save
end
def retry(build)
2015-10-06 10:01:16 +00:00
new_build = Ci::Build.new(status: 'pending')
2015-10-05 10:02:26 +00:00
new_build.ref = build.ref
new_build.tag = build.tag
2015-08-26 01:42:46 +00:00
new_build.options = build.options
new_build.commands = build.commands
new_build.tag_list = build.tag_list
new_build.project = build.project
new_build.pipeline = build.pipeline
2015-08-26 01:42:46 +00:00
new_build.name = build.name
new_build.allow_failure = build.allow_failure
new_build.stage = build.stage
2015-10-02 11:46:38 +00:00
new_build.stage_idx = build.stage_idx
2015-08-26 01:42:46 +00:00
new_build.trigger_request = build.trigger_request
new_build.save
MergeRequests::AddTodoWhenBuildFailsService.new(build.project, nil).close(new_build)
2015-08-26 01:42:46 +00:00
new_build
end
end
state_machine :status, initial: :pending do
after_transition pending: :running do |build|
build.execute_hooks
end
# We use around_transition to create builds for next stage as soon as possible, before the `after_*` is executed
around_transition any => [:success, :failed, :canceled] do |build, block|
block.call
build.pipeline.create_next_builds(build) if build.pipeline
end
after_transition any => [:success, :failed, :canceled] do |build|
2015-12-04 11:55:23 +00:00
build.update_coverage
build.execute_hooks
2015-08-26 01:42:46 +00:00
end
end
def retryable?
2015-12-04 11:55:23 +00:00
project.builds_enabled? && commands.present?
end
def retried?
!self.pipeline.statuses.latest.include?(self)
2016-03-31 17:51:28 +00:00
end
def retry
Ci::Build.retry(self)
end
def depends_on_builds
# Get builds of the same type
latest_builds = self.pipeline.builds.latest
# Return builds from previous stages
latest_builds.where('stage_idx < ?', stage_idx)
end
2015-08-26 01:42:46 +00:00
def trace_html
2016-05-12 14:27:58 +00:00
trace_with_state[:html] || ''
end
def trace_with_state(state = nil)
trace_with_state = Ci::Ansi2html::convert(trace, state) if trace.present?
trace_with_state || {}
2015-08-26 01:42:46 +00:00
end
def timeout
2015-12-04 11:55:23 +00:00
project.build_timeout
2015-08-26 01:42:46 +00:00
end
def variables
predefined_variables + yaml_variables + project_variables + trigger_variables
2015-08-26 01:42:46 +00:00
end
def merge_request
merge_requests = MergeRequest.includes(:merge_request_diff)
.where(source_branch: ref, source_project_id: pipeline.gl_project_id)
.reorder(iid: :asc)
merge_requests.find do |merge_request|
2016-06-03 11:09:49 +00:00
merge_request.commits.any? { |ci| ci.id == pipeline.sha }
end
end
2015-08-26 01:42:46 +00:00
def project_id
2016-06-03 11:09:49 +00:00
pipeline.project_id
2015-08-26 01:42:46 +00:00
end
def project_name
project.name
end
def repo_url
2015-12-04 11:55:23 +00:00
auth = "gitlab-ci-token:#{token}@"
project.http_url_to_repo.sub(/^https?:\/\//) do |prefix|
prefix + auth
end
2015-08-26 01:42:46 +00:00
end
def allow_git_fetch
2015-12-04 11:55:23 +00:00
project.build_allow_git_fetch
2015-08-26 01:42:46 +00:00
end
def update_coverage
return unless project
2015-12-04 11:55:23 +00:00
coverage_regex = project.build_coverage_regex
return unless coverage_regex
coverage = extract_coverage(trace, coverage_regex)
2015-08-26 01:42:46 +00:00
if coverage.is_a? Numeric
update_attributes(coverage: coverage)
end
end
def extract_coverage(text, regex)
begin
matches = text.scan(Regexp.new(regex)).last
matches = matches.last if matches.kind_of?(Array)
2015-08-26 01:42:46 +00:00
coverage = matches.gsub(/\d+(\.\d+)?/).first
if coverage.present?
coverage.to_f
end
2015-10-03 05:56:37 +00:00
rescue
2015-08-26 01:42:46 +00:00
# if bad regex or something goes wrong we dont want to interrupt transition
# so we just silentrly ignore error for now
end
end
def has_trace?
raw_trace.present?
end
def raw_trace
if File.file?(path_to_trace)
2015-08-26 01:42:46 +00:00
File.read(path_to_trace)
elsif project.ci_id && File.file?(old_path_to_trace)
# Temporary fix for build trace data integrity
File.read(old_path_to_trace)
2015-08-26 01:42:46 +00:00
else
# backward compatibility
read_attribute :trace
end
end
def trace
trace = raw_trace
if project && trace.present? && project.runners_token.present?
2015-12-11 12:39:43 +00:00
trace.gsub(project.runners_token, 'xxxxxx')
else
trace
end
end
2015-08-26 01:42:46 +00:00
2016-03-31 11:24:14 +00:00
def trace_length
if raw_trace
raw_trace.bytesize
2016-03-31 11:24:14 +00:00
else
0
2016-03-31 11:24:14 +00:00
end
end
2015-08-26 01:42:46 +00:00
def trace=(trace)
2016-03-29 13:34:18 +00:00
recreate_trace_dir
File.write(path_to_trace, trace)
end
def recreate_trace_dir
unless Dir.exist?(dir_to_trace)
FileUtils.mkdir_p(dir_to_trace)
2015-08-26 01:42:46 +00:00
end
2016-03-29 13:34:18 +00:00
end
private :recreate_trace_dir
2015-08-26 01:42:46 +00:00
def append_trace(trace_part, offset)
2016-03-29 13:34:18 +00:00
recreate_trace_dir
File.truncate(path_to_trace, offset) if File.exist?(path_to_trace)
File.open(path_to_trace, 'ab') do |f|
2016-03-29 13:34:18 +00:00
f.write(trace_part)
end
2015-08-26 01:42:46 +00:00
end
def dir_to_trace
File.join(
2015-09-10 10:42:41 +00:00
Settings.gitlab_ci.builds_path,
2015-08-26 01:42:46 +00:00
created_at.utc.strftime("%Y_%m"),
project.id.to_s
)
end
def path_to_trace
"#{dir_to_trace}/#{id}.log"
end
##
# Deprecated
#
# This is a hotfix for CI build data integrity, see #4246
# Should be removed in 8.4, after CI files migration has been done.
#
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
#
# This is a hotfix for CI build data integrity, see #4246
# Should be removed in 8.4, after CI files migration has been done.
#
def old_path_to_trace
"#{old_dir_to_trace}/#{id}.log"
end
##
# 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
def token
2015-12-11 12:39:43 +00:00
project.runners_token
end
def valid_token?(token)
2015-12-10 16:29:44 +00:00
project.valid_runners_token? token
end
def can_be_served?(runner)
return false unless has_tags? || runner.run_untagged?
(tag_list - runner.tag_list).empty?
end
def has_tags?
tag_list.any?
end
def any_runners_online?
project.any_runners? { |runner| runner.active? && runner.online? && can_be_served?(runner) }
end
def stuck?
pending? && !any_runners_online?
end
def execute_hooks
return unless project
build_data = Gitlab::BuildDataBuilder.build(self)
2015-12-04 11:55:23 +00:00
project.execute_hooks(build_data.dup, :build_hooks)
project.execute_services(build_data.dup, :build_hooks)
2016-06-01 14:50:32 +00:00
project.running_or_pending_build_count(force: true)
end
def artifacts?
artifacts_file.exists?
end
def artifacts_metadata?
artifacts? && artifacts_metadata.exists?
end
def artifacts_metadata_entry(path, **options)
Gitlab::Ci::Build::Artifacts::Metadata.new(artifacts_metadata.path, path, **options).to_entry
end
2016-05-18 20:21:51 +00:00
def erase_artifacts!
remove_artifacts_file!
remove_artifacts_metadata!
end
def erase(opts = {})
return false unless erasable?
2016-05-18 20:21:51 +00:00
erase_artifacts!
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 20:21:51 +00:00
def artifacts_expired?
self.artifacts_expire_at < Time.now && !artifacts?
end
def keep_artifacts
self.update(artifacts_expire_at: nil)
end
private
def erase_trace!
self.trace = nil
end
def update_erased!(user = nil)
self.update(erased_by: user, erased_at: Time.now)
end
2015-08-26 01:42:46 +00:00
def yaml_variables
global_yaml_variables + job_yaml_variables
end
def global_yaml_variables
2016-06-03 11:09:49 +00:00
if pipeline.config_processor
pipeline.config_processor.global_variables.map do |key, value|
2015-08-26 01:42:46 +00:00
{ key: key, value: value, public: true }
end
else
[]
end
end
def job_yaml_variables
2016-06-03 11:09:49 +00:00
if pipeline.config_processor
pipeline.config_processor.job_variables(name).map do |key, value|
{ key: key, value: value, public: true }
end
else
[]
end
end
2015-08-26 01:42:46 +00:00
def project_variables
project.variables.map do |variable|
2015-08-26 01:42:46 +00:00
{ key: variable.key, value: variable.value, public: false }
end
end
def trigger_variables
if trigger_request && trigger_request.variables
trigger_request.variables.map do |key, value|
{ key: key, value: value, public: false }
end
else
[]
end
end
def predefined_variables
variables = []
variables << { key: :CI_BUILD_TAG, value: ref, public: true } if tag?
variables << { key: :CI_BUILD_NAME, value: name, public: true }
variables << { key: :CI_BUILD_STAGE, value: stage, public: true }
variables << { key: :CI_BUILD_TRIGGERED, value: 'true', public: true } if trigger_request
variables
end
2015-08-26 01:42:46 +00:00
end
end