Class: Environment

Inherits:
ApplicationRecord show all
Includes:
FastDestroyAll::Helpers, Gitlab::Utils::StrongMemoize, ReactiveCaching
Defined in:
app/models/environment.rb

Constant Summary

Constants included from ReactiveCaching

ReactiveCaching::ExceededReactiveCacheLimit, ReactiveCaching::InvalidateReactiveCache, ReactiveCaching::WORK_TYPE

Class Method Summary collapse

Instance Method Summary collapse

Methods included from FastDestroyAll::Helpers

#perform_fast_destroy

Methods included from AfterCommitQueue

#run_after_commit, #run_after_commit_or_now

Methods included from Gitlab::Utils::StrongMemoize

#clear_memoization, #strong_memoize, #strong_memoized?

Methods inherited from ApplicationRecord

at_most, id_in, id_not_in, iid_in, pluck_primary_key, primary_key_in, safe_ensure_unique, safe_find_or_create_by, safe_find_or_create_by!, underscore, without_order

Class Method Details

.count_by_stateObject


157
158
159
160
161
162
163
# File 'app/models/environment.rb', line 157

def count_by_state
  environments_count_by_state = group(:state).count

  valid_states.each_with_object({}) do |state, count_hash|
    count_hash[state] = environments_count_by_state[state.to_s] || 0
  end
end

.find_or_create_by_name(name) ⇒ Object


121
122
123
# File 'app/models/environment.rb', line 121

def self.find_or_create_by_name(name)
  find_or_create_by(name: name)
end

.for_id_and_slug(id, slug) ⇒ Object


107
108
109
# File 'app/models/environment.rb', line 107

def self.for_id_and_slug(id, slug)
  find_by(id: id, slug: slug)
end

.max_deployment_id_sqlObject


111
112
113
114
115
# File 'app/models/environment.rb', line 111

def self.max_deployment_id_sql
  Deployment.select(Deployment.arel_table[:id].maximum)
  .where(Deployment.arel_table[:environment_id].eq(arel_table[:id]))
  .to_sql
end

.pluck_namesObject


117
118
119
# File 'app/models/environment.rb', line 117

def self.pluck_names
  pluck(:name)
end

.stop_actionsObject

This method returns stop actions (jobs) for multiple environments within one query. It's useful to avoid N+1 problem.

NOTE: The count of environments should be small~medium (e.g. < 5000)


135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
# File 'app/models/environment.rb', line 135

def stop_actions
  cte = cte_for_deployments_with_stop_action
  ci_builds = Ci::Build.arel_table

  inner_join_stop_actions = ci_builds.join(cte.table).on(
    ci_builds[:project_id].eq(cte.table[:project_id])
      .and(ci_builds[:ref].eq(cte.table[:ref]))
      .and(ci_builds[:name].eq(cte.table[:on_stop]))
  ).join_sources

  pipeline_ids = ci_builds.join(cte.table).on(
    ci_builds[:id].eq(cte.table[:deployable_id])
  ).project(:commit_id)

  Ci::Build.joins(inner_join_stop_actions)
           .with(cte.to_arel)
           .where(ci_builds[:commit_id].in(pipeline_ids))
           .where(status: Ci::HasStatus::BLOCKED_STATUS)
           .preload_project_and_pipeline_project
           .preload(:user, :metadata, :deployment)
end

.valid_statesObject


125
126
127
# File 'app/models/environment.rb', line 125

def self.valid_states
  self.state_machine.states.map(&:name)
end

Instance Method Details

#actions_for(environment) ⇒ Object


257
258
259
260
261
262
263
# File 'app/models/environment.rb', line 257

def actions_for(environment)
  return [] unless manual_actions

  manual_actions.select do |action|
    action.expanded_environment_name == environment
  end
end

#additional_metrics(*args) ⇒ Object


307
308
309
310
311
# File 'app/models/environment.rb', line 307

def additional_metrics(*args)
  return unless has_metrics_and_can_query?

  prometheus_adapter.query(:additional_metrics_environment, self, *args.map(&:to_f))
end

#auto_stop_inObject


362
363
364
# File 'app/models/environment.rb', line 362

def auto_stop_in
  auto_stop_at - Time.current if auto_stop_at
end

#auto_stop_in=(value) ⇒ Object


