2018-07-25 05:30:33 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2019-03-28 09:17:42 -04:00
|
|
|
class Deployment < ApplicationRecord
|
2018-04-20 10:00:15 -04:00
|
|
|
include AtomicInternalId
|
2018-05-11 03:52:48 -04:00
|
|
|
include IidRoutes
|
2018-11-04 19:37:40 -05:00
|
|
|
include AfterCommitQueue
|
2019-11-27 22:06:32 -05:00
|
|
|
include UpdatedAtFilterable
|
2020-01-14 10:07:55 -05:00
|
|
|
include Importable
|
2019-11-29 04:06:31 -05:00
|
|
|
include Gitlab::Utils::StrongMemoize
|
2020-03-25 02:07:58 -04:00
|
|
|
include FastDestroyAll
|
2021-06-02 17:10:00 -04:00
|
|
|
include IgnorableColumns
|
|
|
|
|
|
|
|
ignore_column :deployable_id_convert_to_bigint, remove_with: '14.2', remove_after: '2021-08-22'
|
2016-06-10 17:36:54 -04:00
|
|
|
|
2018-02-15 19:37:33 -05:00
|
|
|
belongs_to :project, required: true
|
|
|
|
belongs_to :environment, required: true
|
2019-06-23 17:48:16 -04:00
|
|
|
belongs_to :cluster, class_name: 'Clusters::Cluster', optional: true
|
2016-06-10 17:36:54 -04:00
|
|
|
belongs_to :user
|
2019-10-16 14:08:01 -04:00
|
|
|
belongs_to :deployable, polymorphic: true, optional: true # rubocop:disable Cop/PolymorphicAssociations
|
2019-11-06 19:06:18 -05:00
|
|
|
has_many :deployment_merge_requests
|
|
|
|
|
|
|
|
has_many :merge_requests,
|
|
|
|
through: :deployment_merge_requests
|
2016-06-10 17:36:54 -04:00
|
|
|
|
2020-02-10 07:08:59 -05:00
|
|
|
has_one :deployment_cluster
|
|
|
|
|
2020-11-10 19:08:58 -05:00
|
|
|
has_internal_id :iid, scope: :project, track_if: -> { !importing? }
|
2018-04-20 10:00:15 -04:00
|
|
|
|
2016-06-14 12:34:48 -04:00
|
|
|
validates :sha, presence: true
|
|
|
|
validates :ref, presence: true
|
2019-12-20 22:07:37 -05:00
|
|
|
validate :valid_sha, on: :create
|
|
|
|
validate :valid_ref, on: :create
|
2016-06-10 17:36:54 -04:00
|
|
|
|
|
|
|
delegate :name, to: :environment, prefix: true
|
2020-02-14 04:08:43 -05:00
|
|
|
delegate :kubernetes_namespace, to: :deployment_cluster, allow_nil: true
|
2016-06-10 17:36:54 -04:00
|
|
|
|
2018-10-12 10:10:34 -04:00
|
|
|
scope :for_environment, -> (environment) { where(environment_id: environment) }
|
2021-05-06 02:10:11 -04:00
|
|
|
scope :for_environment_name, -> (project, name) do
|
2021-05-14 05:10:24 -04:00
|
|
|
where('deployments.environment_id = (?)',
|
|
|
|
Environment.select(:id).where(project: project, name: name).limit(1))
|
2020-01-17 16:08:29 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
scope :for_status, -> (status) { where(status: status) }
|
2020-12-08 19:09:42 -05:00
|
|
|
scope :for_project, -> (project_id) { where(project_id: project_id) }
|
2021-02-16 13:09:24 -05:00
|
|
|
scope :for_projects, -> (projects) { where(project: projects) }
|
2018-10-12 10:10:34 -04:00
|
|
|
|
2019-10-16 14:08:01 -04:00
|
|
|
scope :visible, -> { where(status: %i[running success failed canceled]) }
|
2020-02-13 19:09:07 -05:00
|
|
|
scope :stoppable, -> { where.not(on_stop: nil).where.not(deployable_id: nil).success }
|
2020-02-17 04:08:52 -05:00
|
|
|
scope :active, -> { where(status: %i[created running]) }
|
2020-11-20 01:09:10 -05:00
|
|
|
scope :older_than, -> (deployment) { where('deployments.id < ?', deployment.id) }
|
|
|
|
scope :with_deployable, -> { joins('INNER JOIN ci_builds ON ci_builds.id = deployments.deployable_id').preload(:deployable) }
|
2021-03-29 08:09:14 -04:00
|
|
|
scope :with_api_entity_associations, -> { preload({ deployable: { runner: [], tags: [], user: [], job_artifacts_archive: [] } }) }
|
2019-10-16 14:08:01 -04:00
|
|
|
|
2021-02-16 13:09:24 -05:00
|
|
|
scope :finished_after, ->(date) { where('finished_at >= ?', date) }
|
|
|
|
scope :finished_before, ->(date) { where('finished_at < ?', date) }
|
2020-12-13 16:09:47 -05:00
|
|
|
|
2020-10-05 02:08:45 -04:00
|
|
|
FINISHED_STATUSES = %i[success failed canceled].freeze
|
|
|
|
|
2018-11-04 19:37:40 -05:00
|
|
|
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
|
|
|
|
|
2020-11-18 10:09:08 -05:00
|
|
|
event :skip do
|
|
|
|
transition any - [:skipped] => :skipped
|
|
|
|
end
|
|
|
|
|
2020-10-05 02:08:45 -04:00
|
|
|
before_transition any => FINISHED_STATUSES do |deployment|
|
2020-05-22 05:08:09 -04:00
|
|
|
deployment.finished_at = Time.current
|
2018-11-04 19:37:40 -05:00
|
|
|
end
|
|
|
|
|
2020-10-05 02:08:45 -04:00
|
|
|
after_transition any => :running do |deployment|
|
2020-10-20 02:09:03 -04:00
|
|
|
next unless deployment.project.ci_forward_deployment_enabled?
|
2020-10-05 02:08:45 -04:00
|
|
|
|
2018-11-04 19:37:40 -05:00
|
|
|
deployment.run_after_commit do
|
2020-10-15 11:08:45 -04:00
|
|
|
Deployments::DropOlderDeploymentsWorker.perform_async(id)
|
2018-11-04 19:37:40 -05:00
|
|
|
end
|
|
|
|
end
|
2019-04-26 17:08:41 -04:00
|
|
|
|
2020-10-05 02:08:45 -04:00
|
|
|
after_transition any => :running do |deployment|
|
2019-04-26 17:08:41 -04:00
|
|
|
deployment.run_after_commit do
|
2021-04-29 14:10:23 -04:00
|
|
|
Deployments::HooksWorker.perform_async(deployment_id: id, status_changed_at: Time.current)
|
2019-04-26 17:08:41 -04:00
|
|
|
end
|
|
|
|
end
|
2020-02-17 04:08:52 -05:00
|
|
|
|
2020-10-05 02:08:45 -04:00
|
|
|
after_transition any => :success do |deployment|
|
|
|
|
deployment.run_after_commit do
|
|
|
|
Deployments::UpdateEnvironmentWorker.perform_async(id)
|
|
|
|
Deployments::LinkMergeRequestWorker.perform_async(id)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
after_transition any => FINISHED_STATUSES do |deployment|
|
|
|
|
deployment.run_after_commit do
|
2021-04-29 14:10:23 -04:00
|
|
|
Deployments::HooksWorker.perform_async(deployment_id: id, status_changed_at: Time.current)
|
2020-02-17 04:08:52 -05:00
|
|
|
end
|
|
|
|
end
|
2021-01-06 07:10:58 -05:00
|
|
|
|
|
|
|
after_transition any => any - [:skipped] do |deployment, transition|
|
|
|
|
next if transition.loopback?
|
|
|
|
|
|
|
|
deployment.run_after_commit do
|
|
|
|
::JiraConnect::SyncDeploymentsWorker.perform_async(id)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
after_create unless: :importing? do |deployment|
|
|
|
|
run_after_commit do
|
|
|
|
::JiraConnect::SyncDeploymentsWorker.perform_async(deployment.id)
|
|
|
|
end
|
2018-11-04 19:37:40 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
enum status: {
|
|
|
|
created: 0,
|
|
|
|
running: 1,
|
|
|
|
success: 2,
|
|
|
|
failed: 3,
|
2020-11-18 10:09:08 -05:00
|
|
|
canceled: 4,
|
|
|
|
skipped: 5
|
2018-11-04 19:37:40 -05:00
|
|
|
}
|
|
|
|
|
2018-10-12 10:10:34 -04:00
|
|
|
def self.last_for_environment(environment)
|
|
|
|
ids = self
|
|
|
|
.for_environment(environment)
|
|
|
|
.select('MAX(id) AS id')
|
|
|
|
.group(:environment_id)
|
|
|
|
.map(&:id)
|
|
|
|
find(ids)
|
|
|
|
end
|
|
|
|
|
2019-10-31 17:06:28 -04:00
|
|
|
def self.distinct_on_environment
|
|
|
|
order('environment_id, deployments.id DESC')
|
|
|
|
.select('DISTINCT ON (environment_id) deployments.*')
|
|
|
|
end
|
|
|
|
|
2019-10-16 14:08:01 -04:00
|
|
|
def self.find_successful_deployment!(iid)
|
|
|
|
success.find_by!(iid: iid)
|
|
|
|
end
|
|
|
|
|
2020-03-25 02:07:58 -04:00
|
|
|
class << self
|
|
|
|
##
|
|
|
|
# FastDestroyAll concerns
|
|
|
|
def begin_fast_destroy
|
|
|
|
preload(:project).find_each.map do |deployment|
|
|
|
|
[deployment.project, deployment.ref_path]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
##
|
|
|
|
# FastDestroyAll concerns
|
|
|
|
def finalize_fast_destroy(params)
|
|
|
|
by_project = params.group_by(&:shift)
|
|
|
|
|
|
|
|
by_project.each do |project, ref_paths|
|
|
|
|
project.repository.delete_refs(*ref_paths.flatten)
|
|
|
|
end
|
|
|
|
end
|
2020-11-20 01:09:10 -05:00
|
|
|
|
|
|
|
def latest_for_sha(sha)
|
|
|
|
where(sha: sha).order(id: :desc).take
|
|
|
|
end
|
2020-03-25 02:07:58 -04:00
|
|
|
end
|
|
|
|
|
2016-06-10 17:36:54 -04:00
|
|
|
def commit
|
2021-04-15 14:09:01 -04:00
|
|
|
@commit ||= project.commit(sha)
|
2016-06-10 17:36:54 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def commit_title
|
|
|
|
commit.try(:title)
|
|
|
|
end
|
|
|
|
|
|
|
|
def short_sha
|
2016-06-14 12:34:48 -04:00
|
|
|
Commit.truncate_sha(sha)
|
2016-06-10 17:36:54 -04:00
|
|
|
end
|
2016-06-14 08:47:00 -04:00
|
|
|
|
2021-04-29 14:10:23 -04:00
|
|
|
def execute_hooks(status_changed_at)
|
|
|
|
deployment_data = Gitlab::DataBuilder::Deployment.build(self, status_changed_at)
|
2020-08-24 08:10:17 -04:00
|
|
|
project.execute_hooks(deployment_data, :deployment_hooks)
|
2021-06-30 02:07:17 -04:00
|
|
|
project.execute_integrations(deployment_data, :deployment_hooks)
|
2019-04-26 17:08:41 -04:00
|
|
|
end
|
|
|
|
|
2016-06-14 08:47:00 -04:00
|
|
|
def last?
|
|
|
|
self == environment.last_deployment
|
|
|
|
end
|
2016-06-20 13:22:08 -04:00
|
|
|
|
2016-09-30 12:40:56 -04:00
|
|
|
def create_ref
|
2020-03-02 07:07:57 -05:00
|
|
|
project.repository.create_ref(sha, ref_path)
|
2016-06-20 13:22:08 -04:00
|
|
|
end
|
2016-07-16 12:39:58 -04:00
|
|
|
|
2017-05-23 04:43:55 -04:00
|
|
|
def invalidate_cache
|
|
|
|
environment.expire_etag_cache
|
|
|
|
end
|
|
|
|
|
2016-07-16 17:06:34 -04:00
|
|
|
def manual_actions
|
2018-10-04 04:52:36 -04:00
|
|
|
@manual_actions ||= deployable.try(:other_manual_actions)
|
|
|
|
end
|
|
|
|
|
|
|
|
def scheduled_actions
|
|
|
|
@scheduled_actions ||= deployable.try(:other_scheduled_actions)
|
2016-07-16 12:39:58 -04:00
|
|
|
end
|
2016-08-02 08:01:22 -04:00
|
|
|
|
2019-11-29 04:06:31 -05:00
|
|
|
def playable_build
|
|
|
|
strong_memoize(:playable_build) do
|
|
|
|
deployable.try(:playable?) ? deployable : nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-08-09 09:11:14 -04:00
|
|
|
def includes_commit?(commit)
|
2016-08-02 08:01:22 -04:00
|
|
|
return false unless commit
|
|
|
|
|
2018-01-23 12:42:10 -05:00
|
|
|
project.repository.ancestor?(commit.id, sha)
|
2016-08-02 08:01:22 -04:00
|
|
|
end
|
2016-09-20 05:36:54 -04:00
|
|
|
|
2016-09-20 15:17:37 -04:00
|
|
|
def update_merge_request_metrics!
|
2021-03-30 11:11:08 -04:00
|
|
|
return unless environment.production? && success?
|
2016-09-20 15:17:37 -04:00
|
|
|
|
2017-06-21 09:48:12 -04:00
|
|
|
merge_requests = project.merge_requests
|
|
|
|
.joins(:metrics)
|
|
|
|
.where(target_branch: self.ref, merge_request_metrics: { first_deployed_to_production_at: nil })
|
2018-11-04 19:37:40 -05:00
|
|
|
.where("merge_request_metrics.merged_at <= ?", finished_at)
|
2016-09-20 05:36:54 -04:00
|
|
|
|
2016-09-20 15:17:37 -04:00
|
|
|
if previous_deployment
|
2018-11-04 19:37:40 -05:00
|
|
|
merge_requests = merge_requests.where("merge_request_metrics.merged_at >= ?", previous_deployment.finished_at)
|
2016-09-20 05:36:54 -04:00
|
|
|
end
|
2016-09-20 15:17:37 -04:00
|
|
|
|
2017-06-21 09:48:12 -04:00
|
|
|
MergeRequest::Metrics
|
2019-06-13 09:12:28 -04:00
|
|
|
.where(merge_request_id: merge_requests.select(:id), first_deployed_to_production_at: nil)
|
2018-11-04 19:37:40 -05:00
|
|
|
.update_all(first_deployed_to_production_at: finished_at)
|
2016-09-20 05:36:54 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def previous_deployment
|
|
|
|
@previous_deployment ||=
|
2021-03-24 11:09:19 -04:00
|
|
|
self.class.for_environment(environment_id)
|
2021-04-09 05:09:10 -04:00
|
|
|
.success
|
|
|
|
.where('id < ?', id)
|
|
|
|
.order(id: :desc)
|
|
|
|
.take
|
2016-09-20 05:36:54 -04:00
|
|
|
end
|
2016-09-30 09:45:27 -04:00
|
|
|
|
2016-10-17 06:45:31 -04:00
|
|
|
def stop_action
|
2017-04-06 09:19:52 -04:00
|
|
|
return unless on_stop.present?
|
|
|
|
return unless manual_actions
|
2016-10-06 07:10:50 -04:00
|
|
|
|
2021-04-15 14:09:01 -04:00
|
|
|
@stop_action ||= manual_actions.find { |action| action.name == self.on_stop }
|
2016-10-06 07:10:50 -04:00
|
|
|
end
|
|
|
|
|
2018-11-04 19:37:40 -05:00
|
|
|
def finished_at
|
|
|
|
read_attribute(:finished_at) || legacy_finished_at
|
|
|
|
end
|
|
|
|
|
|
|
|
def deployed_at
|
|
|
|
return unless success?
|
|
|
|
|
|
|
|
finished_at
|
|
|
|
end
|
|
|
|
|
2016-10-12 09:21:01 -04:00
|
|
|
def formatted_deployment_time
|
2018-11-04 19:37:40 -05:00
|
|
|
deployed_at&.to_time&.in_time_zone&.to_s(:medium)
|
2016-10-12 09:21:01 -04:00
|
|
|
end
|
|
|
|
|
2019-08-20 06:29:16 -04:00
|
|
|
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.
|
2019-09-18 10:02:45 -04:00
|
|
|
# TODO: use deployment's user once https://gitlab.com/gitlab-org/gitlab-foss/issues/66442
|
2019-08-20 06:29:16 -04:00
|
|
|
# is completed.
|
|
|
|
deployable&.user || user
|
|
|
|
end
|
|
|
|
|
2019-11-06 19:06:18 -05:00
|
|
|
def link_merge_requests(relation)
|
2020-02-28 10:09:13 -05:00
|
|
|
# NOTE: relation.select will perform column deduplication,
|
|
|
|
# when id == environment_id it will outputs 2 columns instead of 3
|
|
|
|
# i.e.:
|
|
|
|
# MergeRequest.select(1, 2).to_sql #=> SELECT 1, 2 FROM "merge_requests"
|
|
|
|
# MergeRequest.select(1, 1).to_sql #=> SELECT 1 FROM "merge_requests"
|
|
|
|
select = relation.select('merge_requests.id',
|
|
|
|
"#{id} as deployment_id",
|
|
|
|
"#{environment_id} as environment_id").to_sql
|
2019-11-06 19:06:18 -05:00
|
|
|
|
|
|
|
# We don't use `Gitlab::Database.bulk_insert` here so that we don't need to
|
|
|
|
# first pluck lots of IDs into memory.
|
2020-01-03 13:07:40 -05:00
|
|
|
#
|
|
|
|
# We also ignore any duplicates so this method can be called multiple times
|
|
|
|
# for the same deployment, only inserting any missing merge requests.
|
2019-11-06 19:06:18 -05:00
|
|
|
DeploymentMergeRequest.connection.execute(<<~SQL)
|
|
|
|
INSERT INTO #{DeploymentMergeRequest.table_name}
|
2020-02-28 10:09:13 -05:00
|
|
|
(merge_request_id, deployment_id, environment_id)
|
2019-11-06 19:06:18 -05:00
|
|
|
#{select}
|
2020-01-03 13:07:40 -05:00
|
|
|
ON CONFLICT DO NOTHING
|
2019-11-06 19:06:18 -05:00
|
|
|
SQL
|
|
|
|
end
|
|
|
|
|
2020-10-05 02:08:45 -04:00
|
|
|
# Changes the status of a deployment and triggers the corresponding state
|
2019-12-14 10:07:56 -05:00
|
|
|
# machine events.
|
|
|
|
def update_status(status)
|
|
|
|
case status
|
|
|
|
when 'running'
|
|
|
|
run
|
|
|
|
when 'success'
|
|
|
|
succeed
|
|
|
|
when 'failed'
|
|
|
|
drop
|
|
|
|
when 'canceled'
|
|
|
|
cancel
|
2020-11-18 10:09:08 -05:00
|
|
|
when 'skipped'
|
|
|
|
skip
|
2019-12-14 10:07:56 -05:00
|
|
|
else
|
|
|
|
raise ArgumentError, "The status #{status.inspect} is invalid"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-12-20 22:07:37 -05:00
|
|
|
def valid_sha
|
|
|
|
return if project&.commit(sha)
|
|
|
|
|
|
|
|
errors.add(:sha, _('The commit does not exist'))
|
|
|
|
end
|
|
|
|
|
|
|
|
def valid_ref
|
|
|
|
return if project&.commit(ref)
|
|
|
|
|
|
|
|
errors.add(:ref, _('The branch or tag does not exist'))
|
|
|
|
end
|
|
|
|
|
2016-09-30 09:45:27 -04:00
|
|
|
def ref_path
|
2016-10-19 08:49:09 -04:00
|
|
|
File.join(environment.ref_path, 'deployments', iid.to_s)
|
2016-09-30 09:45:27 -04:00
|
|
|
end
|
2018-11-04 19:37:40 -05:00
|
|
|
|
2021-01-19 13:11:04 -05:00
|
|
|
def equal_to?(params)
|
|
|
|
ref == params[:ref] &&
|
|
|
|
tag == params[:tag] &&
|
|
|
|
sha == params[:sha] &&
|
|
|
|
status == params[:status]
|
|
|
|
end
|
|
|
|
|
2020-03-25 02:07:58 -04:00
|
|
|
private
|
|
|
|
|
2018-11-04 19:37:40 -05:00
|
|
|
def legacy_finished_at
|
|
|
|
self.created_at if success? && !read_attribute(:finished_at)
|
|
|
|
end
|
2016-06-10 17:36:54 -04:00
|
|
|
end
|
2019-10-03 20:06:03 -04:00
|
|
|
|
2021-05-11 17:10:21 -04:00
|
|
|
Deployment.prepend_mod_with('Deployment')
|