From 5d820c4d7c4aa1bd2eeef0ad59d51c66128fae58 Mon Sep 17 00:00:00 2001
From: Adam Reese <adam@reese.io>
Date: Fri, 26 Aug 2016 14:49:22 -0700
Subject: [PATCH] feat(cmd): install latest tagged image on `helm init`

closes: #1117
---
 cmd/helm/installer/install.go | 139 +++++++++++++++++++++-------------
 1 file changed, 87 insertions(+), 52 deletions(-)

diff --git a/cmd/helm/installer/install.go b/cmd/helm/installer/install.go
index 8c6b1a0c4..0bfd197e8 100644
--- a/cmd/helm/installer/install.go
+++ b/cmd/helm/installer/install.go
@@ -17,15 +17,20 @@ limitations under the License.
 package installer // import "k8s.io/helm/cmd/helm/installer"
 
 import (
-	"bytes"
 	"fmt"
-	"text/template"
+	"strings"
 
-	"github.com/Masterminds/sprig"
+	"k8s.io/kubernetes/pkg/api"
+	"k8s.io/kubernetes/pkg/api/errors"
+	"k8s.io/kubernetes/pkg/apis/extensions"
+	"k8s.io/kubernetes/pkg/util/intstr"
 
 	"k8s.io/helm/pkg/kube"
+	"k8s.io/helm/pkg/version"
 )
 
+const defaultImage = "gcr.io/kubernetes-helm/tiller"
+
 // Install uses kubernetes client to install tiller
 //
 // Returns the string output received from the operation, and an error if the
@@ -43,59 +48,89 @@ func Install(namespace, image string, verbose bool) error {
 		namespace = ns
 	}
 
-	var b bytes.Buffer
-
-	// Add main install YAML
-	istpl := template.New("install").Funcs(sprig.TxtFuncMap())
-
-	cfg := struct {
-		Namespace, Image string
-	}{namespace, image}
-
-	if err := template.Must(istpl.Parse(InstallYAML)).Execute(&b, cfg); err != nil {
+	c, err := kc.Client()
+	if err != nil {
 		return err
 	}
 
-	if verbose {
-		fmt.Println(b.String())
+	ns := generateNamespace(namespace)
+	if _, err := c.Namespaces().Create(ns); err != nil {
+		if !errors.IsAlreadyExists(err) {
+			return err
+		}
+	}
+
+	if image == "" {
+		// strip git sha off version
+		tag := strings.Split(version.Version, "+")[0]
+		image = fmt.Sprintf("%s:%s", defaultImage, tag)
 	}
 
-	return kc.Create(namespace, &b)
+	rc := generateDeployment(image)
+
+	_, err = c.Deployments(namespace).Create(rc)
+	return err
+}
+
+func generateLabels(labels map[string]string) map[string]string {
+	labels["app"] = "helm"
+	return labels
+}
+
+func generateDeployment(image string) *extensions.Deployment {
+	labels := generateLabels(map[string]string{"name": "tiller"})
+	d := &extensions.Deployment{
+		ObjectMeta: api.ObjectMeta{
+			Name:   "tiller-deploy",
+			Labels: labels,
+		},
+		Spec: extensions.DeploymentSpec{
+			Replicas: 1,
+			Template: api.PodTemplateSpec{
+				ObjectMeta: api.ObjectMeta{
+					Labels: labels,
+				},
+				Spec: api.PodSpec{
+					Containers: []api.Container{
+						{
+							Name:            "tiller",
+							Image:           image,
+							ImagePullPolicy: "Always",
+							Ports:           []api.ContainerPort{{ContainerPort: 44134, Name: "tiller"}},
+							LivenessProbe: &api.Probe{
+								Handler: api.Handler{
+									HTTPGet: &api.HTTPGetAction{
+										Path: "/liveness",
+										Port: intstr.FromInt(44135),
+									},
+								},
+								InitialDelaySeconds: 1,
+								TimeoutSeconds:      1,
+							},
+							ReadinessProbe: &api.Probe{
+								Handler: api.Handler{
+									HTTPGet: &api.HTTPGetAction{
+										Path: "/readiness",
+										Port: intstr.FromInt(44135),
+									},
+								},
+								InitialDelaySeconds: 1,
+								TimeoutSeconds:      1,
+							},
+						},
+					},
+				},
+			},
+		},
+	}
+	return d
 }
 
-// InstallYAML is the installation YAML for DM.
-const InstallYAML = `
----
-apiVersion: extensions/v1beta1
-kind: Deployment
-metadata:
-  name: tiller-deploy
-  namespace: {{ .Namespace }}
-spec:
-  replicas: 1
-  template:
-    metadata:
-      labels:
-        app: helm
-        name: tiller
-    spec:
-      containers:
-      - image: {{default "gcr.io/kubernetes-helm/tiller:canary" .Image}}
-        name: tiller
-        ports:
-        - containerPort: 44134
-          name: tiller
-        imagePullPolicy: Always
-        livenessProbe:
-          httpGet:
-            path: /liveness
-            port: 44135
-          initialDelaySeconds: 1
-          timeoutSeconds: 1
-        readinessProbe:
-          httpGet:
-            path: /readiness
-            port: 44135
-          initialDelaySeconds: 1
-          timeoutSeconds: 1
-`
+func generateNamespace(namespace string) *api.Namespace {
+	return &api.Namespace{
+		ObjectMeta: api.ObjectMeta{
+			Name:   namespace,
+			Labels: generateLabels(map[string]string{"name": "helm-namespace"}),
+		},
+	}
+}
-- 
GitLab