PI4 Stories

Raspberry Pi 4 cluster Series - Spin up graphite and temperature2celsius pods

Graphite is an enterprise-ready monitoring tool that runs equally well on cheap hardware or Cloud infrastructure. Teams use Graphite to track the performance of their websites, applications, business services, and networked servers. It marked the start of a new generation of monitoring tools, making it easier than ever to store, retrieve, share, and visualize time-series data. Info copied from web-site graphiteapp.org.

Spin up graphite pod

To start we need the sources to build of graphite pod, therefore, clone our pi4-graphite github repository [1]:

git clone  https://github.com/gdha/pi4-graphite.git

You will find in the pi4-graphite directory a Dockerfile which will be used to create a new container image:

To build the docker image on your Pi4 system execute the following command:

$ ./build.sh v1.2
Login Succeeded
Building graphite:v1.2
Sending build context to Docker daemon  137.2kB
Step 1/11 : FROM graphiteapp/graphite-statsd:1.1.10-4
1.1.10-4: Pulling from graphiteapp/graphite-statsd
9981e73032c8: Pull complete 
de219cb0be19: Pull complete 
a152a03e3913: Pull complete 
1e02b3e21032: Pull complete 
fb7d8007a803: Pull complete 
Digest: sha256:fb9eb6fdd8f6073dd4ff1acd2169b693d4633045017701713651befbc62fe9f5
Status: Downloaded newer image for graphiteapp/graphite-statsd:1.1.10-4
...
Successfully built afb29746df49
Successfully tagged ghcr.io/gdha/graphite:v1.2
Pushing graphite:v1.2 to GitHub Docker Container registry

In fact, you do not need to build it yourself as GitHub Actions of Package pi4-grahite will do this for us.

To start the graphite pod on our kubernetes cluster do the following:

$ cd kubernetes
$ kubectl apply -f ./graphite-namespace.yaml 
namespace/graphite created
$ kubectl apply -f ./graphite-secret.yaml 
secret/graphite created
$ kubectl apply -f ./ghcr-secret.yaml
secret/dockerconfigjson-github-com created
## Be careful - longhorn has to be installed already before applying persistentvolumeclaim-graphite.yaml
$ kubectl apply -f ./persistentvolumeclaim-graphite.yaml
persistentvolumeclaim/graphite created
$ kubectl apply -f ./statefulset_graphite.yaml
statefulset.apps/graphite created
$ kubectl apply -f ./service_graphite.yaml 
service/graphite-svc created

Open a browser with URL http://n1:30080 to see the result of the tests:

Deploy temperature2celsius pods

The purpose of this pod is to demonstrate that we can send information to our graphite application. We choose to send the temperatire in Celsius once a minute.

Start with cloning our pi4-temperature2graphite github repository [2]:

git clone https://github.com/gdha/pi4-temperature2graphite.git
cd pi4-temperature2graphite

The Dockerfile content is:

# Dockerfile to create the container used to send the temperature of the RPI4
# to the graphite pod
#FROM gdha/rpi-alpine-rootfs/alpine:v1.37
FROM alpine:latest

LABEL org.opencontainers.image.sourcec=https://github.com/gdha/pi4-temperature2graphite
LABEL org.opencontainers.image.description "pi4-temperature2graphite build for the ARM64"
LABEL org.opencontainers.image.licenses "GPL-3.0-or-later"
LABEL maintainer "Gratien Dhaese <gratien.dhaese@gmail.com>"

COPY entrypoint.sh                                  /entrypoint.sh
COPY k3s                                            /usr/bin/kubectl
COPY api_query.sh                                   /api_query.sh

# Update and install dependencies
RUN  apk add --update nodejs npm curl
RUN  chmod a+x                                      /entrypoint.sh \
     && chmod a+x                                   /api_query.sh  \
     && echo "Europe/Brussels" >                    /etc/timezone  \
     && chmod 755 /root

ENTRYPOINT ["/entrypoint.sh"]

