K8S自己动手系列 - 2.5 - StatefulSet & Grafana

本文涉及的产品
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
云数据库 RDS MySQL,高可用系列 2核4GB
简介: 前面的实验,我们通过Deployment+PV/PVC,部署了mysql,wordpress,并通过NodePort类型的Service对服务进行暴露,使集群外可以访问,但是当replicas大于1时,Deployment产生的多个POD是共享一个PV的,这样在性能及业务上都有可能是有问题的,这种情况我们就应该考虑StatefulSet,在官网上StatefulSet用了Mysql的主从作为例子,但是笔者认为Mysql主从本身与Mysql的知识有关,并且例子更偏向于实践拓扑结构的有状态服务,所以后面的实验,我基于目前比较流行,也比较简单的监控系统Grafana+Prometheus来进行实验。

前言

前面的实验,我们通过Deployment+PV/PVC,部署了mysql,wordpress,并通过NodePort类型的Service对服务进行暴露,使集群外可以访问,但是当replicas大于1时,Deployment产生的多个POD是共享一个PV的,这样在性能及业务上都有可能是有问题的,这种情况我们就应该考虑StatefulSet,在官网上StatefulSet用了Mysql的主从作为例子,但是笔者认为Mysql主从本身与Mysql的知识有关,并且例子更偏向于实践拓扑结构的有状态服务,所以后面的实验,我基于目前比较流行,也比较简单的监控系统Grafana+Prometheus来进行实验。

场景

使用StatefulSet部署grafana,使用PV & PVC持久化保存数据。
本文实验所有的源码保存在:
https://github.com/zrbcool/blog-public/tree/master/k8s-hands-on/lab08

动手

准备PV & PVC

 kubectl apply -f 01-1-grafana-pvc.yaml 
persistentvolumeclaim/grafana-pv-claim created
 mkdir -p /data/pv/grafana
 kubectl apply -f 01-2-grafana-pv.yaml 
persistentvolume/grafana-pv-volume created
 kubectl get pvc
NAME               STATUS   VOLUME              CAPACITY   ACCESS MODES   STORAGECLASS   AGE
grafana-pv-claim   Bound    grafana-pv-volume   2Gi        RWX                           24m

如果此处不明白或者有问题,请参考前面讲过的K8S自己动手系列 - 2.3 - PV & PVC

用StatefulSet创建Grafana,使用PVC作为持久化存储

 kubectl apply -f 01-3-grafana-statefulset.yaml 
statefulset.apps/monitor-grafana created
 kubectl get pods
NAME                READY   STATUS             RESTARTS   AGE
monitor-grafana-0   0/1     CrashLoopBackOff   3          2m2s
 kubectl logs monitor-grafana-0
GF_PATHS_DATA='/var/lib/grafana' is not writable.
You may have issues with file permissions, more information here: http://docs.grafana.org/installation/docker/#migration-from-a-previous-version-of-the-docker-container-to-5-1-or-later
mkdir: cannot create directory '/var/lib/grafana/plugins': Permission denied

启动失败了,提示也很清楚是权限问题,也给了解决方案

# 参考文章 http://docs.grafana.org/installation/docker/#migration-from-a-previous-version-of-the-docker-container-to-5-1-or-later
# 修改文件权限
 chown -R 472:472 /data/pv/grafana
# 增加POD配置(我们的01-3-grafana-statefulset.yaml已经加好了)
      securityContext:
        runAsGroup: 472
        runAsUser: 472
        fsGroup: 472

执行上述操作后仍然是不行,最后经过排查发现,POD被调度到worker02节点上了,因为我们的PV使用的Local类型,worker02上并没有这个目录,所以无法创建成功,参考前面的K8S自己动手系列 - 1.3 - Taint & Affinity增加nodeSelector指定节点调度

...
  template:
    metadata:
      labels:
        app: monitor
        type: grafana
    spec:
      nodeSelector:
        kubernetes.io/hostname: worker01 #这个请根据需要修改到你指定的节点上,或者你使用其他的持久化存储方式,则可能没有我这个问题,比如nfs

成功!

 kubectl get pods -o wide
NAME                READY   STATUS    RESTARTS   AGE     IP            NODE       NOMINATED NODE   READINESS GATES
monitor-grafana-0   1/1     Running   0          3m19s   10.244.0.93   worker01   <none>           <none>

最后生效的yaml文件如下:

apiVersion: apps/v1
kind: StatefulSet
metadata:
  labels:
    app: monitor
    type: grafana
  name: monitor-grafana
