2016-06-10 17:36:54 -04:00
|
|
|
class Environment < ActiveRecord::Base
|
2016-12-07 20:09:18 -05:00
|
|
|
# Used to generate random suffixes for the slug
|
2017-01-19 09:42:03 -05:00
|
|
|
LETTERS = 'a'..'z'
|
2016-12-07 20:09:18 -05:00
|
|
|
NUMBERS = '0'..'9'
|
2017-01-19 09:42:03 -05:00
|
|
|
SUFFIX_CHARS = LETTERS.to_a + NUMBERS.to_a
|
2016-12-07 20:09:18 -05:00
|
|
|
|
2016-06-15 09:09:50 -04:00
|
|
|
belongs_to :project, required: true, validate: true
|
2016-06-10 17:36:54 -04:00
|
|
|
|
2017-09-18 18:37:38 -04:00
|
|
|
has_many :deployments, dependent: :destroy # rubocop:disable Cop/ActiveRecordDependent
|
2017-09-13 08:43:03 -04:00
|
|
|
|
2017-01-30 19:26:40 -05:00
|
|
|
has_one :last_deployment, -> { order('deployments.id DESC') }, class_name: 'Deployment'
|
2016-06-10 17:36:54 -04:00
|
|
|
|
2016-07-26 08:19:37 -04:00
|
|
|
before_validation :nullify_external_url
|
2016-12-07 20:09:18 -05:00
|
|
|
before_validation :generate_slug, if: ->(env) { env.slug.blank? }
|
|
|
|
|
2016-09-13 08:14:55 -04:00
|
|
|
before_save :set_environment_type
|
2016-07-26 08:19:37 -04:00
|
|
|
|
2016-06-14 07:04:21 -04:00
|
|
|
validates :name,
|
|
|
|
presence: true,
|
2016-06-14 12:34:48 -04:00
|
|
|
uniqueness: { scope: :project_id },
|
2016-12-02 07:54:57 -05:00
|
|
|
length: { maximum: 255 },
|
2016-06-14 07:04:21 -04:00
|
|
|
format: { with: Gitlab::Regex.environment_name_regex,
|
|
|
|
message: Gitlab::Regex.environment_name_regex_message }
|
2016-06-10 17:36:54 -04:00
|
|
|
|
2016-12-07 20:09:18 -05:00
|
|
|
validates :slug,
|
|
|
|
presence: true,
|
|
|
|
uniqueness: { scope: :project_id },
|
|
|
|
length: { maximum: 24 },
|
|
|
|
format: { with: Gitlab::Regex.environment_slug_regex,
|
|
|
|
message: Gitlab::Regex.environment_slug_regex_message }
|
|
|
|
|
2016-07-26 03:35:47 -04:00
|
|
|
validates :external_url,
|
2016-07-26 08:19:37 -04:00
|
|
|
length: { maximum: 255 },
|
|
|
|
allow_nil: true,
|
|
|
|
addressable_url: true
|
2016-07-26 03:35:47 -04:00
|
|
|
|
2016-11-21 07:47:18 -05:00
|
|
|
delegate :stop_action, :manual_actions, to: :last_deployment, allow_nil: true
|
2016-10-06 07:10:50 -04:00
|
|
|
|
2016-10-17 15:06:10 -04:00
|
|
|
scope :available, -> { with_state(:available) }
|
|
|
|
scope :stopped, -> { with_state(:stopped) }
|
2017-02-06 19:06:46 -05:00
|
|
|
scope :order_by_last_deployed_at, -> do
|
|
|
|
max_deployment_id_sql =
|
2017-06-21 09:48:12 -04:00
|
|
|
Deployment.select(Deployment.arel_table[:id].maximum)
|
|
|
|
.where(Deployment.arel_table[:environment_id].eq(arel_table[:id]))
|
|
|
|
.to_sql
|
2017-02-06 19:06:46 -05:00
|
|
|
order(Gitlab::Database.nulls_first_order("(#{max_deployment_id_sql})", 'ASC'))
|
|
|
|
end
|
2017-06-15 06:50:45 -04:00
|
|
|
scope :in_review_folder, -> { where(environment_type: "review") }
|
2016-10-17 15:06:10 -04:00
|
|
|
|
2016-10-17 06:45:31 -04:00
|
|
|
state_machine :state, initial: :available do
|
|
|
|
event :start do
|
|
|
|
transition stopped: :available
|
2016-10-06 07:10:50 -04:00
|
|
|
end
|
|
|
|
|
2016-10-17 06:45:31 -04:00
|
|
|
event :stop do
|
|
|
|
transition available: :stopped
|
2016-10-06 07:10:50 -04:00
|
|
|
end
|
|
|
|
|
2016-10-17 06:45:31 -04:00
|
|
|
state :available
|
|
|
|
state :stopped
|
2017-05-12 09:19:27 -04:00
|
|
|
|
|
|
|
after_transition do |environment|
|
|
|
|
environment.expire_etag_cache
|
|
|
|
end
|
2016-10-06 07:10:50 -04:00
|
|
|
end
|
|
|
|
|
2016-12-08 11:21:16 -05:00
|
|
|
def predefined_variables
|
|
|
|
[
|
|
|
|
{ key: 'CI_ENVIRONMENT_NAME', value: name, public: true },
|
2017-05-03 07:22:03 -04:00
|
|
|
{ key: 'CI_ENVIRONMENT_SLUG', value: slug, public: true }
|
2016-12-08 11:21:16 -05:00
|
|
|
]
|
|
|
|
end
|
|
|
|
|
2016-11-16 04:54:20 -05:00
|
|
|
def recently_updated_on_branch?(ref)
|
2016-11-16 05:26:36 -05:00
|
|
|
ref.to_s == last_deployment.try(:ref)
|
2016-11-15 08:48:14 -05:00
|
|
|
end
|
|
|
|
|
2016-07-26 08:19:37 -04:00
|
|
|
def nullify_external_url
|
|
|
|
self.external_url = nil if self.external_url.blank?
|
|
|
|
end
|
2016-08-02 08:01:22 -04:00
|
|
|
|
2016-09-13 08:14:55 -04:00
|
|
|
def set_environment_type
|
|
|
|
names = name.split('/')
|
|
|
|
|
2017-08-21 11:46:45 -04:00
|
|
|
self.environment_type = names.many? ? names.first : nil
|
2016-09-13 08:14:55 -04:00
|
|
|
end
|
|
|
|
|
2016-08-09 09:11:14 -04:00
|
|
|
def includes_commit?(commit)
|
2016-08-03 07:37:39 -04:00
|
|
|
return false unless last_deployment
|
2016-08-02 08:01:22 -04:00
|
|
|
|
2016-08-09 09:11:14 -04:00
|
|
|
last_deployment.includes_commit?(commit)
|
2016-08-02 08:01:22 -04:00
|
|
|
end
|
2016-09-20 05:36:54 -04:00
|
|
|
|
2017-01-30 19:26:40 -05:00
|
|
|
def last_deployed_at
|
|
|
|
last_deployment.try(:created_at)
|
|
|
|
end
|
|
|
|
|
2016-09-20 05:36:54 -04:00
|
|
|
def update_merge_request_metrics?
|
2017-08-21 11:46:45 -04:00
|
|
|
folder_name == "production"
|
2016-09-20 05:36:54 -04:00
|
|
|
end
|
2016-09-30 09:45:27 -04:00
|
|
|
|
2016-10-05 07:52:15 -04:00
|
|
|
def first_deployment_for(commit)
|
2016-10-04 10:03:13 -04:00
|
|
|
ref = project.repository.ref_name_for_sha(ref_path, commit.sha)
|
|
|
|
|
|
|
|
return nil unless ref
|
|
|
|
|
2016-10-19 08:49:09 -04:00
|
|
|
deployment_iid = ref.split('/').last
|
|
|
|
deployments.find_by(iid: deployment_iid)
|
2016-10-04 10:03:13 -04:00
|
|
|
end
|
|
|
|
|
2016-09-30 09:45:27 -04:00
|
|
|
def ref_path
|
2017-11-02 18:32:22 -04:00
|
|
|
"refs/#{Repository::REF_ENVIRONMENTS}/#{slug}"
|
2016-09-30 09:45:27 -04:00
|
|
|
end
|
2016-10-12 09:21:01 -04:00
|
|
|
|
|
|
|
def formatted_external_url
|
|
|
|
return nil unless external_url
|
|
|
|
|
|
|
|
external_url.gsub(/\A.*?:\/\//, '')
|
|
|
|
end
|
2016-10-17 10:13:19 -04:00
|
|
|
|
2017-02-06 10:50:03 -05:00
|
|
|
def stop_action?
|
2016-10-17 10:13:19 -04:00
|
|
|
available? && stop_action.present?
|
|
|
|
end
|
2016-10-18 06:02:50 -04:00
|
|
|
|
2017-02-06 10:50:03 -05:00
|
|
|
def stop_with_action!(current_user)
|
2016-11-10 07:59:26 -05:00
|
|
|
return unless available?
|
2016-10-18 06:02:50 -04:00
|
|
|
|
2017-02-06 10:50:03 -05:00
|
|
|
stop!
|
2017-02-13 07:02:53 -05:00
|
|
|
stop_action&.play(current_user)
|
2016-10-18 06:02:50 -04:00
|
|
|
end
|
2016-11-21 07:47:18 -05:00
|
|
|
|
|
|
|
def actions_for(environment)
|
|
|
|
return [] unless manual_actions
|
|
|
|
|
2016-11-21 11:26:35 -05:00
|
|
|
manual_actions.select do |action|
|
|
|
|
action.expanded_environment_name == environment
|
2016-11-10 07:59:26 -05:00
|
|
|
end
|
2016-10-18 06:02:50 -04:00
|
|
|
end
|
2016-12-07 20:09:18 -05:00
|
|
|
|
2016-11-22 14:55:56 -05:00
|
|
|
def has_terminals?
|
|
|
|
project.deployment_service.present? && available? && last_deployment.present?
|
|
|
|
end
|
|
|
|
|
|
|
|
def terminals
|
|
|
|
project.deployment_service.terminals(self) if has_terminals?
|
|
|
|
end
|
|
|
|
|
2017-03-07 11:57:42 -05:00
|
|
|
def has_metrics?
|
|
|
|
project.monitoring_service.present? && available? && last_deployment.present?
|
|
|
|
end
|
|
|
|
|
|
|
|
def metrics
|
2017-04-26 16:09:03 -04:00
|
|
|
project.monitoring_service.environment_metrics(self) if has_metrics?
|
2017-03-07 11:57:42 -05:00
|
|
|
end
|
|
|
|
|
2017-06-06 20:36:59 -04:00
|
|
|
def has_additional_metrics?
|
2017-06-16 14:48:34 -04:00
|
|
|
project.prometheus_service.present? && available? && last_deployment.present?
|
2017-06-06 20:36:59 -04:00
|
|
|
end
|
|
|
|
|
2017-05-10 05:25:30 -04:00
|
|
|
def additional_metrics
|
2017-05-25 08:01:10 -04:00
|
|
|
if has_additional_metrics?
|
2017-06-16 14:48:34 -04:00
|
|
|
project.prometheus_service.additional_environment_metrics(self)
|
2017-05-25 08:01:10 -04:00
|
|
|
end
|
2017-05-10 05:25:30 -04:00
|
|
|
end
|
|
|
|
|
2017-11-02 18:32:22 -04:00
|
|
|
def slug
|
|
|
|
super.presence || generate_slug
|
|
|
|
end
|
|
|
|
|
2016-12-07 20:09:18 -05:00
|
|
|
# An environment name is not necessarily suitable for use in URLs, DNS
|
|
|
|
# or other third-party contexts, so provide a slugified version. A slug has
|
|
|
|
# the following properties:
|
|
|
|
# * contains only lowercase letters (a-z), numbers (0-9), and '-'
|
|
|
|
# * begins with a letter
|
|
|
|
# * has a maximum length of 24 bytes (OpenShift limitation)
|
|
|
|
# * cannot end with `-`
|
|
|
|
def generate_slug
|
|
|
|
# Lowercase letters and numbers only
|
|
|
|
slugified = name.to_s.downcase.gsub(/[^a-z0-9]/, '-')
|
|
|
|
|
|
|
|
# Must start with a letter
|
2017-01-19 09:42:03 -05:00
|
|
|
slugified = 'env-' + slugified unless LETTERS.cover?(slugified[0])
|
|
|
|
|
|
|
|
# Repeated dashes are invalid (OpenShift limitation)
|
|
|
|
slugified.gsub!(/\-+/, '-')
|
2016-12-07 20:09:18 -05:00
|
|
|
|
|
|
|
# Maximum length: 24 characters (OpenShift limitation)
|
|
|
|
slugified = slugified[0..23]
|
|
|
|
|
2017-01-19 09:42:03 -05:00
|
|
|
# Cannot end with a dash (Kubernetes label limitation)
|
|
|
|
slugified.chop! if slugified.end_with?('-')
|
2016-12-07 20:09:18 -05:00
|
|
|
|
|
|
|
# Add a random suffix, shortening the current string if necessary, if it
|
|
|
|
# has been slugified. This ensures uniqueness.
|
2017-01-19 09:42:03 -05:00
|
|
|
if slugified != name
|
|
|
|
slugified = slugified[0..16]
|
|
|
|
slugified << '-' unless slugified.end_with?('-')
|
|
|
|
slugified << random_suffix
|
|
|
|
end
|
2016-12-07 20:09:18 -05:00
|
|
|
|
|
|
|
self.slug = slugified
|
|
|
|
end
|
|
|
|
|
2017-01-29 14:38:00 -05:00
|
|
|
def external_url_for(path, commit_sha)
|
|
|
|
return unless self.external_url
|
|
|
|
|
|
|
|
public_path = project.public_path_for_source_path(path, commit_sha)
|
|
|
|
return unless public_path
|
|
|
|
|
2017-01-29 16:31:13 -05:00
|
|
|
[external_url, public_path].join('/')
|
2017-01-29 14:38:00 -05:00
|
|
|
end
|
|
|
|
|
2017-05-12 09:19:27 -04:00
|
|
|
def expire_etag_cache
|
|
|
|
Gitlab::EtagCaching::Store.new.tap do |store|
|
2017-05-30 17:56:26 -04:00
|
|
|
store.touch(etag_cache_key)
|
2017-05-12 09:19:27 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-05-30 17:56:26 -04:00
|
|
|
def etag_cache_key
|
2017-06-29 13:06:35 -04:00
|
|
|
Gitlab::Routing.url_helpers.project_environments_path(
|
2017-06-12 03:09:51 -04:00
|
|
|
project,
|
|
|
|
format: :json)
|
2017-05-30 17:56:26 -04:00
|
|
|
end
|
|
|
|
|
2017-08-21 11:46:45 -04:00
|
|
|
def folder_name
|
|
|
|
self.environment_type || self.name
|
|
|
|
end
|
|
|
|
|
2016-12-07 20:09:18 -05:00
|
|
|
private
|
|
|
|
|
|
|
|
# Slugifying a name may remove the uniqueness guarantee afforded by it being
|
|
|
|
# based on name (which must be unique). To compensate, we add a random
|
|
|
|
# 6-byte suffix in those circumstances. This is not *guaranteed* uniqueness,
|
|
|
|
# but the chance of collisions is vanishingly small
|
|
|
|
def random_suffix
|
|
|
|
(0..5).map { SUFFIX_CHARS.sample }.join
|
|
|
|
end
|
2016-06-10 17:36:54 -04:00
|
|
|
end
|