366
367
368
369
370
371
# File 'app/models/environment.rb', line 366

def auto_stop_in=(value)
  return unless value
  return unless parsed_result = ChronicDuration.parse(value)

  self.auto_stop_at = parsed_result.seconds.from_now
end

#calculate_reactive_cacheObject


275
276
277
278
279
# File 'app/models/environment.rb', line 275

def calculate_reactive_cache
  return unless has_terminals? && !project.pending_delete?

  deployment_platform.calculate_reactive_cache_for(self)
end

#cancel_deployment_jobs!Object


231
232
233
234
235
236
237
238
239
240
241
242
243
244
# File 'app/models/environment.rb', line 231

def cancel_deployment_jobs!
  jobs = active_deployments.with_deployable
  jobs.each do |deployment|
    # guard against data integrity issues,
    # for example https://gitlab.com/gitlab-org/gitlab/-/issues/218659#note_348823660
    next unless deployment.deployable

    Gitlab::OptimisticLocking.retry_lock(deployment.deployable) do |deployable|
      deployable.cancel! if deployable&.cancelable?
    end
  rescue => e
    Gitlab::ErrorTracking.track_exception(e, environment_id: id, deployment_id: deployment.id)
  end
end

#clear_prometheus_reactive_cache!(query_name) ⇒ Object


175
176
177
# File 'app/models/environment.rb', line 175

def clear_prometheus_reactive_cache!(query_name)
  cluster_prometheus_adapter&.clear_prometheus_reactive_cache!(query_name, self)
end

#cluster_prometheus_adapterObject


179
180
181
# File 'app/models/environment.rb', line 179

def cluster_prometheus_adapter
  @cluster_prometheus_adapter ||= ::Gitlab::Prometheus::Adapter.new(project, deployment_platform&.cluster).cluster_prometheus_adapter
end

#deployment_namespaceObject


281
282
283
284
285
# File 'app/models/environment.rb', line 281

def deployment_namespace
  strong_memoize(:kubernetes_namespace) do
    deployment_platform.cluster.kubernetes_namespace_for(self) if deployment_platform
  end
end

#deployment_platformObject


350
351
352
353
354
# File 'app/models/environment.rb', line 350

def deployment_platform
  strong_memoize(:deployment_platform) do
    project.deployment_platform(environment: self.name)
  end
end

#elastic_stack_available?Boolean

Returns:

  • (Boolean)

373
374
375
# File 'app/models/environment.rb', line 373

def elastic_stack_available?
  !!deployment_platform&.cluster&.application_elastic_stack_available?
end

#etag_cache_keyObject


336
337
338
339
340
# File 'app/models/environment.rb', line 336

def etag_cache_key
  Gitlab::Routing.url_helpers.project_environments_path(
    project,
    format: :json)
end

#expire_etag_cacheObject


330
331
332
333
334
# File 'app/models/environment.rb', line 330

def expire_etag_cache
  Gitlab::EtagCaching::Store.new.tap do |store|
    store.touch(etag_cache_key)
  end
end

#external_url_for(path, commit_sha) ⇒ Object


321
322
323
324
325
326
327
328
# File 'app/models/environment.rb', line 321

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

  [external_url.delete_suffix('/'), public_path.delete_prefix('/')].join('/')
end

#folder_nameObject


342
343
344
# File 'app/models/environment.rb', line 342

def folder_name
  self.environment_type || self.name
end

#for_name_likeObject

Search environments which have names like the given query. Do not set a large limit unless you've confirmed that it works on gitlab.com scale.


79
80
81
# File 'app/models/environment.rb', line 79

scope :for_name_like, -> (query, limit: 5) do
  where(arel_table[:name].matches("#{sanitize_sql_like query}%")).limit(limit)
end

#formatted_external_urlObject


221
222
223
224
225
# File 'app/models/environment.rb', line 221

