Skip to content

Commit e32e5aa

Browse files
committed
initial commit
0 parents  commit e32e5aa

12 files changed

+457
-0
lines changed

.helmignore

+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
# Patterns to ignore when building packages.
2+
# This supports shell glob matching, relative path matching, and
3+
# negation (prefixed with !). Only one pattern per line.
4+
.DS_Store
5+
# Common VCS dirs
6+
.git/
7+
.gitignore
8+
.bzr/
9+
.bzrignore
10+
.hg/
11+
.hgignore
12+
.svn/
13+
# Common backup files
14+
*.swp
15+
*.bak
16+
*.tmp
17+
*.orig
18+
*~
19+
# Various IDEs
20+
.project
21+
.idea/
22+
*.tmproj
23+
.vscode/

Chart.yaml

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
apiVersion: v2
2+
name: redis-cluster
3+
description: A redis-cluster helm chart
4+
type: application
5+
version: 0.1.0
6+
appVersion: "0.1.0"

README.md

+41
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
2+
## Redis Cluster Kubernetes Helm Charts
3+
4+
This repository contains Redis cluster helm charts that can survive pod restarts in which the pod IPs get changed afterwards, the cluster will be recovered successfully.
5+
6+
![Redis-Cluster]()
7+
8+
### Parameters
9+
10+
| Name | Description | Example |
11+
|--|--|--|
12+
`nameOverride` | String to override `redis-cluster.name` template | `redis-cluster`
13+
`fullnameOverride` | String to override `redis-cluster.fullname` `template` | `redis-cluster`
14+
`redis.image` | Redis docker image | `redis:7.2.3`
15+
`redis.port` | Redis server port | `6379`
16+
`redis.bus` | Redis cluster bus port | `16379`
17+
`redis.resources` | The resources of the redis container | `{}`
18+
`cluster.init` | A boolean to specify whether the cluster should be initialized. (Can be false when cluster is already created and maybe you just want to change the resources of the cluster) | `true`
19+
`cluster.master` | Number of master nodes | `3`
20+
`cluster.replicas` | Number of replicas of each master | `1`
21+
`metrics.enabled` | Turn on/off Redis exporter | `true`
22+
`metrics.image` | Docker image of Redis exporter | `oliver006/redis_exporter:v1.55.0`
23+
`metrics.resources` | Resources of metrics container | `{}`
24+
`metrics.serviceMonitor.enabled` | Create a service monitor if `metrics` is enabled | `true`
25+
`metrics.serviceMonitor.interval` | Metrics scraping interval | `30s`
26+
`service.enabled` | Create a service to access the cluster | `true`
27+
`service.type` | Service type | `ClusterIP`
28+
`persistence.storage` | volume storage | `2Gi`
29+
`persistence.accessModes` | volume access modes | `["ReadWriteOnce"]`
30+
`persistence.storageClassName`| Storage class name of PVC | `ceph-rdb`
31+
`extraConfig`| Passing extra key-value configs to redis.conf | `{}`
32+
33+
Please see the values.yaml to see how these parameters are used.
34+
35+
### Deploy
36+
37+
Clone the repo and `cd` to the repo directory.
38+
39+
```
40+
helm upgrade -i redis-cluster . -f values.yaml
41+
```

templates/_helpers.tpl

