debian-mirror-gitlab/app/models/deployment.rb

194 lines
4.5 KiB
Ruby
Raw Normal View History

2018-11-18 11:00:15 +05:30
# frozen_string_literal: true
class Deployment < ActiveRecord::Base
2018-10-15 14:42:47 +05:30
include AtomicInternalId
2018-11-08 19:23:39 +05:30
include IidRoutes
2018-12-13 13:39:08 +05:30
include AfterCommitQueue
2018-03-27 19:54:05 +05:30
belongs_to :project, required: true
belongs_to :environment, required: true
belongs_to :user
2017-09-10 17:25:29 +05:30
belongs_to :deployable, polymorphic: true # rubocop:disable Cop/PolymorphicAssociations
2018-12-13 13:39:08 +05:30
has_internal_id :iid, scope: :project, init: ->(s) do
Deployment.where(project: s.project).maximum(:iid) if s&.project
end
2018-10-15 14:42:47 +05:30
validates :sha, presence: true
validates :ref, presence: true
delegate :name, to: :environment, prefix: true
2018-12-13 13:39:08 +05:30
scope :for_environment, -> (environment) { where(environment_id: environment) }
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
end
enum status: {
created: 0,
running: 1,
success: 2,
failed: 3,
canceled: 4
}
def self.last_for_environment(environment)
ids = self
.for_environment(environment)
.select('MAX(id) AS id')
.group(:environment_id)
.map(&:id)
find(ids)
end
2016-08-24 12:49:21 +05:30
def commit
project.commit(sha)
end
def commit_title
commit.try(:title)
end
def short_sha
Commit.truncate_sha(sha)
end
def last?
self == environment.last_deployment
end
2016-08-24 12:49:21 +05:30
2016-11-03 12:29:30 +05:30
def create_ref
project.repository.create_ref(ref, ref_path)
2016-08-24 12:49:21 +05:30
end
2017-09-10 17:25:29 +05:30
def invalidate_cache
environment.expire_etag_cache
end
2016-08-24 12:49:21 +05:30
def manual_actions
2018-12-13 13:39:08 +05:30
@manual_actions ||= deployable.try(:other_manual_actions)
end
def scheduled_actions
@scheduled_actions ||= deployable.try(:other_scheduled_actions)
2016-08-24 12:49:21 +05:30
end
2016-09-13 17:45:13 +05:30
def includes_commit?(commit)
return false unless commit
2018-03-17 18:26:18 +05:30
project.repository.ancestor?(commit.id, sha)
2016-09-13 17:45:13 +05:30
end
2016-09-29 09:46:39 +05:30
def update_merge_request_metrics!
2018-12-13 13:39:08 +05:30
return unless environment.update_merge_request_metrics? && success?
2016-09-29 09:46:39 +05:30
2017-09-10 17:25:29 +05:30
merge_requests = project.merge_requests
.joins(:metrics)
.where(target_branch: self.ref, merge_request_metrics: { first_deployed_to_production_at: nil })
2018-12-13 13:39:08 +05:30
.where("merge_request_metrics.merged_at <= ?", finished_at)
2016-09-29 09:46:39 +05:30
if previous_deployment
2018-12-13 13:39:08 +05:30
merge_requests = merge_requests.where("merge_request_metrics.merged_at >= ?", previous_deployment.finished_at)
2016-09-29 09:46:39 +05:30
end
# 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
2017-09-10 17:25:29 +05:30
MergeRequest::Metrics
.where(merge_request_id: merge_request_ids, first_deployed_to_production_at: nil)
2018-12-13 13:39:08 +05:30
.update_all(first_deployed_to_production_at: finished_at)
2016-09-29 09:46:39 +05:30
end
def previous_deployment
@previous_deployment ||=
2017-09-10 17:25:29 +05:30
project.deployments.joins(:environment)
.where(environments: { name: self.environment.name }, ref: self.ref)
.where.not(id: self.id)
.take
2016-09-29 09:46:39 +05:30
end
2016-11-03 12:29:30 +05:30
def stop_action
2017-08-17 22:00:37 +05:30
return unless on_stop.present?
return unless manual_actions
2016-11-03 12:29:30 +05:30
@stop_action ||= manual_actions.find_by(name: on_stop)
end
2018-12-13 13:39:08 +05:30
def finished_at
read_attribute(:finished_at) || legacy_finished_at
end
def deployed_at
return unless success?
finished_at
end
2016-11-03 12:29:30 +05:30
def formatted_deployment_time
2018-12-13 13:39:08 +05:30
deployed_at&.to_time&.in_time_zone&.to_s(:medium)
2016-11-03 12:29:30 +05:30
end
2017-08-17 22:00:37 +05:30
def has_metrics?
2018-12-13 13:39:08 +05:30
prometheus_adapter&.can_query? && success?
2017-08-17 22:00:37 +05:30
end
2017-09-10 17:25:29 +05:30
def metrics
2017-08-17 22:00:37 +05:30
return {} unless has_metrics?
2018-03-27 19:54:05 +05:30
metrics = prometheus_adapter.query(:deployment, self)
2018-12-13 13:39:08 +05:30
metrics&.merge(deployment_time: finished_at.to_i) || {}
2017-09-10 17:25:29 +05:30
end
def additional_metrics
2018-03-27 19:54:05 +05:30
return {} unless has_metrics?
2017-08-17 22:00:37 +05:30
2018-03-27 19:54:05 +05:30
metrics = prometheus_adapter.query(:additional_metrics_deployment, self)
2018-12-13 13:39:08 +05:30
metrics&.merge(deployment_time: finished_at.to_i) || {}
2017-08-17 22:00:37 +05:30
end
2016-11-03 12:29:30 +05:30
private
2018-03-27 19:54:05 +05:30
def prometheus_adapter
environment.prometheus_adapter
end
2016-11-03 12:29:30 +05:30
def ref_path
File.join(environment.ref_path, 'deployments', iid.to_s)
end
2018-12-13 13:39:08 +05:30
def legacy_finished_at
self.created_at if success? && !read_attribute(:finished_at)
end
end