mirror of
https://github.com/itzg/mc-router.git
synced 2025-01-21 20:51:39 +01:00
auto scale up and down with many named backends in k8s (#270)
This commit is contained in:
parent
555033b54d
commit
5dc576fc2e
278
docs/k8s-autoscale.yaml
Normal file
278
docs/k8s-autoscale.yaml
Normal file
@ -0,0 +1,278 @@
|
||||
|
||||
# This YAML is an example and is not intended to be directly applied.
|
||||
# It consists of 3 parts
|
||||
# 1. the mc-router with service account an service
|
||||
# 2. the shutdown cronjob with service account
|
||||
# 3. the actual server with service and storage
|
||||
|
||||
# part 3 is the only part you need to replicate for the number of servers you want
|
||||
|
||||
# mc-router
|
||||
---
|
||||
apiVersion: v1
|
||||
kind: ServiceAccount
|
||||
metadata:
|
||||
name: mc-router
|
||||
---
|
||||
apiVersion: rbac.authorization.k8s.io/v1
|
||||
kind: ClusterRole
|
||||
metadata:
|
||||
name: services-watcher
|
||||
rules:
|
||||
- apiGroups: [""]
|
||||
resources: ["services"]
|
||||
verbs: ["watch","list"]
|
||||
- apiGroups: ["apps"]
|
||||
resources: ["statefulsets", "statefulsets/scale"]
|
||||
verbs: ["watch","list","get","update"]
|
||||
---
|
||||
apiVersion: rbac.authorization.k8s.io/v1
|
||||
kind: ClusterRoleBinding
|
||||
metadata:
|
||||
name: mc-router-services-watcher
|
||||
subjects:
|
||||
- kind: ServiceAccount
|
||||
name: mc-router
|
||||
namespace: default
|
||||
roleRef:
|
||||
kind: ClusterRole
|
||||
name: services-watcher
|
||||
apiGroup: rbac.authorization.k8s.io
|
||||
# Use whatever tcp ingress method you want I just used a node port here for simplicity
|
||||
---
|
||||
apiVersion: v1
|
||||
kind: Service
|
||||
metadata:
|
||||
name: mc-router
|
||||
spec:
|
||||
type: NodePort
|
||||
externalIPs:
|
||||
- 192.168.1.100
|
||||
ports:
|
||||
- targetPort: web
|
||||
name: web
|
||||
port: 8080
|
||||
nodePort: 30001
|
||||
- targetPort: proxy
|
||||
name: proxy
|
||||
port: 25565
|
||||
nodePort: 30000
|
||||
selector:
|
||||
run: mc-router
|
||||
---
|
||||
apiVersion: apps/v1
|
||||
kind: Deployment
|
||||
metadata:
|
||||
labels:
|
||||
run: mc-router
|
||||
name: mc-router
|
||||
spec:
|
||||
selector:
|
||||
matchLabels:
|
||||
run: mc-router
|
||||
strategy:
|
||||
type: Recreate
|
||||
template:
|
||||
metadata:
|
||||
labels:
|
||||
run: mc-router
|
||||
spec:
|
||||
serviceAccountName: mc-router
|
||||
containers:
|
||||
- image: aapjeisbaas/mc-router:latest
|
||||
imagePullPolicy: Always
|
||||
name: mc-router
|
||||
args: ["--api-binding", ":8080", "--in-kube-cluster","--auto-scale-up", "--debug"]
|
||||
env:
|
||||
- name: AUTO_SCALE_UP
|
||||
value: "true"
|
||||
ports:
|
||||
- name: proxy
|
||||
containerPort: 25565
|
||||
- name: web
|
||||
containerPort: 8080
|
||||
resources:
|
||||
requests:
|
||||
memory: 50Mi
|
||||
cpu: "100m"
|
||||
limits:
|
||||
memory: 100Mi
|
||||
cpu: "250m"
|
||||
|
||||
|
||||
# Cron job for stopping empty servers
|
||||
---
|
||||
apiVersion: rbac.authorization.k8s.io/v1
|
||||
kind: ClusterRole
|
||||
metadata:
|
||||
name: mc-shutdown
|
||||
rules:
|
||||
- apiGroups: ["apps"]
|
||||
resources: ["statefulsets", "statefulsets/scale"]
|
||||
verbs: ["list","get","update", "patch"]
|
||||
- apiGroups: [""]
|
||||
resources: ["pods", "pods/log"]
|
||||
verbs: ["get", "list"]
|
||||
- apiGroups: [""]
|
||||
resources: ["pods/exec"]
|
||||
verbs: ["create"]
|
||||
---
|
||||
kind: ClusterRoleBinding
|
||||
apiVersion: rbac.authorization.k8s.io/v1
|
||||
metadata:
|
||||
name: mc-shutdown
|
||||
subjects:
|
||||
- kind: ServiceAccount
|
||||
name: mc-shutdown
|
||||
namespace: default
|
||||
roleRef:
|
||||
kind: ClusterRole
|
||||
name: mc-shutdown
|
||||
apiGroup: "rbac.authorization.k8s.io"
|
||||
---
|
||||
apiVersion: v1
|
||||
kind: ServiceAccount
|
||||
metadata:
|
||||
name: mc-shutdown
|
||||
---
|
||||
apiVersion: batch/v1
|
||||
kind: CronJob
|
||||
metadata:
|
||||
name: mc-shutdown
|
||||
spec:
|
||||
schedule: "*/5 * * * *"
|
||||
concurrencyPolicy: Forbid
|
||||
jobTemplate:
|
||||
spec:
|
||||
template:
|
||||
spec:
|
||||
serviceAccountName: mc-shutdown
|
||||
restartPolicy: OnFailure
|
||||
containers:
|
||||
- name: shutdown
|
||||
image: bitnami/kubectl:latest
|
||||
imagePullPolicy: IfNotPresent
|
||||
command:
|
||||
- /bin/bash
|
||||
- -c
|
||||
- source shutdown-script.sh
|
||||
volumeMounts:
|
||||
- name: shutdown-script
|
||||
mountPath: /shutdown-script.sh
|
||||
subPath: shutdown-script.sh
|
||||
readOnly: true
|
||||
volumes:
|
||||
- name: shutdown-script
|
||||
configMap:
|
||||
name: shutdown-script
|
||||
items:
|
||||
- key: shutdown-script.sh
|
||||
path: shutdown-script.sh
|
||||
# uses container label containertype=minecraft-server to find running servers
|
||||
# TODO: get ownerReferences link to StatefulSet/name from pod metadate instead of sed string manipulation
|
||||
---
|
||||
apiVersion: v1
|
||||
kind: ConfigMap
|
||||
metadata:
|
||||
name: shutdown-script
|
||||
data:
|
||||
shutdown-script.sh: |
|
||||
#!/bin/bash
|
||||
MC_PODS=$(kubectl get pods -l containertype=minecraft-server -o=jsonpath="{range .items[*]}{.metadata.name},"| sed 's/,/\n/g')
|
||||
for p in $MC_PODS; do
|
||||
echo "found minecraft pod $p, sleeping 120 seconds to prevent shutdown before login"
|
||||
sleep 120
|
||||
deployment=$(echo $p |sed 's/-0//g')
|
||||
# check online player count in the mc server
|
||||
if [[ $(kubectl exec -i $p -- /usr/local/bin/mc-monitor status) == *"online=0"* ]] ;then
|
||||
kubectl scale statefulset $deployment --replicas=0
|
||||
fi
|
||||
done
|
||||
|
||||
|
||||
# The actual minecraft servers, services and storage, repeat this block for as many servers as you want
|
||||
# make sure you have the label containertype=minecraft-server this is used to find running servers
|
||||
---
|
||||
apiVersion: apps/v1
|
||||
kind: StatefulSet
|
||||
metadata:
|
||||
name: minecraft-servername-deployment
|
||||
labels:
|
||||
app: minecraft-servername-container
|
||||
spec:
|
||||
serviceName: minecraft-servername-deployment
|
||||
selector:
|
||||
matchLabels:
|
||||
app: minecraft-servername-container
|
||||
replicas: 0
|
||||
template:
|
||||
metadata:
|
||||
labels:
|
||||
app: minecraft-servername-container
|
||||
containertype: minecraft-server
|
||||
spec:
|
||||
containers:
|
||||
- name: minecraft-servername-deployment
|
||||
image: itzg/minecraft-server:latest
|
||||
imagePullPolicy: IfNotPresent
|
||||
resources:
|
||||
limits:
|
||||
memory: "2048Mi"
|
||||
env:
|
||||
# Use secret in real usage
|
||||
- name: EULA
|
||||
value: "true"
|
||||
# let the JVM figure out mem management
|
||||
- name: "MEMORY"
|
||||
value: ""
|
||||
- name: JVM_XX_OPTS
|
||||
value: "-XX:MaxRAMPercentage=75"
|
||||
ports:
|
||||
- containerPort: 25565
|
||||
name: main
|
||||
readinessProbe:
|
||||
exec:
|
||||
command: [ "/usr/local/bin/mc-monitor", "status", "--host", "localhost" ]
|
||||
# Give it i + p * f seconds to be ready, so 120 seconds
|
||||
initialDelaySeconds: 20
|
||||
periodSeconds: 5
|
||||
failureThreshold: 20
|
||||
# Monitor ongoing liveness
|
||||
livenessProbe:
|
||||
exec:
|
||||
command: ["/usr/local/bin/mc-monitor", "status", "--host", "localhost"]
|
||||
initialDelaySeconds: 120
|
||||
periodSeconds: 60
|
||||
volumeMounts:
|
||||
- name: mc-data
|
||||
mountPath: /data
|
||||
volumes:
|
||||
- name: mc-data
|
||||
persistentVolumeClaim:
|
||||
claimName: minecraft-servername-pvc
|
||||
readOnly: false
|
||||
---
|
||||
apiVersion: v1
|
||||
kind: PersistentVolumeClaim
|
||||
metadata:
|
||||
name: minecraft-servername-pvc
|
||||
namespace: default
|
||||
spec:
|
||||
accessModes:
|
||||
- ReadWriteMany
|
||||
resources:
|
||||
requests:
|
||||
storage: 20Gi
|
||||
---
|
||||
apiVersion: v1
|
||||
kind: Service
|
||||
metadata:
|
||||
name: minecraft-servername-deployment
|
||||
namespace: default
|
||||
annotations:
|
||||
"mc-router.itzg.me/externalServerName": "your-awesome-server.public-domain.com"
|
||||
spec:
|
||||
ports:
|
||||
- port: 25565
|
||||
selector:
|
||||
app: minecraft-servername-container
|
Loading…
Reference in New Issue
Block a user