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

224 lines
5.2 KiB
Ruby
Raw Normal View History

2015-08-25 21:42:46 -04:00
# == Schema Information
#
# Table name: ci_commits
2015-08-25 21:42:46 -04:00
#
# id :integer not null, primary key
# project_id :integer
# ref :string(255)
# sha :string(255)
# before_sha :string(255)
# push_data :text
# created_at :datetime
# updated_at :datetime
# tag :boolean default(FALSE)
# yaml_errors :text
# committed_at :datetime
# gl_project_id :integer
2015-08-25 21:42:46 -04:00
#
module Ci
class Commit < ActiveRecord::Base
extend Ci::Model
2015-09-24 11:09:33 -04:00
belongs_to :gl_project, class_name: '::Project', foreign_key: :gl_project_id
2015-10-06 06:01:16 -04:00
has_many :statuses, dependent: :destroy, class_name: 'CommitStatus'
has_many :builds, class_name: 'Ci::Build'
2015-08-25 21:42:46 -04:00
has_many :trigger_requests, dependent: :destroy, class_name: 'Ci::TriggerRequest'
scope :ordered, -> { order('CASE WHEN ci_commits.committed_at IS NULL THEN 0 ELSE 1 END', :committed_at, :id) }
2015-10-02 07:46:38 -04:00
validates_presence_of :sha
2015-08-25 21:42:46 -04:00
validate :valid_commit_sha
def self.truncate_sha(sha)
sha[0...8]
end
def to_param
sha
end
2015-09-28 07:14:34 -04:00
def project
@project ||= gl_project.ensure_gitlab_ci_project
2015-09-28 07:35:26 -04:00
end
def project_id
project.id
2015-09-28 07:14:34 -04:00
end
2015-08-25 21:42:46 -04:00
def last_build
builds.order(:id).last
end
def retry
latest_builds.each do |build|
2015-08-25 21:42:46 -04:00
Ci::Build.retry(build)
end
end
def valid_commit_sha
if self.sha == Ci::Git::BLANK_SHA
self.errors.add(:sha, " cant be 00000000 (branch removal)")
end
end
def git_author_name
2015-10-02 07:46:38 -04:00
commit_data.author_name if commit_data
2015-08-25 21:42:46 -04:00
end
def git_author_email
2015-10-02 07:46:38 -04:00
commit_data.author_email if commit_data
2015-08-25 21:42:46 -04:00
end
def git_commit_message
2015-10-02 07:46:38 -04:00
commit_data.message if commit_data
2015-08-25 21:42:46 -04:00
end
def short_sha
Ci::Commit.truncate_sha(sha)
end
def commit_data
2015-10-02 07:46:38 -04:00
@commit ||= gl_project.commit(sha)
2015-08-25 21:42:46 -04:00
rescue
nil
end
def stage
2015-10-12 15:35:52 -04:00
running_or_pending = statuses.latest.running_or_pending.ordered
2015-10-06 06:01:16 -04:00
running_or_pending.first.try(:stage)
2015-08-25 21:42:46 -04:00
end
def create_builds(ref, tag, user, trigger_request = nil)
2015-08-25 21:42:46 -04:00
return unless config_processor
2015-10-05 06:02:26 -04:00
config_processor.stages.any? do |stage|
CreateBuildsService.new.execute(self, stage, ref, tag, user, trigger_request, 'success').present?
2015-10-05 06:02:26 -04:00
end
end
def create_next_builds(build)
2015-10-05 06:02:26 -04:00
return unless config_processor
# don't create other builds if this one is retried
latest_builds = builds.similar(build).latest
return unless latest_builds.exists?(build.id)
2015-10-05 06:02:26 -04:00
# get list of stages after this build
next_stages = config_processor.stages.drop_while { |stage| stage != build.stage }
next_stages.delete(build.stage)
# get status for all prior builds
prior_builds = latest_builds.reject { |other_build| next_stages.include?(other_build.stage) }
status = Ci::Status.get_status(prior_builds)
# create builds for next stages based
next_stages.any? do |stage|
CreateBuildsService.new.execute(self, stage, build.ref, build.tag, build.user, build.trigger_request, status).present?
2015-10-05 06:02:26 -04:00
end
2015-08-25 21:42:46 -04:00
end
2015-10-02 07:46:38 -04:00
def refs
2015-10-12 10:32:58 -04:00
statuses.order(:ref).pluck(:ref).uniq
2015-08-25 21:42:46 -04:00
end
2015-10-12 10:32:58 -04:00
def latest_statuses
@latest_statuses ||= statuses.latest.to_a
2015-10-05 06:38:00 -04:00
end
def latest_builds
@latest_builds ||= builds.latest.to_a
2015-10-12 10:32:58 -04:00
end
def latest_builds_for_ref(ref)
latest_builds.select { |build| build.ref == ref }
2015-08-25 21:42:46 -04:00
end
2015-10-06 06:01:16 -04:00
def retried
@retried ||= (statuses.order(id: :desc) - statuses.latest)
2015-08-25 21:42:46 -04:00
end
2015-10-12 10:32:58 -04:00
def status
2015-10-06 06:01:16 -04:00
if yaml_errors.present?
2015-08-25 21:42:46 -04:00
return 'failed'
2015-10-06 06:01:16 -04:00
end
@status ||= Ci::Status.get_status(latest_statuses)
2015-08-25 21:42:46 -04:00
end
def pending?
2015-10-06 06:01:16 -04:00
status == 'pending'
2015-08-25 21:42:46 -04:00
end
def running?
2015-10-06 06:01:16 -04:00
status == 'running'
2015-08-25 21:42:46 -04:00
end
def success?
2015-10-06 06:01:16 -04:00
status == 'success'
2015-08-25 21:42:46 -04:00
end
def failed?
status == 'failed'
end
def canceled?
2015-10-06 06:01:16 -04:00
status == 'canceled'
2015-08-25 21:42:46 -04:00
end
2015-10-12 10:32:58 -04:00
def duration
duration_array = latest_statuses.map(&:duration).compact
duration_array.reduce(:+).to_i
2015-10-02 07:46:38 -04:00
end
2015-08-25 21:42:46 -04:00
def finished_at
2015-10-06 06:01:16 -04:00
@finished_at ||= statuses.order('finished_at DESC').first.try(:finished_at)
2015-08-25 21:42:46 -04:00
end
def coverage
if project.coverage_enabled?
coverage_array = latest_builds.map(&:coverage).compact
2015-08-25 21:42:46 -04:00
if coverage_array.size >= 1
'%.2f' % (coverage_array.reduce(:+) / coverage_array.size)
end
end
end
2015-10-12 10:32:58 -04:00
def matrix_for_ref?(ref)
latest_builds_for_ref(ref).size > 1
2015-08-25 21:42:46 -04:00
end
def config_processor
@config_processor ||= Ci::GitlabCiYamlProcessor.new(ci_yaml_file, gl_project.path_with_namespace)
2015-08-25 21:42:46 -04:00
rescue Ci::GitlabCiYamlProcessor::ValidationError => e
save_yaml_error(e.message)
nil
rescue Exception => e
logger.error e.message + "\n" + e.backtrace.join("\n")
save_yaml_error("Undefined yaml error")
nil
end
2015-10-02 07:46:38 -04:00
def ci_yaml_file
2015-10-06 06:00:15 -04:00
gl_project.repository.blob_at(sha, '.gitlab-ci.yml').data
2015-10-02 07:46:38 -04:00
rescue
nil
end
2015-08-25 21:42:46 -04:00
def skip_ci?
2015-10-02 07:46:38 -04:00
git_commit_message =~ /(\[ci skip\])/ if git_commit_message
2015-08-25 21:42:46 -04:00
end
def update_committed!
update!(committed_at: DateTime.now)
end
private
def save_yaml_error(error)
return if self.yaml_errors?
self.yaml_errors = error
save
end
end
end