+93
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,93 @@
1+
{{/*
2+
Expand the name of the chart.
3+
*/}}
4+
{{- define "redis-cluster.name" -}}
5+
{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" }}
6+
{{- end }}
7+
8+
{{/*
9+
Create a default fully qualified app name.
10+
We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec).
11+
If release name contains chart name it will be used as a full name.
12+
*/}}
13+
{{- define "redis-cluster.fullname" -}}
14+
{{- if .Values.fullnameOverride }}
15+
{{- .Values.fullnameOverride | trunc 63 | trimSuffix "-" }}
16+
{{- else }}
17+
{{- $name := default .Chart.Name .Values.nameOverride }}
18+
{{- if contains $name .Release.Name }}
19+
{{- .Release.Name | trunc 63 | trimSuffix "-" }}
20+
{{- else }}
21+
{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" }}
22+
{{- end }}
23+
{{- end }}
24+
{{- end }}
25+
26+
{{/*
27+
Create chart name and version as used by the chart label.
28+
*/}}
29+
{{- define "redis-cluster.chart" -}}
30+
{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }}
31+
{{- end }}
32+
33+
{{/*
34+
Common labels
35+
*/}}
36+
{{- define "redis-cluster.labels" -}}
37+
helm.sh/chart: {{ include "redis-cluster.chart" . }}
38+
{{ include "redis-cluster.selectorLabels" . }}
39+
{{- if .Chart.AppVersion }}
40+
app.kubernetes.io/version: {{ .Chart.AppVersion | quote }}
41+
{{- end }}
42+
app.kubernetes.io/managed-by: {{ .Release.Service }}
43+
{{- end }}
44+
45+
{{/*
46+
Selector labels
47+
*/}}
48+
{{- define "redis-cluster.selectorLabels" -}}
49+
app.kubernetes.io/name: {{ include "redis-cluster.name" . }}
50+
app.kubernetes.io/instance: {{ .Release.Name }}
51+
{{- end }}
52+
53+
{{/*
54+
Create the name of the service account to use
55+
*/}}
56+
{{- define "redis-cluster.serviceAccountName" -}}
57+
{{- if .Values.serviceAccount.create }}
58+
{{- default (include "redis-cluster.fullname" .) .Values.serviceAccount.name }}
59+
{{- else }}
60+
{{- default "default" .Values.serviceAccount.name }}
61+
{{- end }}
62+
{{- end }}
63+
64+
{{/*
65+
replicaCount = master + master * replica
66+
*/}}
67+
{{- define "redis-cluster.replicaCount" -}}
68+
{{- mul .Values.cluster.master .Values.cluster.replicas | add .Values.cluster.master }}
69+
{{- end }}
70+
71+
{{- define "redis-cluster.addToConfig" -}}
72+
{{- if eq (printf "%s" .value) "" }}
73+
{{ .key }} ""
74+
{{- else }}
75+
{{ .key }} {{ .value }}
76+
{{- end }}
77+
{{- end }}
78+
79+
{{- define "redis-cluster.getHostPorts" -}}
80+
{{- $hostPorts := "" -}}
81+
{{- range $i := until (include "redis-cluster.replicaCount" . | int) }}
82+
{{- $hostPorts = printf "%s%s-%d.%s-headless.%s.svc.cluster.local:%d " $hostPorts (include "redis-cluster.name" $) $i (include "redis-cluster.name" $) $.Release.Namespace ($.Values.redis.port | int) }}
83+
{{- end }}
84+
{{- $hostPorts }}
85+
{{- end }}
86+
87+
{{- define "redis-cluster.getRedisAddrs" -}}
88+
{{- $hostPorts := "" -}}
89+
{{- range $i := until (include "redis-cluster.replicaCount" . | int) }}
90+
{{- $hostPorts = printf "%sredis://%s-%d.%s-headless.%s.svc.cluster.local:%d " $hostPorts (include "redis-cluster.name" $) $i (include "redis-cluster.name" $) $.Release.Namespace ($.Values.redis.port | int) }}
91+
{{- end }}
92+
{{- $hostPorts | trimSuffix " " }}
93+
{{- end }}

templates/configmap.yaml

+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
apiVersion: v1
2+
kind: ConfigMap
3+
metadata:
4+
name: {{ include "redis-cluster.name" . }}-config-map
5+
namespace: {{ .Release.Namespace | quote }}
6+
labels: {{ include "redis-cluster.labels" . | nindent 4 }}
7+
data:
8+
redis.conf: |
9+
dir /data/redis
10+
cluster-enabled yes
11+
cluster-config-file nodes.conf
12+
port {{ .Values.redis.port }}
13+
cluster-port {{ .Values.redis.bus }}
14+
cluster-preferred-endpoint-type hostname
15+
{{- range $key, $value := $.Values.extraConfig }}
16+
{{- include "redis-cluster.addToConfig" (dict "key" $key "value" $value) | indent 4 }}
17+
{{- end }}

templates/job.yaml

+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
{{- if .Values.cluster.init }}
2+
apiVersion: batch/v1
3+
kind: Job
4+
metadata:
5+
name: {{ include "redis-cluster.name" . }}-job
6+
namespace: {{ .Release.Namespace | quote }}
7+
labels: {{ include "redis-cluster.labels" . | nindent 4 }}
8+
spec:
9+
template:
10+
spec:
11+
restartPolicy: "OnFailure"
12+
initContainers:
13+
- name: ping-redises
14+
image: atkrad/wait4x:2.13.0
15+
command:
16+
- /bin/sh
17+
- /scripts/ping.sh
18+
volumeMounts:
19+
- name: scripts
20+
mountPath: /scripts
21+
containers:
22+
- name: init-cluster
23+
image: {{ .Values.redis.image }}
24+
command:
25+
- /bin/bash
26+
- /scripts/init-cluster.sh
27+
volumeMounts:
28+
- name: scripts
29+
mountPath: /scripts
30+
volumes:
31+
- name: scripts
32+
configMap:
33+
name: {{ include "redis-cluster.name" . }}-scripts
34+
{{- end }}

templates/scripts.yaml

+56
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,56 @@
1+
apiVersion: v1
2+
kind: ConfigMap
3+
metadata:
4+
name: {{ include "redis-cluster.name" . }}-scripts
5+
namespace: {{ .Release.Namespace | quote }}
6+
labels: {{ include "redis-cluster.labels" . | nindent 4 }}
7+
data:
8+
init-cluster.sh: |
9+
#!/bin/bash
10+
11+
redis-cli --cluster create --cluster-yes {{ include "redis-cluster.getHostPorts" . }} --cluster-replicas {{ .Values.cluster.replicas }}
12+
13+
run.sh: |
14+
#!/bin/bash
15+
16+
HOSTNAME=$(eval "hostname")
17+
redis-server /etc/redis/redis.conf --cluster-announce-hostname ${HOSTNAME}.{{ include "redis-cluster.name" . }}-headless.{{ .Release.Namespace }}.svc.cluster.local
18+
19+
fix-ips.sh: |
20+
#!/bin/bash
21+
22+
CLUSTER_CONFIG="/data/redis/nodes.conf"
23+
24+
if [ -f ${CLUSTER_CONFIG} ]; then
25+
OLD_IP=$(eval "cat ${CLUSTER_CONFIG} | grep 'myself' | cut -d' ' -f2 | cut -d'@' -f1 | cut -d':' -f1")
26+
27+
if [ "${OLD_IP}" == "${POD_IP}" ]; then
28+
echo "IPs match."
29+
elif [ "${OLD_IP}" == "" ]; then
30+
echo "OLD_IP is empty, make sure the cluster is initialized."
31+
elif [ "${POD_IP}" == "" ]; then
32+
echo "NEW_IP is empty."
33+
exit 1
34+
else
35+
echo "IPs don't match. OLD_IP=${OLD_IP}, NEW_IP=${POD_IP}."
36+
sed -i.bak -e "/myself/ s/[0-9]\{1,3\}\.[0-9]\{1,3\}\.[0-9]\{1,3\}\.[0-9]\{1,3\}/${POD_IP}/" ${CLUSTER_CONFIG}
37+
echo "Node IP was updated successfully, new nodes.conf:"
38+
cat ${CLUSTER_CONFIG} | grep "myself"
39+
fi
40+
else
41+
echo "${CLUSTER_CONFIG} doesn't exist, no need to update IPs."
42+
fi
43+
44+
ping.sh: |
45+
#!/bin/sh
46+
47+
REDIS_ADDRS="{{ include "redis-cluster.getRedisAddrs" . }}"
48+
49+
for addr in ${REDIS_ADDRS}; do
50+
echo "PINGING ${addr}"
51+
wait4x redis ${addr} --timeout 600s --connection-timeout 600s --interval 5s
52+
echo "PONG!"
53+
echo ""
54+
done
55+
56+
echo "All Redises are ready!"

0 commit comments

Comments
 (0)