And, the center of the image is the /entrypoint.sh script:

#!/bin/sh

# As this script runs on Alpine (busybox) we cannot use bash syntax
#set -e
POD=$(echo $HOSTNAME)
HOSTNAME=$(cat /etc/hostname)
#SERVER=$(/usr/local/bin/kubectl -n graphite describe pod graphite-0 | grep -i node: | cut  -d/ -f2)
while true
do
  # move the SERVER line inside the loop as at each restart the graphite pod gets a new IP address
  SERVER=$(/usr/bin/kubectl -n graphite get pods -o wide | tail -1 | awk '{print $6}')
  cpu_temp=$(cat /sys/class/thermal/thermal_zone0/temp)
  #cpu_temp=$(($cpu_temp/1000))
  cpu_temp=$(expr $cpu_temp / 1000)
  echo "carbon.celsius.$HOSTNAME $cpu_temp $(date +%s)" | timeout 2 nc $SERVER 2003 
  if [[ $? -eq 0 ]] ; then
    echo $(date '+%F %T') {"caller":"entrypoint.sh:16","pod":"$POD","level":"info","msg":"temperature $cpu_temp"}
  else
    echo $(date '+%F %T') {"caller":"entrypoint.sh:18","pod":"$POD","level":"error","msg":"cannot connect to server $SERVER"}
  fi
  # echo $cpu_temp
  sleep 60
done

To build the image use the build.sh script and it pushes the image to ghcr.io/gdha/pi4-temperature2graphite:v1.7 (which is also the latest). To build version v1.7 we did the following:

./build v1.7

To bring the pods alive on our kubernetes cluster go to the kubernetes directory and execute:

$ kubectl apply -f celsius-namespace.yaml
namespace/celsius created

 kubectl apply -f celsius-secret.yaml
secret/celsius created

$ kubectl apply -f ghcr-secret.yaml 
secret/dockerconfigjson-github-com created

$ kubectl apply -f celsius-rbac.yaml 
erviceaccount/celsius-sa created
clusterrole.rbac.authorization.k8s.io/celsius-cluster-role created
rolebinding.rbac.authorization.k8s.io/celsius-cluster-role-binding-ns-celsius created
rolebinding.rbac.authorization.k8s.io/celsius-cluster-role-binding-ns-graphite created

$ kubectl apply -f celsius-deployment.yaml
deployment.apps/celsius created

$ kubectl get pods -n celsius -w
NAME                       READY   STATUS              RESTARTS   AGE
celsius-6ffb4f4bcc-jl579   0/1     ContainerCreating   0          16s
celsius-6ffb4f4bcc-kh4dh   0/1     ContainerCreating   0          16s
celsius-6ffb4f4bcc-qzt8w   0/1     ContainerCreating   0          16s
celsius-6ffb4f4bcc-nfgnx   0/1     ContainerCreating   0          16s
celsius-6ffb4f4bcc-nptk6   0/1     ContainerCreating   0          16s
celsius-6ffb4f4bcc-kh4dh   1/1     Running             0          81s
celsius-6ffb4f4bcc-nfgnx   1/1     Running             0          83s
celsius-6ffb4f4bcc-nptk6   1/1     Running             0          84s
celsius-6ffb4f4bcc-jl579   1/1     Running             0          86s
celsius-6ffb4f4bcc-qzt8w   1/1     Running             0          89s

$ kubectl logs -n celsius celsius-6ffb4f4bcc-qzt8w
INFO[0000] Acquiring lock file /var/lib/rancher/k3s/data/.lock 
INFO[0000] Preparing data dir /var/lib/rancher/k3s/data/8c4262cf7fdd652cccb03a99a99fdffc96d9ad41d7e57af9eb08c7ac2867c72a 

After a couple of minutes you can check the graphite site again:

When you would like to replace the pi4-temperature2graphite container with a newer version in the k3s cluster edit the file kubernetes/celsius-deployment.yaml and update the line containing the image name and replace the old version number with the new one, e.g. v1.8:

image: ghcr.io/gdha/pi4-temperature2graphite:v1.8

To perform a rolling upgrade and watch the upgrade process execute the commands:

$ kubectl replace -f kuvernetes/celsius-deployment.yaml
$ kubectl get pods -n celsius -w
NAME                       READY   STATUS              RESTARTS       AGE
celsius-6ffb4f4bcc-qzt8w   1/1     Running             3 (134m ago)   14d
celsius-6ffb4f4bcc-kh4dh   1/1     Running             3 (134m ago)   14d
celsius-6ffb4f4bcc-nptk6   1/1     Running             3 (134m ago)   14d
celsius-6ffb4f4bcc-nfgnx   1/1     Running             3 (134m ago)   14d
celsius-6ffb4f4bcc-jl579   1/1     Terminating         3 (135m ago)   14d
celsius-779654f68d-9lmq8   0/1     ContainerCreating   0              10s
celsius-779654f68d-d56q6   0/1     ContainerCreating   0              10s
celsius-779654f68d-j6m6t   0/1     ContainerCreating   0              10s
celsius-6ffb4f4bcc-jl579   0/1     Terminating         3              14d
celsius-6ffb4f4bcc-jl579   0/1     Terminating         3              14d
celsius-6ffb4f4bcc-jl579   0/1     Terminating         3              14d
celsius-779654f68d-9lmq8   1/1     Running             0              50s
celsius-6ffb4f4bcc-nptk6   1/1     Terminating         3 (135m ago)   14d
celsius-779654f68d-jbxg8   0/1     Pending             0              0s
celsius-779654f68d-jbxg8   0/1     Pending             0              0s
celsius-779654f68d-jbxg8   0/1     ContainerCreating   0              0s
celsius-779654f68d-j6m6t   1/1     Running             0              54s
celsius-6ffb4f4bcc-kh4dh   1/1     Terminating         3 (135m ago)   14d
celsius-779654f68d-q5rf9   0/1     Pending             0              0s
celsius-779654f68d-q5rf9   0/1     Pending             0              0s
celsius-779654f68d-q5rf9   0/1     ContainerCreating   0              0s
celsius-779654f68d-d56q6   1/1     Running             0              57s
celsius-6ffb4f4bcc-qzt8w   1/1     Terminating         3 (135m ago)   14d
celsius-6ffb4f4bcc-nptk6   0/1     Terminating         3              14d
celsius-6ffb4f4bcc-nptk6   0/1     Terminating         3              14d
celsius-6ffb4f4bcc-nptk6   0/1     Terminating         3              14d
celsius-779654f68d-jbxg8   1/1     Running             0              39s
celsius-6ffb4f4bcc-nfgnx   1/1     Terminating         3 (135m ago)   14d
celsius-6ffb4f4bcc-kh4dh   0/1     Terminating         3              14d
celsius-6ffb4f4bcc-kh4dh   0/1     Terminating         3              14d
celsius-6ffb4f4bcc-kh4dh   0/1     Terminating         3              14d
celsius-6ffb4f4bcc-qzt8w   0/1     Terminating         3              14d
celsius-6ffb4f4bcc-qzt8w   0/1     Terminating         3              14d
celsius-6ffb4f4bcc-qzt8w   0/1     Terminating         3              14d
celsius-779654f68d-q5rf9   1/1     Running             0              44s
celsius-6ffb4f4bcc-nfgnx   0/1     Terminating         3 (136m ago)   14d
celsius-6ffb4f4bcc-nfgnx   0/1     Terminating         3 (136m ago)   14d
celsius-6ffb4f4bcc-nfgnx   0/1     Terminating         3 (136m ago)   14d

References

[1] pi4-graphite GitHub repository

[2] pi4-temperature2graphite GitHub repository

Edit history

  • 09/Feb/2023: adding the references
  • 08/Feb/2023: improved the content of temperature2celsius section