Jerome Brette's Blog

Jerome Brette's Blog

Deploy Helm and Tiller on Rasberry PI Cluster

Goal

The main purpose of this exercise is to be able to use Helm on the Rapsberry PI Cluster.

Key Aspects

  • The goal is to setup helm and tiller on the Raspberry PI cluster
  • Having the golang, glide…and related libraries setup in a PI for compilation is kind of complicated. I started but encounter too many issues (even small), had to install too many compilation related packages on my PI system, hence decided to use an Ubuntu VM to compile and prepare the binaries for image for helm and tiller. This will be usefull to setup CI/CD pipeline with Travis-CI.

Build Tiller executable and Docker image

Cross Compiling from Ubuntu Machine

First check the go setup. Fetch the code

Let’s check the go environment. See Setup GOLANG environment

which go

/usr/bin/go
go version

go version go1.10.1 linux/amd64
export GOPATH=$HOME/src
mkdir -p src/k8s.io
cd src/k8s.io

git clone -b release-2.9 git@github.com:kubernetes/helm.git
cd $HOME/src/k8s.io/helm
make clean bootstrap build-cross dist APP=helm VERSION=2.9 TARGETS=linux/arm

...

go build -o bin/protoc-gen-go ./vendor/github.com/golang/protobuf/protoc-gen-go
CGO_ENABLED=0 gox -parallel=3 -output="_dist/{{.OS}}-{{.Arch}}/{{.Dir}}" -osarch='linux/arm'  -tags '' -ldflags '-w -s -X k8s.io/helm/pkg/version.Version=2.9 -X k8s.io/helm/pkg/version.BuildMetadata= -X k8s.io/helm/pkg/version.GitCommit=20adb27c7c5868466912eebdf6664e7390ebe710 -X k8s.io/helm/pkg/version.GitTreeState=clean -extldflags "-static"' k8s.io/helm/cmd/helm
Number of parallel builds: 3

-->       linux/arm: k8s.io/helm/cmd/helm
( \
        cd _dist && \
        find * -type d -exec cp ../LICENSE {} \; && \
        find * -type d -exec cp ../README.md {} \; && \
        find * -type d -exec tar -zcf helm-2.9-{}.tar.gz {} \; && \
        find * -type d -exec zip -r helm-2.9-{}.zip {} \; \
)
  adding: linux-arm/ (stored 0%)
  adding: linux-arm/LICENSE (deflated 65%)
  adding: linux-arm/README.md (deflated 59%)
  adding: linux-arm/helm (deflated 67%)

Let’s transfer the helm binarie to RPI

scp -r _dist/linux-arm/ rpiuser@192.168.1.95:/home/rpiuser/helm_binaries

Cross Compiling and Image cration from Travis-CI

  • WIP

Helm

Cross Compiling from Ubuntu Machine

Let’s assume we cloned the code for helm already.

make clean bootstrap build-cross dist APP=tiller VERSION=2.9 TARGETS=linux/arm

...
go build -o bin/protoc-gen-go ./vendor/github.com/golang/protobuf/protoc-gen-go
CGO_ENABLED=0 gox -parallel=3 -output="_dist/{{.OS}}-{{.Arch}}/{{.Dir}}" -osarch='linux/arm'  -tags '' -ldflags '-w -s -X k8s.io/helm/pkg/version.Version=2.9 -X k8s.io/helm/pkg/version.BuildMetadata= -X k8s.io/helm/pkg/version.GitCommit=20adb27c7c5868466912eebdf6664e7390ebe710 -X k8s.io/helm/pkg/version.GitTreeState=clean -extldflags "-static"' k8s.io/helm/cmd/tiller
Number of parallel builds: 3

-->       linux/arm: k8s.io/helm/cmd/tiller
( \
        cd _dist && \
        find * -type d -exec cp ../LICENSE {} \; && \
        find * -type d -exec cp ../README.md {} \; && \
        find * -type d -exec tar -zcf helm-2.9-{}.tar.gz {} \; && \
        find * -type d -exec zip -r helm-2.9-{}.zip {} \; \
)
  adding: linux-arm/ (stored 0%)
  adding: linux-arm/LICENSE (deflated 65%)
  adding: linux-arm/README.md (deflated 59%)
  adding: linux-arm/tiller (deflated 67%)

Build the docker image on the remote PI

Create a Dockerfile called rootfs/Dockerfile.arm32v7

FROM debian:jessie-slim

