debian-mirror-gitlab/app/models/project_services/kubernetes_service.rb

268 lines
7.2 KiB
Ruby
Raw Normal View History

2018-11-20 20:47:30 +05:30
# frozen_string_literal: true
2018-03-17 18:26:18 +05:30
##
# NOTE:
# We'll move this class to Clusters::Platforms::Kubernetes, which contains exactly the same logic.
# After we've migrated data, we'll remove KubernetesService. This would happen in a few months.
# If you're modyfiyng this class, please note that you should update the same change in Clusters::Platforms::Kubernetes.
2019-09-04 21:01:54 +05:30
class KubernetesService < Service
2017-08-17 22:00:37 +05:30
include Gitlab::Kubernetes
include ReactiveCaching
2019-09-04 21:01:54 +05:30
default_value_for :category, 'deployment'
2017-08-17 22:00:37 +05:30
self.reactive_cache_key = ->(service) { [service.class.model_name.singular, service.project_id] }
# Namespace defaults to the project path, but can be overridden in case that
# is an invalid or inappropriate name
prop_accessor :namespace
# Access to kubernetes is directly through the API
prop_accessor :api_url
# Bearer authentication
# TODO: user/password auth, client certificates
prop_accessor :token
# Provide a custom CA bundle for self-signed deployments
prop_accessor :ca_pem
with_options presence: true, if: :activated? do
2018-11-08 19:23:39 +05:30
validates :api_url, public_url: true
2017-08-17 22:00:37 +05:30
validates :token
end
2018-03-17 18:26:18 +05:30
before_validation :enforce_namespace_to_lower_case
2019-09-04 21:01:54 +05:30
attr_accessor :skip_deprecation_validation
validate :deprecation_validation, unless: :skip_deprecation_validation
2017-08-17 22:00:37 +05:30
validates :namespace,
allow_blank: true,
length: 1..63,
if: :activated?,
format: {
with: Gitlab::Regex.kubernetes_namespace_regex,
message: Gitlab::Regex.kubernetes_namespace_regex_message
}
after_save :clear_reactive_cache!
2019-09-04 21:01:54 +05:30
def self.supported_events
%w()
end
def can_test?
false
end
2017-08-17 22:00:37 +05:30
def initialize_properties
self.properties = {} if properties.nil?
end
def title
'Kubernetes'
end
def description
2019-02-15 15:39:39 +05:30
'Kubernetes / OpenShift integration'
2017-08-17 22:00:37 +05:30
end
def self.to_param
'kubernetes'
end
def fields
[
{ type: 'text',
name: 'api_url',
title: 'API URL',
placeholder: 'Kubernetes API URL, like https://kube.example.com/' },
{ type: 'textarea',
name: 'ca_pem',
2017-09-10 17:25:29 +05:30
title: 'CA Certificate',
2017-08-17 22:00:37 +05:30
placeholder: 'Certificate Authority bundle (PEM format)' },
2017-09-10 17:25:29 +05:30
{ type: 'text',
name: 'namespace',
title: 'Project namespace (optional/unique)',
placeholder: namespace_placeholder },
{ type: 'text',
name: 'token',
title: 'Token',
placeholder: 'Service token' }
2017-08-17 22:00:37 +05:30
]
end
2019-09-04 21:01:54 +05:30
def kubernetes_namespace_for(project)
2017-08-17 22:00:37 +05:30
if namespace.present?
namespace
else
default_namespace
end
end
# Check we can connect to the Kubernetes API
def test(*args)
2018-11-20 20:47:30 +05:30
kubeclient = build_kube_client!
2017-08-17 22:00:37 +05:30
2018-11-20 20:47:30 +05:30
kubeclient.core_client.discover
{ success: kubeclient.core_client.discovered, result: "Checked API discovery endpoint" }
2017-08-17 22:00:37 +05:30
rescue => err
{ success: false, result: err }
end
2018-12-13 13:39:08 +05:30
# Project param was added on
# https://gitlab.com/gitlab-org/gitlab-ce/merge_requests/22011,
# as a way to keep this service compatible with
# Clusters::Platforms::Kubernetes, it won't be used on this method
# as it's only needed for Clusters::Cluster.
def predefined_variables(project:)
2018-05-09 12:01:36 +05:30
Gitlab::Ci::Variables::Collection.new.tap do |variables|
variables
.append(key: 'KUBE_URL', value: api_url)
2019-07-07 11:18:12 +05:30
.append(key: 'KUBE_TOKEN', value: token, public: false, masked: true)
2019-09-04 21:01:54 +05:30
.append(key: 'KUBE_NAMESPACE', value: kubernetes_namespace_for(project))
2019-02-15 15:39:39 +05:30
.append(key: 'KUBECONFIG', value: kubeconfig, public: false, file: true)
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
2017-08-17 22:00:37 +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-09-04 21:01:54 +05:30
project = environment.project
2019-07-07 11:18:12 +05:30
pods = filter_by_project_environment(data[:pods], project.full_path_slug, environment.slug)
2019-09-04 21:01:54 +05:30
terminals = pods.flat_map { |pod| terminals_for_pod(api_url, kubernetes_namespace_for(project), pod) }.compact
2017-09-10 17:25:29 +05:30
terminals.each { |terminal| add_terminal_auth(terminal, terminal_auth) }
2017-08-17 22:00:37 +05:30
end
end
2017-09-10 17:25:29 +05:30
# Caches resources in the namespace so other calls don't need to block on
# network access
2017-08-17 22:00:37 +05:30
def calculate_reactive_cache
return unless active? && project && !project.pending_delete?
# We may want to cache extra things in the future
2017-09-10 17:25:29 +05:30
{ pods: read_pods }
2017-08-17 22:00:37 +05:30
end
2018-03-17 18:26:18 +05:30
def kubeclient
2018-12-13 13:39:08 +05:30
@kubeclient ||= build_kube_client!
2018-03-17 18:26:18 +05:30
end
def deprecated?
2019-09-04 21:01:54 +05:30
true
end
def editable?
false
2018-03-17 18:26:18 +05:30
end
def deprecation_message
2019-09-04 21:01:54 +05:30
content = if project
_("Kubernetes service integration has been deprecated. %{deprecated_message_content} your Kubernetes clusters using the new <a href=\"%{url}\"/>Kubernetes Clusters</a> page") % {
deprecated_message_content: deprecated_message_content,
url: Gitlab::Routing.url_helpers.project_clusters_path(project)
}
else
_("The instance-level Kubernetes service integration is deprecated. Your data has been migrated to an <a href=\"%{url}\"/>instance-level cluster</a>.") % {
url: Gitlab::Routing.url_helpers.admin_clusters_path
}
end
2018-03-17 18:26:18 +05:30
content.html_safe
end
2017-08-17 22:00:37 +05:30
TEMPLATE_PLACEHOLDER = 'Kubernetes namespace'.freeze
private
2017-09-10 17:25:29 +05:30
def kubeconfig
to_kubeconfig(
url: api_url,
2019-09-04 21:01:54 +05:30
namespace: kubernetes_namespace_for(project),
2017-09-10 17:25:29 +05:30
token: token,
ca_pem: ca_pem)
end
2017-08-17 22:00:37 +05:30
def namespace_placeholder
default_namespace || TEMPLATE_PLACEHOLDER
end
def default_namespace
2018-03-17 18:26:18 +05:30
return unless project
slug = "#{project.path}-#{project.id}".downcase
slug.gsub(/[^-a-z0-9]/, '-').gsub(/^-+/, '')
2017-08-17 22:00:37 +05:30
end
2018-12-13 13:39:08 +05:30
def build_kube_client!
2019-09-04 21:01:54 +05:30
raise "Incomplete settings" unless api_url && kubernetes_namespace_for(project) && token
2017-08-17 22:00:37 +05:30
2018-11-20 20:47:30 +05:30
Gitlab::Kubernetes::KubeClient.new(
api_url,
2017-08-17 22:00:37 +05:30
auth_options: kubeclient_auth_options,
ssl_options: kubeclient_ssl_options,
http_proxy_uri: ENV['http_proxy']
)
end
2017-09-10 17:25:29 +05:30
# Returns a hash of all pods in the namespace
def read_pods
2018-11-20 20:47:30 +05:30
kubeclient = build_kube_client!
2017-09-10 17:25:29 +05:30
2019-09-04 21:01:54 +05:30
kubeclient.get_pods(namespace: kubernetes_namespace_for(project)).as_json
2019-02-15 15:39:39 +05:30
rescue Kubeclient::ResourceNotFoundError
2017-09-10 17:25:29 +05:30
[]
end
2017-08-17 22:00:37 +05:30
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,
2018-03-17 18:26:18 +05:30
max_session_time: Gitlab::CurrentSettings.terminal_max_session_time
2017-08-17 22:00:37 +05:30
}
end
2018-03-17 18:26:18 +05:30
def enforce_namespace_to_lower_case
self.namespace = self.namespace&.downcase
end
def deprecation_validation
2018-11-08 19:23:39 +05:30
return if active_changed?(from: true, to: false) || (new_record? && !active?)
2018-03-17 18:26:18 +05:30
if deprecated?
errors[:base] << deprecation_message
end
end
def deprecated_message_content
2019-09-04 21:01:54 +05:30
_("Fields on this page are now uneditable, you can configure")
2018-03-17 18:26:18 +05:30
end
2017-08-17 22:00:37 +05:30
end