193 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			Ruby
		
	
	
	
			
		
		
	
	
			193 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			Ruby
		
	
	
	
class CommitStatus < ActiveRecord::Base
 | 
						|
  include HasStatus
 | 
						|
  include Importable
 | 
						|
  include AfterCommitQueue
 | 
						|
 | 
						|
  self.table_name = 'ci_builds'
 | 
						|
 | 
						|
  belongs_to :user
 | 
						|
  belongs_to :project
 | 
						|
  belongs_to :pipeline, class_name: 'Ci::Pipeline', foreign_key: :commit_id
 | 
						|
  belongs_to :auto_canceled_by, class_name: 'Ci::Pipeline'
 | 
						|
 | 
						|
  delegate :commit, to: :pipeline
 | 
						|
  delegate :sha, :short_sha, to: :pipeline
 | 
						|
 | 
						|
  validates :pipeline, presence: true, unless: :importing?
 | 
						|
  validates :name, presence: true, unless: :importing?
 | 
						|
 | 
						|
  alias_attribute :author, :user
 | 
						|
  alias_attribute :pipeline_id, :commit_id
 | 
						|
 | 
						|
  scope :failed_but_allowed, -> do
 | 
						|
    where(allow_failure: true, status: [:failed, :canceled])
 | 
						|
  end
 | 
						|
 | 
						|
  scope :exclude_ignored, -> do
 | 
						|
    # We want to ignore failed but allowed to fail jobs.
 | 
						|
    #
 | 
						|
    # TODO, we also skip ignored optional manual actions.
 | 
						|
    where("allow_failure = ? OR status IN (?)",
 | 
						|
      false, all_state_names - [:failed, :canceled, :manual])
 | 
						|
  end
 | 
						|
 | 
						|
  scope :latest, -> { where(retried: [false, nil]) }
 | 
						|
  scope :retried, -> { where(retried: true) }
 | 
						|
  scope :ordered, -> { order(:name) }
 | 
						|
  scope :latest_ordered, -> { latest.ordered.includes(project: :namespace) }
 | 
						|
  scope :retried_ordered, -> { retried.ordered.includes(project: :namespace) }
 | 
						|
  scope :after_stage, -> (index) { where('stage_idx > ?', index) }
 | 
						|
 | 
						|
  enum failure_reason: {
 | 
						|
    unknown_failure: nil,
 | 
						|
    script_failure: 1,
 | 
						|
    api_failure: 2,
 | 
						|
    stuck_or_timeout_failure: 3,
 | 
						|
    runner_system_failure: 4,
 | 
						|
    missing_dependency_failure: 5
 | 
						|
  }
 | 
						|
 | 
						|
  ##
 | 
						|
  # We still create some CommitStatuses outside of CreatePipelineService.
 | 
						|
  #
 | 
						|
  # These are pages deployments and external statuses.
 | 
						|
  #
 | 
						|
  before_create unless: :importing? do
 | 
						|
    Ci::EnsureStageService.new(project, user).execute(self) do |stage|
 | 
						|
      self.run_after_commit { StageUpdateWorker.perform_async(stage.id) }
 | 
						|
    end
 | 
						|
  end
 | 
						|
 | 
						|
  state_machine :status do
 | 
						|
    event :process do
 | 
						|
      transition [:skipped, :manual] => :created
 | 
						|
    end
 | 
						|
 | 
						|
    event :enqueue do
 | 
						|
      transition [:created, :skipped, :manual] => :pending
 | 
						|
    end
 | 
						|
 | 
						|
    event :run do
 | 
						|
      transition pending: :running
 | 
						|
    end
 | 
						|
 | 
						|
    event :skip do
 | 
						|
      transition [:created, :pending] => :skipped
 | 
						|
    end
 | 
						|
 | 
						|
    event :drop do
 | 
						|
      transition [:created, :pending, :running] => :failed
 | 
						|
    end
 | 
						|
 | 
						|
    event :success do
 | 
						|
      transition [:created, :pending, :running] => :success
 | 
						|
    end
 | 
						|
 | 
						|
    event :cancel do
 | 
						|
      transition [:created, :pending, :running, :manual] => :canceled
 | 
						|
    end
 | 
						|
 | 
						|
    before_transition created: [:pending, :running] do |commit_status|
 | 
						|
      commit_status.queued_at = Time.now
 | 
						|
    end
 | 
						|
 | 
						|
    before_transition [:created, :pending] => :running do |commit_status|
 | 
						|
      commit_status.started_at = Time.now
 | 
						|
    end
 | 
						|
 | 
						|
    before_transition any => [:success, :failed, :canceled] do |commit_status|
 | 
						|
      commit_status.finished_at = Time.now
 | 
						|
    end
 | 
						|
 | 
						|
    before_transition any => :failed do |commit_status, transition|
 | 
						|
      failure_reason = transition.args.first
 | 
						|
      commit_status.failure_reason = failure_reason
 | 
						|
    end
 | 
						|
 | 
						|
    after_transition do |commit_status, transition|
 | 
						|
      next unless commit_status.project
 | 
						|
      next if transition.loopback?
 | 
						|
 | 
						|
      commit_status.run_after_commit do
 | 
						|
        if pipeline_id
 | 
						|
          if complete? || manual?
 | 
						|
            PipelineProcessWorker.perform_async(pipeline_id)
 | 
						|
          else
 | 
						|
            PipelineUpdateWorker.perform_async(pipeline_id)
 | 
						|
          end
 | 
						|
        end
 | 
						|
 | 
						|
        StageUpdateWorker.perform_async(stage_id)
 | 
						|
        ExpireJobCacheWorker.perform_async(id)
 | 
						|
      end
 | 
						|
    end
 | 
						|
 | 
						|
    after_transition any => :failed do |commit_status|
 | 
						|
      next unless commit_status.project
 | 
						|
 | 
						|
      commit_status.run_after_commit do
 | 
						|
        MergeRequests::AddTodoWhenBuildFailsService
 | 
						|
          .new(project, nil).execute(self)
 | 
						|
      end
 | 
						|
    end
 | 
						|
  end
 | 
						|
 | 
						|
  def locking_enabled?
 | 
						|
    status_changed?
 | 
						|
  end
 | 
						|
 | 
						|
  def before_sha
 | 
						|
    pipeline.before_sha || Gitlab::Git::BLANK_SHA
 | 
						|
  end
 | 
						|
 | 
						|
  def group_name
 | 
						|
    name.to_s.gsub(/\d+[\s:\/\\]+\d+\s*/, '').strip
 | 
						|
  end
 | 
						|
 | 
						|
  def failed_but_allowed?
 | 
						|
    allow_failure? && (failed? || canceled?)
 | 
						|
  end
 | 
						|
 | 
						|
  def duration
 | 
						|
    calculate_duration
 | 
						|
  end
 | 
						|
 | 
						|
  def playable?
 | 
						|
    false
 | 
						|
  end
 | 
						|
 | 
						|
  # To be overriden when inherrited from
 | 
						|
  def retryable?
 | 
						|
    false
 | 
						|
  end
 | 
						|
 | 
						|
  # To be overriden when inherrited from
 | 
						|
  def cancelable?
 | 
						|
    false
 | 
						|
  end
 | 
						|
 | 
						|
  def stuck?
 | 
						|
    false
 | 
						|
  end
 | 
						|
 | 
						|
  def has_trace?
 | 
						|
    false
 | 
						|
  end
 | 
						|
 | 
						|
  def auto_canceled?
 | 
						|
    canceled? && auto_canceled_by_id?
 | 
						|
  end
 | 
						|
 | 
						|
  def detailed_status(current_user)
 | 
						|
    Gitlab::Ci::Status::Factory
 | 
						|
      .new(self, current_user)
 | 
						|
      .fabricate!
 | 
						|
  end
 | 
						|
 | 
						|
  def sortable_name
 | 
						|
    name.to_s.split(/(\d+)/).map do |v|
 | 
						|
      v =~ /\d+/ ? v.to_i : v
 | 
						|
    end
 | 
						|
  end
 | 
						|
end
 |