debian-mirror-gitlab/app/models/clusters/platforms/kubernetes.rb

242 lines
7.5 KiB
Ruby
Raw Normal View History

2018-11-18 11:00:15 +05:30
# frozen_string_literal: true
2018-03-17 18:26:18 +05:30
module Clusters
module Platforms
2019-05-18 00:54:41 +05:30
class Kubernetes < ApplicationRecord
2018-03-17 18:26:18 +05:30
include Gitlab::Kubernetes
include ReactiveCaching
2018-11-20 20:47:30 +05:30
include EnumWithNil
2018-12-13 13:39:08 +05:30
include AfterCommitQueue
RESERVED_NAMESPACES = %w(gitlab-managed-apps).freeze
2018-03-17 18:26:18 +05:30
self.table_name = 'cluster_platforms_kubernetes'
self.reactive_cache_key = ->(kubernetes) { [kubernetes.class.model_name.singular, kubernetes.id] }
belongs_to :cluster, inverse_of: :platform_kubernetes, class_name: 'Clusters::Cluster'
attr_encrypted :password,
mode: :per_attribute_iv,
2018-11-08 19:23:39 +05:30
key: Settings.attr_encrypted_db_key_base_truncated,
2018-03-17 18:26:18 +05:30
algorithm: 'aes-256-cbc'
attr_encrypted :token,
mode: :per_attribute_iv,
2018-11-08 19:23:39 +05:30
key: Settings.attr_encrypted_db_key_base_truncated,
2018-03-17 18:26:18 +05:30
algorithm: 'aes-256-cbc'
before_validation :enforce_namespace_to_lower_case
2018-12-13 13:39:08 +05:30
before_validation :enforce_ca_whitespace_trimming
2018-03-17 18:26:18 +05:30
validates :namespace,
allow_blank: true,
length: 1..63,
format: {
with: Gitlab::Regex.kubernetes_namespace_regex,
message: Gitlab::Regex.kubernetes_namespace_regex_message
}
2018-12-13 13:39:08 +05:30
validates :namespace, exclusion: { in: RESERVED_NAMESPACES }
2019-02-15 15:39:39 +05:30
validate :no_namespace, unless: :allow_user_defined_namespace?
2018-03-17 18:26:18 +05:30
# We expect to be `active?` only when enabled and cluster is created (the api_url is assigned)
2019-03-13 22:55:13 +05:30
validates :api_url, public_url: true, presence: true
2018-03-17 18:26:18 +05:30
validates :token, presence: true
2019-05-18 00:54:41 +05:30
validates :ca_cert, certificate: true, allow_blank: true, if: :ca_cert_changed?
2018-03-17 18:26:18 +05:30
validate :prevent_modification, on: :update
after_save :clear_reactive_cache!
2018-12-13 13:39:08 +05:30
after_update :update_kubernetes_namespace
2018-03-17 18:26:18 +05:30
alias_attribute :ca_pem, :ca_cert
delegate :project, to: :cluster, allow_nil: true
delegate :enabled?, to: :cluster, allow_nil: true
2019-05-18 00:54:41 +05:30
delegate :provided_by_user?, to: :cluster, allow_nil: true
2019-02-15 15:39:39 +05:30
delegate :allow_user_defined_namespace?, to: :cluster, allow_nil: true
2018-12-13 13:39:08 +05:30
delegate :kubernetes_namespace, to: :cluster
2018-03-17 18:26:18 +05:30
alias_method :active?, :enabled?
2018-11-20 20:47:30 +05:30
enum_with_nil authorization_type: {
unknown_authorization: nil,
rbac: 1,
abac: 2
}
2019-02-15 15:39:39 +05:30
default_value_for :authorization_type, :rbac
2018-03-17 18:26:18 +05:30
def actual_namespace
if namespace.present?
namespace
else
default_namespace
end
end
2018-12-13 13:39:08 +05:30
def predefined_variables(project:)
2018-05-09 12:01:36 +05:30
Gitlab::Ci::Variables::Collection.new.tap do |variables|
2018-12-13 13:39:08 +05:30
variables.append(key: 'KUBE_URL', value: api_url)
2018-05-09 12:01:36 +05:30
if ca_pem.present?
variables
.append(key: 'KUBE_CA_PEM', value: ca_pem)
.append(key: 'KUBE_CA_PEM_FILE', value: ca_pem, file: true)
end
2018-12-13 13:39:08 +05:30
if kubernetes_namespace = cluster.kubernetes_namespaces.has_service_account_token.find_by(project: project)
variables.concat(kubernetes_namespace.predefined_variables)
2019-02-15 15:39:39 +05:30
elsif cluster.project_type?
2018-12-13 13:39:08 +05:30
# From 11.5, every Clusters::Project should have at least one
# Clusters::KubernetesNamespace, so once migration has been completed,
# this 'else' branch will be removed. For more information, please see
# https://gitlab.com/gitlab-org/gitlab-ce/merge_requests/22433
variables
.append(key: 'KUBE_URL', value: api_url)
2019-05-18 00:54:41 +05:30
.append(key: 'KUBE_TOKEN', value: token, public: false, masked: true)
2018-12-13 13:39:08 +05:30
.append(key: 'KUBE_NAMESPACE', value: actual_namespace)
2019-02-15 15:39:39 +05:30
.append(key: 'KUBECONFIG', value: kubeconfig, public: false, file: true)
2018-12-13 13:39:08 +05:30
end
2019-03-02 22:35:43 +05:30
variables.concat(cluster.predefined_variables)
2018-03-17 18:26:18 +05:30
end
end
# Constructs a list of terminals from the reactive cache
#
# Returns nil if the cache is empty, in which case you should try again a
# short time later
def terminals(environment)
with_reactive_cache do |data|
2019-05-18 00:54:41 +05:30
pods = filter_by_project_environment(data[:pods], project.full_path_slug, environment.slug)
2019-02-15 15:39:39 +05:30
terminals = pods.flat_map { |pod| terminals_for_pod(api_url, actual_namespace, pod) }.compact
2018-03-17 18:26:18 +05:30
terminals.each { |terminal| add_terminal_auth(terminal, terminal_auth) }
end
end
# Caches resources in the namespace so other calls don't need to block on
# network access
def calculate_reactive_cache
return unless enabled? && project && !project.pending_delete?
# We may want to cache extra things in the future
{ pods: read_pods }
end
def kubeclient
2018-12-13 13:39:08 +05:30
@kubeclient ||= build_kube_client!
2018-03-17 18:26:18 +05:30
end
private
def kubeconfig
to_kubeconfig(
url: api_url,
namespace: actual_namespace,
token: token,
ca_pem: ca_pem)
end
def default_namespace
2018-12-13 13:39:08 +05:30
kubernetes_namespace&.namespace.presence || fallback_default_namespace
end
# DEPRECATED
#
# On 11.4 Clusters::KubernetesNamespace was introduced, this model will allow to
# have multiple namespaces per project. This method will be removed after migration
# has been completed.
def fallback_default_namespace
2018-03-17 18:26:18 +05:30
return unless project
slug = "#{project.path}-#{project.id}".downcase
2018-12-13 13:39:08 +05:30
Gitlab::NamespaceSanitizer.sanitize(slug)
2018-03-17 18:26:18 +05:30
end
2018-12-13 13:39:08 +05:30
def build_kube_client!
2019-02-15 15:39:39 +05:30
raise "Incomplete settings" unless api_url
2019-03-02 22:35:43 +05:30
raise "No namespace" if cluster.project_type? && actual_namespace.empty? # can probably remove this line once we remove #actual_namespace
2018-03-17 18:26:18 +05:30
unless (username && password) || token
raise "Either username/password or token is required to access API"
end
2018-11-20 20:47:30 +05:30
Gitlab::Kubernetes::KubeClient.new(
api_url,
2018-03-17 18:26:18 +05:30
auth_options: kubeclient_auth_options,
ssl_options: kubeclient_ssl_options,
http_proxy_uri: ENV['http_proxy']
)
end
# Returns a hash of all pods in the namespace
def read_pods
2018-11-20 20:47:30 +05:30
kubeclient = build_kube_client!
2018-03-17 18:26:18 +05:30
kubeclient.get_pods(namespace: actual_namespace).as_json
2019-02-15 15:39:39 +05:30
rescue Kubeclient::ResourceNotFoundError
2018-03-17 18:26:18 +05:30
[]
end
def kubeclient_ssl_options
opts = { verify_ssl: OpenSSL::SSL::VERIFY_PEER }
if ca_pem.present?
opts[:cert_store] = OpenSSL::X509::Store.new
opts[:cert_store].add_cert(OpenSSL::X509::Certificate.new(ca_pem))
end
opts
end
def kubeclient_auth_options
{ bearer_token: token }
end
def terminal_auth
{
token: token,
ca_pem: ca_pem,
max_session_time: Gitlab::CurrentSettings.terminal_max_session_time
}
end
def enforce_namespace_to_lower_case
self.namespace = self.namespace&.downcase
end
2018-12-13 13:39:08 +05:30
def enforce_ca_whitespace_trimming
self.ca_pem = self.ca_pem&.strip
self.token = self.token&.strip
end
2019-02-15 15:39:39 +05:30
def no_namespace
if namespace
errors.add(:namespace, 'only allowed for project cluster')
end
end
2018-03-17 18:26:18 +05:30
def prevent_modification
2019-05-18 00:54:41 +05:30
return if provided_by_user?
2018-03-17 18:26:18 +05:30
if api_url_changed? || token_changed? || ca_pem_changed?
errors.add(:base, _('Cannot modify managed Kubernetes cluster'))
return false
end
true
end
2018-12-13 13:39:08 +05:30
def update_kubernetes_namespace
return unless namespace_changed?
run_after_commit do
2019-02-15 15:39:39 +05:30
ClusterConfigureWorker.perform_async(cluster_id)
2018-12-13 13:39:08 +05:30
end
end
2018-03-17 18:26:18 +05:30
end
end
end