RUN apt-get update \
    && apt-get install -y --no-install-recommends ca-certificates \
    && rm -rf /var/lib/apt/lists/*

ENV HOME /tmp

COPY _dist/linux-arm /

EXPOSE 44134

CMD ["/tiller"]
export DHUBREPO=jbrette/tiller-arm32v7
export VERSION=2.9
docker build -t $DHUBREPO:$VERSION -f rootfs/Dockerfile.arm32v7 .

Sending build context to Docker daemon  284.1MB
Step 1/6 : FROM debian:jessie-slim
 ---> d273fca45b31
Step 2/6 : RUN apt-get update     && apt-get install -y --no-install-recommends ca-certificates     && rm -rf /var/lib/apt/lists/*
 ---> Running in e51ea5c31ad7
Get:1 http://security.debian.org jessie/updates InRelease [44.9 kB]
Ign http://deb.debian.org jessie InRelease
Get:2 http://deb.debian.org jessie-updates InRelease [145 kB]
Get:3 http://deb.debian.org jessie Release.gpg [2420 B]
Get:4 http://deb.debian.org jessie Release [148 kB]
Get:5 http://security.debian.org jessie/updates/main armel Packages [576 kB]
Get:6 http://deb.debian.org jessie-updates/main armel Packages [23.7 kB]
Get:7 http://deb.debian.org jessie/main armel Packages [8902 kB]
Fetched 9843 kB in 49s (199 kB/s)
Reading package lists...
Reading package lists...
Building dependency tree...
Reading state information...
The following extra packages will be installed:
  libssl1.0.0 openssl
The following NEW packages will be installed:
  ca-certificates libssl1.0.0 openssl
0 upgraded, 3 newly installed, 0 to remove and 1 not upgraded.
Need to get 1692 kB of archives.
After this operation, 3770 kB of additional disk space will be used.
Get:1 http://security.debian.org/debian-security/ jessie/updates/main ca-certificates all 20141019+deb8u4 [185 kB]
Get:2 http://deb.debian.org/debian/ jessie/main libssl1.0.0 armel 1.0.1t-1+deb8u8 [852 kB]
Get:3 http://deb.debian.org/debian/ jessie/main openssl armel 1.0.1t-1+deb8u8 [655 kB]
debconf: delaying package configuration, since apt-utils is not installed
Fetched 1692 kB in 7s (235 kB/s)
Selecting previously unselected package libssl1.0.0:armel.
(Reading database ... 7451 files and directories currently installed.)
Preparing to unpack .../libssl1.0.0_1.0.1t-1+deb8u8_armel.deb ...
Unpacking libssl1.0.0:armel (1.0.1t-1+deb8u8) ...
Selecting previously unselected package openssl.
Preparing to unpack .../openssl_1.0.1t-1+deb8u8_armel.deb ...
Unpacking openssl (1.0.1t-1+deb8u8) ...
Selecting previously unselected package ca-certificates.
Preparing to unpack .../ca-certificates_20141019+deb8u4_all.deb ...
Unpacking ca-certificates (20141019+deb8u4) ...
Setting up libssl1.0.0:armel (1.0.1t-1+deb8u8) ...
debconf: unable to initialize frontend: Dialog
debconf: (TERM is not set, so the dialog frontend is not usable.)
debconf: falling back to frontend: Readline
debconf: unable to initialize frontend: Readline
debconf: (Can't locate Term/ReadLine.pm in @INC (you may need to install the Term::ReadLine module) (@INC contains: /etc/perl /usr/local/lib/arm-linux-gnueabi/perl/5.20.2 /usr/local/share/perl/5.20.2 /usr/lib/arm-linux-gnueabi/perl5/5.20 /usr/share/perl5 /usr/lib/arm-linux-gnueabi/perl/5.20 /usr/share/perl/5.20 /usr/local/lib/site_perl .) at /usr/share/perl5/Debconf/FrontEnd/Readline.pm line 7.)
debconf: falling back to frontend: Teletype
Setting up openssl (1.0.1t-1+deb8u8) ...
Setting up ca-certificates (20141019+deb8u4) ...
debconf: unable to initialize frontend: Dialog
debconf: (TERM is not set, so the dialog frontend is not usable.)
debconf: falling back to frontend: Readline
debconf: unable to initialize frontend: Readline
debconf: (Can't locate Term/ReadLine.pm in @INC (you may need to install the Term::ReadLine module) (@INC contains: /etc/perl /usr/local/lib/arm-linux-gnueabi/perl/5.20.2 /usr/local/share/perl/5.20.2 /usr/lib/arm-linux-gnueabi/perl5/5.20 /usr/share/perl5 /usr/lib/arm-linux-gnueabi/perl/5.20 /usr/share/perl/5.20 /usr/local/lib/site_perl .) at /usr/share/perl5/Debconf/FrontEnd/Readline.pm line 7.)
debconf: falling back to frontend: Teletype
Updating certificates in /etc/ssl/certs... 152 added, 0 removed; done.
Processing triggers for libc-bin (2.19-18+deb8u10) ...
Processing triggers for ca-certificates (20141019+deb8u4) ...
Updating certificates in /etc/ssl/certs... 0 added, 0 removed; done.
Running hooks in /etc/ca-certificates/update.d....done.
Removing intermediate container e51ea5c31ad7
 ---> 1dccf46769e7
Step 3/6 : ENV HOME /tmp
 ---> Running in 45de8c047bd3
Removing intermediate container 45de8c047bd3
 ---> baa6d8b15164
Step 4/6 : COPY _dist/linux-arm /
 ---> 59a6cc8bfdbe
Step 5/6 : EXPOSE 44134
 ---> Running in d2c6d9b3a3bc
Removing intermediate container d2c6d9b3a3bc
 ---> 28a331217c52
Step 6/6 : CMD ["/tiller"]
 ---> Running in 9760ea4dc6c5
Removing intermediate container 9760ea4dc6c5
 ---> 9384b4f39ab3
Successfully built 9384b4f39ab3
Successfully tagged jbrette/tiller-arm32v7:2.9

Cross Compiling and Image cration from Travis-CI

  • WIP

Install Tiller and Helm on RPI

Install helm on PI

ssh rpiuser@192.168.1.95
sudo cp /home/rpiuser/helm_binaries/helm /usr/bin/helm

Deploy Tiller POD

First push the tiller docker image to the repo

docker login

docker push jbrette/tiller-arm32v7:2.9

Prepare the POD deployment files

mkdir -p ~/kube-deployments/tiller
cd  ~/kube-deployments/tiller

Create the kubectl file for tiller service account

cat tiller-serviceaccount.yaml
---
apiVersion: v1
kind: ServiceAccount
metadata:
  name: tiller
  namespace: kube-system
...
---
apiVersion: rbac.authorization.k8s.io/v1beta1
kind: ClusterRoleBinding
metadata:
  name: tiller
roleRef:
  apiGroup: rbac.authorization.k8s.io
  kind: ClusterRole
  name: cluster-admin
subjects:
  - kind: ServiceAccount
    name: tiller
    namespace: kube-system
...

Create the tiller service account

kubectl create -f tiller-serviceaccount.yaml

serviceaccount/tiller created
clusterrolebinding.rbac.authorization.k8s.io/tiller created

Create manually the tiller deployment file and leverage the tiller service account

helm init --service-account tiller --output yaml > tiller.yaml

Edit the image and replace gcr.io/kubernetes-helm/tiller:2.9 by jbrette/tiller-arm32v7:2.9

apiVersion: extensions/v1beta1
kind: Deployment
metadata:
  creationTimestamp: null
  labels:
    app: helm
    name: tiller
  name: tiller-deploy
  namespace: kube-system
spec:
  replicas: 1
  strategy: {}
  template:
    metadata:
      creationTimestamp: null
      labels:
        app: helm
        name: tiller
    spec:
      containers:
      - env:
        - name: TILLER_NAMESPACE
          value: kube-system
        - name: TILLER_HISTORY_MAX
          value: "0"
        image: gcr.io/kubernetes-helm/tiller:2.9
        imagePullPolicy: IfNotPresent
        livenessProbe:
          httpGet:
            path: /liveness
            port: 44135
          initialDelaySeconds: 1
          timeoutSeconds: 1
        name: tiller
        ports:
        - containerPort: 44134
          name: tiller
        - containerPort: 44135
          name: http
        readinessProbe:
          httpGet:
            path: /readiness
            port: 44135
          initialDelaySeconds: 1
          timeoutSeconds: 1
        resources: {}
      serviceAccountName: tiller
status: {}

Deploy tiller in the kubernetes cluster

kubectl create -f tiller.yaml

Check the deployment state

kubectl get all -n kube-system

NAME                                        READY     STATUS             RESTARTS   AGE
pod/tiller-deploy-b59fcc885-66l7s           1/1       Running            0          5m

NAME                                   DESIRED   CURRENT   UP-TO-DATE   AVAILABLE   AGE
deployment.apps/tiller-deploy          1         1         1            1           5m

NAME                                              DESIRED   CURRENT   READY     AGE
replicaset.apps/tiller-deploy-b59fcc885           1         1         1         5m

Check the logs

kubectl logs pod/tiller-deploy-b59fcc885-66l7s -n kube-system

[main] 2018/07/15 18:08:10 Starting Tiller 2.9 (tls=false)
[main] 2018/07/15 18:08:10 GRPC listening on :44134
[main] 2018/07/15 18:08:10 Probes listening on :44135
[main] 2018/07/15 18:08:10 Storage driver is ConfigMap
[main] 2018/07/15 18:08:10 Max history per release is 0

Init the Helm client side

helm init --client-only

Creating /home/rpiuser/.helm
Creating /home/rpiuser/.helm/repository
Creating /home/rpiuser/.helm/repository/cache
Creating /home/rpiuser/.helm/repository/local
Creating /home/rpiuser/.helm/plugins
Creating /home/rpiuser/.helm/starters
Creating /home/rpiuser/.helm/cache/archive
Creating /home/rpiuser/.helm/repository/repositories.yaml
Adding stable repo with URL: https://kubernetes-charts.storage.googleapis.com
Adding local repo with URL: http://127.0.0.1:8879/charts
$HELM_HOME has been configured at /home/rpiuser/.helm.
Not installing Tiller due to 'client-only' flag having been set
Happy Helming!

Check helm commands are working properly

helm ls
helm search stable/nginx-ingress
helm inspect stable/nginx-ingress

Use helm on RPI

Let’s access the simple helm repo designed for

helm repo add kubeplay 'https://raw.githubusercontent.com/jbrette/kubeplay/arm32v7/helmrepo/'
helm repo list

NAME            URL
stable          https://kubernetes-charts.storage.googleapis.com
local           http://127.0.0.1:8879/charts
kubeplay        https://raw.githubusercontent.com/jbrette/kubeplay/arm32v7/helmrepo/
helm search kubeplay

NAME                            CHART VERSION   APP VERSION     DESCRIPTION
kubeplay/kubeplay-arm32v7       0.1.0           0.1.0           A Helm chart for Kubernetes

Install simple hello world helm chart from kubeplay

helm install kubeplay/kubeplay-arm32v7 --name helm-rpi

NAME:   helm-rpi
LAST DEPLOYED: Sun Jul 15 19:23:58 2018
NAMESPACE: default
STATUS: DEPLOYED

RESOURCES:
==> v1/Service
NAME                       TYPE      CLUSTER-IP     EXTERNAL-IP  PORT(S)         AGE
helm-rpi-kubeplay-arm32v7  NodePort  10.96.168.184  <none>       8005:31439/TCP  2s

==> v1beta1/Deployment
NAME                       DESIRED  CURRENT  UP-TO-DATE  AVAILABLE  AGE
helm-rpi-kubeplay-arm32v7  1        0        0           0          2s

Check the installation

helm ls

NAME            REVISION        UPDATED                         STATUS          CHART                   NAMESPACE
helm-rpi        1               Sun Jul 15 19:23:58 2018        DEPLOYED        kubeplay-arm32v7-0.1.0  default

Check with kubectl if the POD is ready

kubectl get all -n default

NAME                                             READY     STATUS    RESTARTS   AGE
pod/helm-rpi-kubeplay-arm32v7-58567c4756-fj968   0/1       Running   4          4m

NAME                                TYPE        CLUSTER-IP      EXTERNAL-IP   PORT(S)          AGE
service/helm-rpi-kubeplay-arm32v7   NodePort    10.96.168.184   <none>        8005:31439/TCP   4m
service/kubernetes                  ClusterIP   10.96.0.1       <none>        443/TCP          9d

NAME                                        DESIRED   CURRENT   UP-TO-DATE   AVAILABLE   AGE
deployment.apps/helm-rpi-kubeplay-arm32v7   1         1         1            0           4m

NAME                                                   DESIRED   CURRENT   READY     AGE
replicaset.apps/helm-rpi-kubeplay-arm32v7-58567c4756   1         1         0         4m

Helm chart is not really consistent. Current image hardcoded to listen on port 80 Let’s use helm upgrade to change the value

helm upgrade --set service.internalPort=80 helm-rpi kubeplay/kubeplay-arm32v7

Release "helm-rpi" has been upgraded. Happy Helming!
LAST DEPLOYED: Sun Jul 15 20:39:17 2018
NAMESPACE: default
STATUS: DEPLOYED

RESOURCES:
==> v1beta1/Deployment
NAME                       DESIRED  CURRENT  UP-TO-DATE  AVAILABLE  AGE
helm-rpi-kubeplay-arm32v7  1        1        0           0          1h

==> v1/Pod(related)
NAME                                        READY  STATUS       RESTARTS  AGE
helm-rpi-kubeplay-arm32v7-58567c4756-fj968  0/1    Terminating  25        1h
helm-rpi-kubeplay-arm32v7-6cb66496c6-9w97m  0/1    Pending      0         0s

==> v1/Service
NAME                       TYPE      CLUSTER-IP     EXTERNAL-IP  PORT(S)         AGE
helm-rpi-kubeplay-arm32v7  NodePort  10.96.168.184  <none>       8005:31439/TCP  1h

The pod seems to now be running properly.

kubectl get all -n default

NAME                                             READY     STATUS    RESTARTS   AGE
pod/helm-rpi-kubeplay-arm32v7-6cb66496c6-9w97m   1/1       Running   0          3m

NAME                                TYPE        CLUSTER-IP      EXTERNAL-IP   PORT(S)          AGE
service/helm-rpi-kubeplay-arm32v7   NodePort    10.96.168.184   <none>        8005:31439/TCP   1h
service/kubernetes                  ClusterIP   10.96.0.1       <none>        443/TCP          9d

NAME                                        DESIRED   CURRENT   UP-TO-DATE   AVAILABLE   AGE
deployment.apps/helm-rpi-kubeplay-arm32v7   1         1         1            1           1h

NAME                                                   DESIRED   CURRENT   READY     AGE
replicaset.apps/helm-rpi-kubeplay-arm32v7-58567c4756   0         0         0         1h
replicaset.apps/helm-rpi-kubeplay-arm32v7-6cb66496c6   1         1         1         3m

Check the POD

$ kubectl describe pod/helm-rpi-kubeplay-arm32v7-6cb66496c6-9w97m
Name:           helm-rpi-kubeplay-arm32v7-6cb66496c6-9w97m
Namespace:      default
Node:           nas-pi/192.168.1.93
Start Time:     Sun, 15 Jul 2018 20:39:18 +0000
Labels:         app=kubeplay-arm32v7
                pod-template-hash=2762205272
                release=helm-rpi
Annotations:    <none>
Status:         Running
IP:             10.244.2.6
Controlled By:  ReplicaSet/helm-rpi-kubeplay-arm32v7-6cb66496c6
Containers:
  kubeplay-arm32v7:
    Container ID:   docker://737cd78bebb982c782dae2c07f4604ba71f45bc112e91aa20652e29b25bf5d63
    Image:          jbrette/kubeplay-arm32v7:0.1.0
    Image ID:       docker-pullable://jbrette/kubeplay-arm32v7@sha256:86441999035e35514fc647ff9af90bdfc152299636b48b9d3dba9107574f3957
    Port:           80/TCP
    Host Port:      0/TCP
    State:          Running
      Started:      Sun, 15 Jul 2018 20:39:23 +0000
    Ready:          True
    Restart Count:  0
    Liveness:       http-get http://:80/ delay=0s timeout=1s period=10s #success=1 #failure=3
    Readiness:      http-get http://:80/ delay=0s timeout=1s period=10s #success=1 #failure=3
    Environment:    <none>
    Mounts:
      /var/run/secrets/kubernetes.io/serviceaccount from default-token-c5v7s (ro)
Conditions:
  Type              Status
  Initialized       True
  Ready             True
  ContainersReady   True
  PodScheduled      True
Volumes:
  default-token-c5v7s:
    Type:        Secret (a volume populated by a Secret)
    SecretName:  default-token-c5v7s
    Optional:    false
QoS Class:       BestEffort
Node-Selectors:  <none>
Tolerations:     node.kubernetes.io/not-ready:NoExecute for 300s
                 node.kubernetes.io/unreachable:NoExecute for 300s
Events:
  Type    Reason     Age   From               Message
  ----    ------     ----  ----               -------
  Normal  Scheduled  10m   default-scheduler  Successfully assigned default/helm-rpi-kubeplay-arm32v7-6cb66496c6-9w97m to nas-pi
  Normal  Pulling    10m   kubelet, nas-pi    pulling image "jbrette/kubeplay-arm32v7:0.1.0"
  Normal  Pulled     10m   kubelet, nas-pi    Successfully pulled image "jbrette/kubeplay-arm32v7:0.1.0"
  Normal  Created    10m   kubelet, nas-pi    Created container
  Normal  Started    10m   kubelet, nas-pi    Started container

Let’s open a browser towards the URL:

Conclusion

  • Need to cleanup the helm chart and docker image to ensure consistency of the internal
  • Need to improve the NodePort/ClusterIP/LoadBalancer handling.
Last updated on 13 Jul 2018
Published on 13 Jul 2018
 Edit on GitHub