deployment.rb 2.7 KB
Newer Older
1 2 3
class Deployment < ActiveRecord::Base
  include InternalId

4 5
  belongs_to :project, required: true, validate: true
  belongs_to :environment, required: true, validate: true
6 7 8
  belongs_to :user
  belongs_to :deployable, polymorphic: true

K
Kamil Trzcinski 已提交
9 10
  validates :sha, presence: true
  validates :ref, presence: true
11 12 13

  delegate :name, to: :environment, prefix: true

14
  after_save :create_ref
15

16 17 18 19 20 21 22 23 24
  def commit
    project.commit(sha)
  end

  def commit_title
    commit.try(:title)
  end

  def short_sha
K
Kamil Trzcinski 已提交
25
    Commit.truncate_sha(sha)
26
  end
27 28 29 30

  def last?
    self == environment.last_deployment
  end
31

Z
Zeger-Jan van de Weg 已提交
32
  def create_ref
33
    project.repository.create_ref(ref, ref_path)
34
  end
35

36
  def manual_actions
37
    @manual_actions ||= deployable.try(:other_actions)
38
  end
39

40
  def includes_commit?(commit)
41 42
    return false unless commit

43 44 45 46 47 48 49 50
    # Before 8.10, deployments didn't have keep-around refs. Any deployment
    # created before then could have a `sha` referring to a commit that no
    # longer exists in the repository, so just ignore those.
    begin
      project.repository.is_ancestor?(commit.id, sha)
    rescue Rugged::OdbError
      false
    end
51
  end
52

53 54 55 56 57 58 59
  def update_merge_request_metrics!
    return unless environment.update_merge_request_metrics?

    merge_requests = project.merge_requests.
                     joins(:metrics).
                     where(target_branch: self.ref, merge_request_metrics: { first_deployed_to_production_at: nil }).
                     where("merge_request_metrics.merged_at <= ?", self.created_at)
60

61 62
    if previous_deployment
      merge_requests = merge_requests.where("merge_request_metrics.merged_at >= ?", previous_deployment.created_at)
63
    end
64 65 66 67 68 69 70 71 72 73 74 75 76

    # Need to use `map` instead of `select` because MySQL doesn't allow `SELECT`ing from the same table
    # that we're updating.
    merge_request_ids =
      if Gitlab::Database.postgresql?
        merge_requests.select(:id)
      elsif Gitlab::Database.mysql?
        merge_requests.map(&:id)
      end

    MergeRequest::Metrics.
      where(merge_request_id: merge_request_ids, first_deployed_to_production_at: nil).
      update_all(first_deployed_to_production_at: self.created_at)
77 78 79 80 81 82 83 84 85
  end

  def previous_deployment
    @previous_deployment ||=
      project.deployments.joins(:environment).
      where(environments: { name: self.environment.name }, ref: self.ref).
      where.not(id: self.id).
      take
  end
86

87
  def stop_action
K
Kamil Trzcinski 已提交
88
    return nil unless on_stop.present?
89 90
    return nil unless manual_actions

91
    @stop_action ||= manual_actions.find_by(name: on_stop)
92 93
  end

94
  def stoppable?
K
Kamil Trzcinski 已提交
95
    stop_action.present?
96 97
  end

Z
Z.J. van de Weg 已提交
98 99 100 101
  def formatted_deployment_time
    created_at.to_time.in_time_zone.to_s(:medium)
  end

102 103 104
  private

  def ref_path
105
    File.join(environment.ref_path, 'deployments', iid.to_s)
106
  end
107
end