spec:
  replicas: 1 
  selector:
    matchLabels:
      app: monitor
      type: grafana
  serviceName: grafana
  template:
    metadata:
      labels:
        app: monitor
        type: grafana
    spec:
      nodeSelector:
        kubernetes.io/hostname: worker01 #保证调度到我们创建了本地目录的主机上
      containers:
      - name: grafana
        image: grafana/grafana:latest
        imagePullPolicy: IfNotPresent
        ports:
        - containerPort: 3000
          name: grafana
        volumeMounts:
        - name: grafana-pv-storage
          mountPath: /var/lib/grafana
      securityContext:
        runAsGroup: 472
        runAsUser: 472
        fsGroup: 472
      volumes:
      - name: grafana-pv-storage
        persistentVolumeClaim:
          claimName: grafana-pv-claim

使用Service暴露grafana,使可以在集群外访问

 kubectl apply -f 01-4-grafana-svc.yaml 
service/grafana-svc created

如果这块有疑问,可以回顾K8S自己动手系列 - 2.4 - Service
现在我们的grafana可以访问了,如图所示:

相关实践学习
通过Ingress进行灰度发布
本场景您将运行一个简单的应用,部署一个新的应用用于新的发布,并通过Ingress能力实现灰度发布。
容器应用与集群管理
欢迎来到《容器应用与集群管理》课程,本课程是“云原生容器Clouder认证“系列中的第二阶段。课程将向您介绍与容器集群相关的概念和技术,这些概念和技术可以帮助您了解阿里云容器服务ACK/ACK Serverless的使用。同时,本课程也会向您介绍可以采取的工具、方法和可操作步骤,以帮助您了解如何基于容器服务ACK Serverless构建和管理企业级应用。 学习完本课程后,您将能够: 掌握容器集群、容器编排的基本概念 掌握Kubernetes的基础概念及核心思想 掌握阿里云容器服务ACK/ACK Serverless概念及使用方法 基于容器服务ACK Serverless搭建和管理企业级网站应用
目录
相关文章
|
17天前
|
Kubernetes 容器 Perl
【Azure K8S | AKS】在AKS中创建 StatefulSet 示例
【Azure K8S | AKS】在AKS中创建 StatefulSet 示例
|
29天前
|
Prometheus Kubernetes 监控
Kubernetes(K8S) 监控 Prometheus + Grafana
Kubernetes(K8S) 监控 Prometheus + Grafana
104 2
|
8天前
|
运维 Kubernetes 监控
Loki+Promtail+Grafana监控K8s日志
综上,Loki+Promtail+Grafana 监控组合对于在 K8s 环境中优化日志管理至关重要,它不仅提供了强大且易于扩展的日志收集与汇总工具,还有可视化这些日志的能力。通过有效地使用这套工具,可以显著地提高对应用的运维监控能力和故障诊断效率。
17 0
|
19天前
|
存储 Kubernetes 网络协议
在K8S中,Deployment和Statefulset有何区别?
在K8S中,Deployment和Statefulset有何区别?
|
23天前
|
存储 消息中间件 Kubernetes
在K8S中,deploy和Statefulset有何区别?
在K8S中,deploy和Statefulset有何区别?
|
29天前
|
存储 Kubernetes 关系型数据库
Kubernetes(K8S) Controller - StatefulSet、DaemonSet 介绍
Kubernetes(K8S) Controller - StatefulSet、DaemonSet 介绍
24 0
|
2月前
|
Kubernetes Cloud Native 持续交付
云原生架构的核心组成部分通常包括容器化(如Docker)、容器编排(如Kubernetes)、微服务架构、服务网格、持续集成/持续部署(CI/CD)、自动化运维(如Prometheus监控和Grafana可视化)等。
云原生架构的核心组成部分通常包括容器化(如Docker)、容器编排(如Kubernetes)、微服务架构、服务网格、持续集成/持续部署(CI/CD)、自动化运维(如Prometheus监控和Grafana可视化)等。
|
3月前
|
Kubernetes Ubuntu Linux
k8s部署grafana beyla实现app应用服务依赖图可观测
k8s部署grafana beyla实现app应用服务依赖图可观测
58 4
|
4月前
|
存储 Kubernetes 网络协议
k8s学习-StatefulSet(模板、更新、扩缩容、删除等)
k8s学习-StatefulSet(模板、更新、扩缩容、删除等)
273 0
|
4月前
|
JSON Kubernetes 数据格式
Grafana 系列文章(十三):如何用 Loki 收集查看 Kubernetes Events
Grafana 系列文章(十三):如何用 Loki 收集查看 Kubernetes Events