def formatted_external_url
  return unless external_url

  external_url.gsub(%r{\A.*?://}, '')
end

#has_metrics?Boolean

Returns:

  • (Boolean)

287
288
289
# File 'app/models/environment.rb', line 287

def has_metrics?
  available? && (prometheus_adapter&.configured? || has_sample_metrics?)
end

#has_opened_alert?Boolean

Returns:

  • (Boolean)

295
296
297
# File 'app/models/environment.rb', line 295

def has_opened_alert?
  latest_opened_most_severe_alert.present?
end

#has_sample_metrics?Boolean

Returns:

  • (Boolean)

291
292
293
# File 'app/models/environment.rb', line 291

def has_sample_metrics?
  !!ENV['USE_SAMPLE_METRICS']
end

#has_terminals?Boolean

Returns:

  • (Boolean)

265
266
267
# File 'app/models/environment.rb', line 265

def has_terminals?
  available? && deployment_platform.present? && last_deployment.present?
end

#includes_commit?(commit) ⇒ Boolean

Returns:

  • (Boolean)

203
204
205
206
207
# File 'app/models/environment.rb', line 203

def includes_commit?(commit)
  return false unless last_deployment

  last_deployment.includes_commit?(commit)
end

#knative_services_finderObject


356
357
358
359
360
# File 'app/models/environment.rb', line 356

def knative_services_finder
  if last_deployment&.cluster
    Clusters::KnativeServicesFinder.new(last_deployment.cluster, self)
  end
end

#last_deployed_atObject


209
210
211
# File 'app/models/environment.rb', line 209

def last_deployed_at
  last_deployment.try(:created_at)
end

#metricsObject


299
300
301
# File 'app/models/environment.rb', line 299

def metrics
  prometheus_adapter.query(:environment, self) if has_metrics_and_can_query?
end

#name_without_typeObject


346
347
348
# File 'app/models/environment.rb', line 346

def name_without_type
  @name_without_type ||= name.delete_prefix("#{environment_type}/")
end

#nullify_external_urlObject


193
194
195
# File 'app/models/environment.rb', line 193

def nullify_external_url
  self.external_url = nil if self.external_url.blank?
end

#predefined_variablesObject


183
184
185
186
187
# File 'app/models/environment.rb', line 183

def predefined_variables
  Gitlab::Ci::Variables::Collection.new
    .append(key: 'CI_ENVIRONMENT_NAME', value: name)
    .append(key: 'CI_ENVIRONMENT_SLUG', value: slug)
end

#prometheus_adapterObject


313
314
315
# File 'app/models/environment.rb', line 313

def prometheus_adapter
  @prometheus_adapter ||= Gitlab::Prometheus::Adapter.new(project, deployment_platform&.cluster).prometheus_adapter
end

#prometheus_statusObject


303
304
305
# File 'app/models/environment.rb', line 303

def prometheus_status
  deployment_platform&.cluster&.application_prometheus&.status_name
end

#recently_updated_on_branch?(ref) ⇒ Boolean

Returns:

  • (Boolean)

189
190
191
# File 'app/models/environment.rb', line 189

def recently_updated_on_branch?(ref)
  ref.to_s == last_deployment.try(:ref)
end

#ref_pathObject


217
218
219
# File 'app/models/environment.rb', line 217

def ref_path
  "refs/#{Repository::REF_ENVIRONMENTS}/#{slug}"
end

#reset_auto_stopObject


253
254
255
# File 'app/models/environment.rb', line 253

def reset_auto_stop
  update_column(:auto_stop_at, nil)
end

#set_environment_typeObject


197
198
199
200
201
# File 'app/models/environment.rb', line 197

def set_environment_type
  names = name.split('/')

  self.environment_type = names.many? ? names.first : nil
end

#slugObject


317
318
319
# File 'app/models/environment.rb', line 317

def slug
  super.presence || generate_slug
end

#stop_action_available?Boolean

Returns:

  • (Boolean)

227
228
229
# File 'app/models/environment.rb', line 227

def stop_action_available?
  available? && stop_action.present?
end

#stop_with_action!(current_user) ⇒ Object


246
247
248
249
250
251
# File 'app/models/environment.rb', line 246

def stop_with_action!(current_user)
  return unless available?

  stop!
  stop_action&.play(current_user)
end

#terminalsObject


269
270
271
272
273
# File 'app/models/environment.rb', line 269

def terminals
  with_reactive_cache do |data|
    deployment_platform.terminals(self, data)
  end
end

#update_merge_request_metrics?Boolean

Returns:

  • (Boolean)

213
214
215
# File 'app/models/environment.rb', line 213

def update_merge_request_metrics?
  folder_name == "production"
end