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

322 lines
7.9 KiB
Ruby
Raw Normal View History

2015-08-26 01:42:46 +00:00
# == Schema Information
#
# Table name: ci_builds
2015-08-26 01:42:46 +00:00
#
# id :integer not null, primary key
# project_id :integer
# status :string(255)
# finished_at :datetime
# trace :text
# created_at :datetime
# updated_at :datetime
# started_at :datetime
# runner_id :integer
# coverage :float
# commit_id :integer
2015-08-26 01:42:46 +00:00
# commands :text
# job_id :integer
# name :string(255)
# deploy :boolean default(FALSE)
2015-08-26 01:42:46 +00:00
# options :text
# allow_failure :boolean default(FALSE), not null
# stage :string(255)
# trigger_request_id :integer
# stage_idx :integer
# tag :boolean
# ref :string(255)
# user_id :integer
# type :string(255)
# target_url :string(255)
# description :string(255)
# artifacts_file :text
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
LAZY_ATTRIBUTES = ['trace']
belongs_to :runner, class_name: 'Ci::Runner'
belongs_to :trigger_request, class_name: 'Ci::TriggerRequest'
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) }
2015-10-05 12:15:15 +00:00
scope :similar, ->(build) { where(ref: build.ref, tag: build.tag, trigger_request_id: build.trigger_request_id) }
2015-08-26 01:42:46 +00:00
mount_uploader :artifacts_file, ArtifactUploader
2015-08-26 01:42:46 +00:00
acts_as_taggable
# To prevent db load megabytes of data from trace
default_scope -> { select(Ci::Build.columns_without_lazy) }
class << self
def columns_without_lazy
(column_names - LAZY_ATTRIBUTES).map do |column_name|
"#{table_name}.#{column_name}"
end
end
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.commit_id = build.commit_id
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
new_build
end
end
state_machine :status, initial: :pending do
after_transition pending: :running do |build, transition|
build.execute_hooks
end
2015-08-26 01:42:46 +00:00
after_transition any => [:success, :failed, :canceled] do |build, transition|
return unless build.gl_project
2015-08-26 01:42:46 +00:00
project = build.project
if project.coverage_enabled?
build.update_coverage
end
build.commit.create_next_builds(build)
build.execute_hooks
2015-08-26 01:42:46 +00:00
end
end
2015-10-06 10:01:16 +00:00
def ignored?
failed? && allow_failure?
2015-10-05 11:51:28 +00:00
end
def retryable?
commands.present?
end
def retried?
!self.commit.latest_builds_for_ref(self.ref).include?(self)
end
2015-08-26 01:42:46 +00:00
def trace_html
html = Ci::Ansi2html::convert(trace) if trace.present?
2015-10-03 05:56:37 +00:00
html || ''
2015-08-26 01:42:46 +00:00
end
def timeout
project.timeout
end
def variables
predefined_variables + yaml_variables + project_variables + trigger_variables
2015-08-26 01:42:46 +00:00
end
def project
commit.project
end
def project_id
2015-09-28 11:35:26 +00:00
commit.project.id
2015-08-26 01:42:46 +00:00
end
def project_name
project.name
end
2015-10-02 11:46:38 +00:00
def project_recipients
recipients = project.email_recipients.split(' ')
if project.email_add_pusher? && user.present? && user.notification_email.present?
recipients << user.notification_email
2015-10-02 11:46:38 +00:00
end
recipients.uniq
end
2015-08-26 01:42:46 +00:00
def repo_url
project.repo_url_with_auth
end
def allow_git_fetch
project.allow_git_fetch
end
def update_coverage
coverage = extract_coverage(trace, project.coverage_regex)
if coverage.is_a? Numeric
update_attributes(coverage: coverage)
end
end
def extract_coverage(text, regex)
begin
matches = text.gsub(Regexp.new(regex)).to_a.last
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 raw_trace
2015-08-26 01:42:46 +00:00
if File.exist?(path_to_trace)
File.read(path_to_trace)
else
# backward compatibility
read_attribute :trace
end
end
def trace
trace = raw_trace
if project && trace.present?
trace.gsub(project.token, 'xxxxxx')
else
trace
end
end
2015-08-26 01:42:46 +00:00
def trace=(trace)
unless Dir.exists? dir_to_trace
FileUtils.mkdir_p dir_to_trace
end
File.write(path_to_trace, trace)
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
def token
project.token
end
def valid_token? token
project.valid_token? token
end
2015-10-06 10:01:16 +00:00
def target_url
Gitlab::Application.routes.url_helpers.
namespace_project_build_url(gl_project.namespace, gl_project, self)
end
2015-10-12 13:45:46 +00:00
def cancel_url
if active?
2015-10-13 09:50:37 +00:00
Gitlab::Application.routes.url_helpers.
2015-10-13 14:51:13 +00:00
cancel_namespace_project_build_path(gl_project.namespace, gl_project, self)
2015-10-12 13:45:46 +00:00
end
end
def retry_url
if retryable?
2015-10-13 09:50:37 +00:00
Gitlab::Application.routes.url_helpers.
2015-10-13 14:51:13 +00:00
retry_namespace_project_build_path(gl_project.namespace, gl_project, self)
2015-10-12 13:45:46 +00:00
end
end
def can_be_served?(runner)
(tag_list - runner.tag_list).empty?
end
def any_runners_online?
project.any_runners? { |runner| runner.active? && runner.online? && can_be_served?(runner) }
end
def show_warning?
pending? && !any_runners_online?
end
def download_url
if artifacts_file.exists?
Gitlab::Application.routes.url_helpers.
download_namespace_project_build_path(gl_project.namespace, gl_project, self)
end
end
def execute_hooks
build_data = Gitlab::BuildDataBuilder.build(self)
gl_project.execute_hooks(build_data.dup, :build_hooks)
gl_project.execute_services(build_data.dup, :build_hooks)
end
2015-08-26 01:42:46 +00:00
private
def yaml_variables
if commit.config_processor
commit.config_processor.variables.map do |key, value|
{ key: key, value: value, public: true }
end
else
[]
end
end
def project_variables
project.variables.map do |variable|
{ 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