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

258 lines
6.7 KiB
Ruby
Raw Normal View History

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.
2017-08-17 22:00:37 +05:30
class KubernetesService < DeploymentService
include Gitlab::Kubernetes
include ReactiveCaching
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
validate :deprecation_validation, unless: :template?
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!
def initialize_properties
self.properties = {} if properties.nil?
end
def title
'Kubernetes'
end
def description
'Kubernetes / Openshift integration'
end
def help
'To enable terminal access to Kubernetes environments, label your ' \
'deployments with `app=$CI_ENVIRONMENT_SLUG`'
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
def actual_namespace
if namespace.present?
namespace
else
default_namespace
end
end
# Check we can connect to the Kubernetes API
def test(*args)
kubeclient = build_kubeclient!
kubeclient.discover
{ success: kubeclient.discovered, result: "Checked API discovery endpoint" }
rescue => err
{ success: false, result: err }
end
def predefined_variables
2017-09-10 17:25:29 +05:30
config = YAML.dump(kubeconfig)
2018-05-09 12:01:36 +05:30
Gitlab::Ci::Variables::Collection.new.tap do |variables|
variables
.append(key: 'KUBE_URL', value: api_url)
.append(key: 'KUBE_TOKEN', value: token, public: false)
.append(key: 'KUBE_NAMESPACE', value: actual_namespace)
.append(key: 'KUBECONFIG', value: config, public: false, file: true)
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|
2017-09-10 17:25:29 +05:30
pods = filter_by_label(data[:pods], app: environment.slug)
terminals = pods.flat_map { |pod| terminals_for_pod(api_url, actual_namespace, pod) }
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
@kubeclient ||= build_kubeclient!
end
def deprecated?
!active
end
def deprecation_message
content = _("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)
}
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,
namespace: actual_namespace,
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
def build_kubeclient!(api_path: 'api', api_version: 'v1')
raise "Incomplete settings" unless api_url && actual_namespace && token
::Kubeclient::Client.new(
join_api_url(api_path),
api_version,
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
kubeclient = build_kubeclient!
kubeclient.get_pods(namespace: actual_namespace).as_json
2018-05-09 12:01:36 +05:30
rescue Kubeclient::HttpError => err
2017-09-10 17:25:29 +05:30
raise err unless err.error_code == 404
2018-03-17 18:26:18 +05:30
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
2017-09-10 17:25:29 +05:30
def join_api_url(api_path)
2017-08-17 22:00:37 +05:30
url = URI.parse(api_url)
prefix = url.path.sub(%r{/+\z}, '')
2017-09-10 17:25:29 +05:30
url.path = [prefix, api_path].join("/")
2017-08-17 22:00:37 +05:30
url.to_s
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
if active?
_("Your Kubernetes cluster information on this page is still editable, but you are advised to disable and reconfigure")
else
_("Fields on this page are now uneditable, you can configure")
end
end
2017-08-17 22:00:37 +05:30
end