201 lines
5 KiB
Ruby
201 lines
5 KiB
Ruby
|
class KubernetesService < DeploymentService
|
||
|
include Gitlab::CurrentSettings
|
||
|
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
|
||
|
validates :api_url, url: true
|
||
|
validates :token
|
||
|
end
|
||
|
|
||
|
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: 'namespace',
|
||
|
title: 'Kubernetes namespace',
|
||
|
placeholder: namespace_placeholder },
|
||
|
{ type: 'text',
|
||
|
name: 'api_url',
|
||
|
title: 'API URL',
|
||
|
placeholder: 'Kubernetes API URL, like https://kube.example.com/' },
|
||
|
{ type: 'text',
|
||
|
name: 'token',
|
||
|
title: 'Service token',
|
||
|
placeholder: 'Service token' },
|
||
|
{ type: 'textarea',
|
||
|
name: 'ca_pem',
|
||
|
title: 'Custom CA bundle',
|
||
|
placeholder: 'Certificate Authority bundle (PEM format)' },
|
||
|
]
|
||
|
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
|
||
|
variables = [
|
||
|
{ key: 'KUBE_URL', value: api_url, public: true },
|
||
|
{ key: 'KUBE_TOKEN', value: token, public: false },
|
||
|
{ key: 'KUBE_NAMESPACE', value: actual_namespace, public: true }
|
||
|
]
|
||
|
|
||
|
if ca_pem.present?
|
||
|
variables << { key: 'KUBE_CA_PEM', value: ca_pem, public: true }
|
||
|
variables << { key: 'KUBE_CA_PEM_FILE', value: ca_pem, public: true, file: true }
|
||
|
end
|
||
|
|
||
|
variables
|
||
|
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|
|
||
|
pods = data.fetch(:pods, nil)
|
||
|
filter_pods(pods, app: environment.slug).
|
||
|
flat_map { |pod| terminals_for_pod(api_url, actual_namespace, pod) }.
|
||
|
each { |terminal| add_terminal_auth(terminal, terminal_auth) }
|
||
|
end
|
||
|
end
|
||
|
|
||
|
# Caches all pods in the namespace so other calls don't need to block on
|
||
|
# network access.
|
||
|
def calculate_reactive_cache
|
||
|
return unless active? && project && !project.pending_delete?
|
||
|
|
||
|
kubeclient = build_kubeclient!
|
||
|
|
||
|
# Store as hashes, rather than as third-party types
|
||
|
pods = begin
|
||
|
kubeclient.get_pods(namespace: actual_namespace).as_json
|
||
|
rescue KubeException => err
|
||
|
raise err unless err.error_code == 404
|
||
|
[]
|
||
|
end
|
||
|
|
||
|
# We may want to cache extra things in the future
|
||
|
{ pods: pods }
|
||
|
end
|
||
|
|
||
|
TEMPLATE_PLACEHOLDER = 'Kubernetes namespace'.freeze
|
||
|
|
||
|
private
|
||
|
|
||
|
def namespace_placeholder
|
||
|
default_namespace || TEMPLATE_PLACEHOLDER
|
||
|
end
|
||
|
|
||
|
def default_namespace
|
||
|
"#{project.path}-#{project.id}" if project.present?
|
||
|
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
|
||
|
|
||
|
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 join_api_url(*parts)
|
||
|
url = URI.parse(api_url)
|
||
|
prefix = url.path.sub(%r{/+\z}, '')
|
||
|
|
||
|
url.path = [prefix, *parts].join("/")
|
||
|
|
||
|
url.to_s
|
||
|
end
|
||
|
|
||
|
def terminal_auth
|
||
|
{
|
||
|
token: token,
|
||
|
ca_pem: ca_pem,
|
||
|
max_session_time: current_application_settings.terminal_max_session_time
|
||
|
}
|
||
|
end
|
||
|
end
|