在K8S中Label是一等公民,跟Pod有相同的地位,标签不仅可以提供给工程师查看,也是可以被K8S系统玩出各种花样来
在配置文件中可以为资源对象打上多个标签
每个标签都是键值对,我们可以将“键”或者“键值对”作为过滤条件过滤出我们想要的资源对象,例如Pods,Namespaces或者Nodes
操作标签有两种手段,一是命令行,二是配置文件

例:

我们为一个名为my-nginx的Pod创建几个自定义标签

1. 首先创建该Pod:

# vim my-nginx.yaml
apiVersion: v1
kind: Pod
metadata:
  creationTimestamp: null
  labels:
    run: my-nginx
  name: my-nginx
spec:
  containers:
  - image: nginx:1.23
    name: my-nginx
    resources: {}
  dnsPolicy: ClusterFirst
  restartPolicy: Always
# kubectl apply -f my-nginx.yaml 
pod/my-nginx created
# kubectl get pods 
NAME       READY   STATUS    RESTARTS       AGE
busybox    1/1     Running   48 (12m ago)   2d
my-nginx   1/1     Running   0              2m12s
nginx      1/1     Running   0              2d

查看一下该Pod是否存在默认的标签值。从结果中我们能够看到该Pod的默认标签是 Labels: run=my-nginx

# kubectl describe pods my-nginx
Name:         my-nginx
Namespace:    default
Priority:     0
Node:         node-2/222.1.1.22
Start Time:   Tue, 11 Oct 2022 14:01:15 +0000
Labels:       run=my-nginx
Annotations:  cni.projectcalico.org/containerID: cb62225509217afb6e4b3e6777b76a3ef19139f244aca6774330d21905061a7c
              cni.projectcalico.org/podIP: 10.200.247.2/32
              cni.projectcalico.org/podIPs: 10.200.247.2/32
Status:       Running
...

2. 然后我们用两种方式为该Pod增删标签:

2.1 命令行添加:

为该Pod增加一个标签test1=aaa

# kubectl label pods my-nginx test1=aaa
pod/my-nginx labeled

再次查看该Pod的标签情况,可以发现标签信息中增加了一个名为 test1=aaa 的标签

# kubectl describe pods my-nginx
Name:         my-nginx
Namespace:    default
Priority:     0
Node:         node-2/222.1.1.22
Start Time:   Tue, 11 Oct 2022 14:01:15 +0000
Labels:       run=my-nginx
              test1=aaa
Annotations:  cni.projectcalico.org/containerID: cb62225509217afb6e4b3e6777b76a3ef19139f244aca6774330d21905061a7c
              cni.projectcalico.org/podIP: 10.200.247.2/32
              cni.projectcalico.org/podIPs: 10.200.247.2/32
Status:       Running

通过以下4种方法我们都可以过滤出我们创建的名为my-nginx的Pod

root@node-1:~# kubectl get pods -l test1
NAME       READY   STATUS    RESTARTS   AGE
my-nginx   1/1     Running   0          27m

root@node-1:~# kubectl get pods -l test1=aaa
NAME       READY   STATUS    RESTARTS   AGE
my-nginx   1/1     Running   0          27m

root@node-1:~# kubectl get pods -l run
NAME       READY   STATUS    RESTARTS   AGE
my-nginx   1/1     Running   0          28m

root@node-1:~# kubectl get pods -l run=my-nginx
NAME       READY   STATUS    RESTARTS   AGE
my-nginx   1/1     Running   0          28m

我们怎么通过命令行删除刚刚增加的标签呢?在标签的键名后加一个减号就OK啦!

# kubectl label pods my-nginx test1-    
pod/my-nginx unlabeled

如果想要覆盖原有标签键的值,则执行如下指令。注意 –overwrite是一个必要的后缀,否则将报错:

# kubectl label pods my-nginx test1=bbb --overwrite
pod/my-nginx labeled

再次查看该Pod的标签,我们使用一个新的指令来查询,该指令就是在get指令后添加后缀 –show-labels
如此一来我们可以很方便的查询Pod当前携带的标签键值对

# kubectl get pods my-nginx --show-labels
NAME       READY   STATUS    RESTARTS   AGE   LABELS
my-nginx   1/1     Running   0          55m   run=my-nginx,test1=bbb

总结一下跟标签相关的指令

增:kubectl label pods my-nginx test1=aaa

删:kubectl label pods my-nginx test1-

改:kubectl label pods my-nginx test1=bbb --overwrite

查:
kubectl describe pods my-nginx
kubectl get pods -l test1
kubectl get pods -l test1=aaa
kubectl get pods my-nginx --show-labels

2.2 配置文件添加:

我们将my-nginx的配置文件修改为如下所示,即增加了一个标签 test2: bbb

apiVersion: v1
kind: Pod
metadata:
  creationTimestamp: null
  labels:
    run: my-nginx
    test2: bbb
  name: my-nginx
spec:
  containers:
  - image: nginx:1.23
    name: my-nginx
    resources: {}
  dnsPolicy: ClusterFirst
  restartPolicy: Always
status: {}

然后重新apply该配置文件

# kubectl apply -f my-nginx.yaml 
pod/my-nginx configured

查看该Pod的标签信息,我们发现my-nginx的标签信息中增加了标签test2=bbb,但是之前通过命令行添加的test1=bbb仍然存在,命令行增加的标签不会被配置文件中的标签配置

# kubectl get pods --show-labels
NAME       READY   STATUS    RESTARTS       AGE    LABELS
busybox    1/1     Running   49 (23m ago)   2d1h   <none>
my-nginx   1/1     Running   0              73m    run=my-nginx,test1=bbb,test2=bbb
nginx      1/1     Running   0              2d1h   <none>

开脑洞时间到,非战斗人员请迅速离场,并空降到下一小节

在这里我们思绪飞扬一下,如果配置文件中将命令行配置的标签test1=bbb做一下修改,是否会被覆盖呢?
首先修改配置文件如下所示:

apiVersion: v1
kind: Pod
metadata:
  creationTimestamp: null
  labels:
    run: my-nginx
    test1: abc
    test2: bbb
  name: my-nginx
spec:
  containers:
  - image: nginx:1.23
    name: my-nginx
    resources: {}
  dnsPolicy: ClusterFirst
  restartPolicy: Always
status: {}

然后应用该配置文件:

# kubectl apply -f my-nginx.yaml 
pod/my-nginx configured

揭晓答案的时候到了,查看此时my-nginx的标签信息:
答案是:配置文件将命令行生成的标签键的键值给覆盖掉了,从原来的bbb变为abc

# kubectl get pods my-nginx --show-labels     
NAME       READY   STATUS    RESTARTS   AGE   LABELS
my-nginx   1/1     Running   0          81m   run=my-nginx,test1=abc,test2=bbb

那么命令行是否可以将test1=abc给改回来呢?

root@node-1:~# kubectl label pods my-nginx test1=bbb --overwrite
pod/my-nginx labeled
root@node-1:~# kubectl get pods my-nginx --show-labels
NAME       READY   STATUS    RESTARTS   AGE   LABELS
my-nginx   1/1     Running   0          87m   run=my-nginx,test1=bbb,test2=bbb

答案是:可以
结论:只要键值对的键名不同,命令行和配置文件就相安无事,否则必将是一场标签争夺战

以上就是我们的废话脑洞

3. 标签使用案例

标签的作用可以等同于Excel表格中的过滤功能

实验场景:我们有3个Pod都在运行nginx,属性如下表所示

主机

版本

部门

nginx1

old

test

nginx2

old

dev

nginx3

new

test

按照如上要求为nginx1,nginx2,nginx3打上了标签

root@node-1:~# kubectl get pods --show-labels            
NAME       READY   STATUS      RESTARTS       AGE     LABELS
busybox    0/1     Completed   49 (60m ago)   2d2h    <none>
my-nginx   1/1     Running     0              109m    run=my-nginx,test1=bbb,test2=bbb
nginx      1/1     Running     0              2d1h    <none>
nginx1     1/1     Running     0              7m6s    department=test,special=special,version=old
nginx2     1/1     Running     0              7m3s    department=dev,version=old
nginx3     1/1     Running     0              6m59s   department=test,version=new

下面我们开始通过标签来选择性的查看pod:

筛选部门为test的Pod

root@node-1:~# kubectl get pods -l department=test
NAME     READY   STATUS    RESTARTS   AGE
nginx1   1/1     Running   0          9m1s
nginx3   1/1     Running   0          8m54s

筛选部门为dev的Pod

root@node-1:~# kubectl get pods -l department=dev
NAME     READY   STATUS    RESTARTS   AGE
nginx2   1/1     Running   0          9m29s

筛选版本为旧的Pod

root@node-1:~# kubectl get pods -l version=old
NAME     READY   STATUS    RESTARTS   AGE
nginx1   1/1     Running   0          10m
nginx2   1/1     Running   0          10m

筛选版本为新的Pod

root@node-1:~# kubectl get pods -l version=new
NAME     READY   STATUS    RESTARTS   AGE
nginx3   1/1     Running   0          10m

筛选出带有特殊标签的Pod

root@node-1:~# kubectl get pods -l special
NAME     READY   STATUS    RESTARTS   AGE
nginx1   1/1     Running   0          10m