module Gitlab module Kubernetes class Helm HELM_VERSION = '2.7.0'.freeze NAMESPACE = 'gitlab-managed-apps'.freeze INSTALL_DEPS = <<-EOS.freeze set -eo pipefail apk add -U ca-certificates openssl >/dev/null wget -q -O - https://kubernetes-helm.storage.googleapis.com/helm-v${HELM_VERSION}-linux-amd64.tar.gz | tar zxC /tmp >/dev/null mv /tmp/linux-amd64/helm /usr/bin/ EOS InstallCommand = Struct.new(:name, :install_helm, :chart) do def pod_name "install-#{name}" end end def initialize(kubeclient) @kubeclient = kubeclient @namespace = Gitlab::Kubernetes::Namespace.new(NAMESPACE, kubeclient) end def install(command) @namespace.ensure_exists! @kubeclient.create_pod(pod_resource(command)) end ## # Returns Pod phase # # https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#pod-phase # # values: "Pending", "Running", "Succeeded", "Failed", "Unknown" # def installation_status(pod_name) @kubeclient.get_pod(pod_name, @namespace.name).status.phase end def installation_log(pod_name) @kubeclient.get_pod_log(pod_name, @namespace.name).body end def delete_installation_pod!(pod_name) @kubeclient.delete_pod(pod_name, @namespace.name) end private def pod_resource(command) labels = { 'gitlab.org/action': 'install', 'gitlab.org/application': command.name } metadata = { name: command.pod_name, namespace: @namespace.name, labels: labels } container = { name: 'helm', image: 'alpine:3.6', env: generate_pod_env(command), command: %w(/bin/sh), args: %w(-c $(COMMAND_SCRIPT)) } spec = { containers: [container], restartPolicy: 'Never' } ::Kubeclient::Resource.new(metadata: metadata, spec: spec) end def generate_pod_env(command) { HELM_VERSION: HELM_VERSION, TILLER_NAMESPACE: @namespace.name, COMMAND_SCRIPT: generate_script(command) }.map { |key, value| { name: key, value: value } } end def generate_script(command) [ INSTALL_DEPS, helm_init_command(command), helm_install_command(command) ].join("\n") end def helm_init_command(command) if command.install_helm 'helm init >/dev/null' else 'helm init --client-only >/dev/null' end end def helm_install_command(command) return if command.chart.nil? "helm install #{command.chart} --name #{command.name} --namespace #{@namespace.name} >/dev/null" end end end end