deployment.rb 5.56 KB
Newer Older
1 2
# frozen_string_literal: true

3
class Deployment < ApplicationRecord
4
  include AtomicInternalId
Shinya Maeda's avatar
Shinya Maeda committed
5
  include IidRoutes
6
  include AfterCommitQueue
7
  include UpdatedAtFilterable
8

9 10
  belongs_to :project, required: true
  belongs_to :environment, required: true
11
  belongs_to :cluster, class_name: 'Clusters::Cluster', optional: true
12
  belongs_to :user
13
  belongs_to :deployable, polymorphic: true, optional: true # rubocop:disable Cop/PolymorphicAssociations
14 15 16 17
  has_many :deployment_merge_requests

  has_many :merge_requests,
    through: :deployment_merge_requests
18

19 20 21
  has_internal_id :iid, scope: :project, init: ->(s) do
    Deployment.where(project: s.project).maximum(:iid) if s&.project
  end
22

23 24
  validates :sha, presence: true
  validates :ref, presence: true
25 26 27

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

28 29
  scope :for_environment, -> (environment) { where(environment_id: environment) }

30 31
  scope :visible, -> { where(status: %i[running success failed canceled]) }

32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57
  state_machine :status, initial: :created do
    event :run do
      transition created: :running
    end

    event :succeed do
      transition any - [:success] => :success
    end

    event :drop do
      transition any - [:failed] => :failed
    end

    event :cancel do
      transition any - [:canceled] => :canceled
    end

    before_transition any => [:success, :failed, :canceled] do |deployment|
      deployment.finished_at = Time.now
    end

    after_transition any => :success do |deployment|
      deployment.run_after_commit do
        Deployments::SuccessWorker.perform_async(id)
      end
    end
58 59 60 61 62 63

    after_transition any => [:success, :failed, :canceled] do |deployment|
      deployment.run_after_commit do
        Deployments::FinishedWorker.perform_async(id)
      end
    end
64 65 66 67 68 69 70 71 72 73
  end

  enum status: {
    created: 0,
    running: 1,
    success: 2,
    failed: 3,
    canceled: 4
  }

74 75 76 77 78 79 80 81 82
  def self.last_for_environment(environment)
    ids = self
      .for_environment(environment)
      .select('MAX(id) AS id')
      .group(:environment_id)
      .map(&:id)
    find(ids)
  end

83 84 85 86 87
  def self.distinct_on_environment
    order('environment_id, deployments.id DESC')
      .select('DISTINCT ON (environment_id) deployments.*')
  end

88 89 90 91
  def self.find_successful_deployment!(iid)
    success.find_by!(iid: iid)
  end

92 93 94 95 96 97 98 99 100
  def commit
    project.commit(sha)
  end

  def commit_title
    commit.try(:title)
  end

  def short_sha
101
    Commit.truncate_sha(sha)
102
  end
103

104 105 106 107 108
  def execute_hooks
    deployment_data = Gitlab::DataBuilder::Deployment.build(self)
    project.execute_services(deployment_data, :deployment_hooks)
  end

109 110 111
  def last?
    self == environment.last_deployment
  end
112

Zeger-Jan van de Weg's avatar
Zeger-Jan van de Weg committed
113
  def create_ref
114
    project.repository.create_ref(ref, ref_path)
115
  end
116

117 118 119 120
  def invalidate_cache
    environment.expire_etag_cache
  end

121
  def manual_actions
122 123 124 125 126
    @manual_actions ||= deployable.try(:other_manual_actions)
  end

  def scheduled_actions
    @scheduled_actions ||= deployable.try(:other_scheduled_actions)
127
  end
128

129
  def includes_commit?(commit)
130 131
    return false unless commit

132
    project.repository.ancestor?(commit.id, sha)
133
  end
134

135
  def update_merge_request_metrics!
136
    return unless environment.update_merge_request_metrics? && success?
137

138 139 140
    merge_requests = project.merge_requests
                     .joins(:metrics)
                     .where(target_branch: self.ref, merge_request_metrics: { first_deployed_to_production_at: nil })
141
                     .where("merge_request_metrics.merged_at <= ?", finished_at)
142

143
    if previous_deployment
144
      merge_requests = merge_requests.where("merge_request_metrics.merged_at >= ?", previous_deployment.finished_at)
145
    end
146

147
    MergeRequest::Metrics
Nick Thomas's avatar
Nick Thomas committed
148
      .where(merge_request_id: merge_requests.select(:id), first_deployed_to_production_at: nil)
149
      .update_all(first_deployed_to_production_at: finished_at)
150 151 152 153
  end

  def previous_deployment
    @previous_deployment ||=
154
      project.deployments.joins(:environment)
155 156
      .where(environments: { name: self.environment.name }, ref: self.ref)
      .where.not(id: self.id)
157 158 159 160 161 162 163 164 165 166 167 168
      .order(id: :desc)
      .take
  end

  def previous_environment_deployment
    project
      .deployments
      .success
      .joins(:environment)
      .where(environments: { name: environment.name })
      .where.not(id: self.id)
      .order(id: :desc)
169
      .take
170
  end
171

172
  def stop_action
173 174
    return unless on_stop.present?
    return unless manual_actions
175

176
    @stop_action ||= manual_actions.find_by(name: on_stop)
177 178
  end

179 180 181 182 183 184 185 186 187 188
  def finished_at
    read_attribute(:finished_at) || legacy_finished_at
  end

  def deployed_at
    return unless success?

    finished_at
  end

Z.J. van de Weg's avatar
Z.J. van de Weg committed
189
  def formatted_deployment_time
190
    deployed_at&.to_time&.in_time_zone&.to_s(:medium)
Z.J. van de Weg's avatar
Z.J. van de Weg committed
191 192
  end

193 194 195
  def deployed_by
    # We use deployable's user if available because Ci::PlayBuildService
    # does not update the deployment's user, just the one for the deployable.
196
    # TODO: use deployment's user once https://gitlab.com/gitlab-org/gitlab-foss/issues/66442
197 198 199 200
    # is completed.
    deployable&.user || user
  end

201 202 203 204 205 206 207 208 209 210 211 212
  def link_merge_requests(relation)
    select = relation.select(['merge_requests.id', id]).to_sql

    # We don't use `Gitlab::Database.bulk_insert` here so that we don't need to
    # first pluck lots of IDs into memory.
    DeploymentMergeRequest.connection.execute(<<~SQL)
      INSERT INTO #{DeploymentMergeRequest.table_name}
      (merge_request_id, deployment_id)
      #{select}
    SQL
  end

213 214 215
  private

  def ref_path
216
    File.join(environment.ref_path, 'deployments', iid.to_s)
217
  end
218 219 220 221

  def legacy_finished_at
    self.created_at if success? && !read_attribute(:finished_at)
  end
222
end
223 224

Deployment.prepend_if_ee('EE::Deployment')