Kubernetes 系统化学习之 集群安全篇(七)

news2025/1/16 1:51:24

Kubernetes 作为一个分布式集群管理的工具,保证集群的安全性是其一个重要的任务。API Server 是集群内部各个组件通信的中介,也是外部控制的入口,所以 K8s 的安全机制就是围绕保护 API Server 来设计的。K8s 使用了认证(Authentication)、鉴权(Authorization)、准入控制(Admission Control)三步来保证 API Service 的安全。

1. 用户的概念

认证和授权的过程中最重要的一个概念是用户。所以在正式讲解之前,我们需要了解清楚 k8s 里面的用户的怎样的。

kubernetes 有两种用户:一是 集群内部的 Service Account,二是 外部的用户账户

为什么 kubernetes 中会有 service account 和用户账户这两种形式呢?回答这个问题之前我们要先搞清楚一个问题:谁发起的请求? 回答是:与 kubernetes 交互的实际上有两类东西,一类是真实的人类,另一类就是程序。真实的人类使用的就是用户账户,而某一个程序访问 kubernetes 时就需要使用 service acocunt 了。

用户账户

就是个人用户,比如某个运维人员或外部应用的账户。但是 kubernetes 没有相应的资源对象或者 api 来支持常规的个人账户,拥有 kubernetes 集群的 CA 证书签名的有效证书,个人用户就可以访问 kubernetes 集群了,在这种情况下,证书中的 Subject(主题)会被 api-server 解析成为一个用户。 举个例子,X509 证书中 Subject 的内容如下:O = system:masters, CN = kubernetes:admin ,其中,kubernetes:admin 会被解析为 User,system:masters 会被解析为 User 所在的 Group。

Service Account

一个 Pod 必须要以某一个 Service Account 的身份去运行。一个 Service Account 对应着一个 Secret,一个 Secret 保存着一个 Token 和公钥文件,而这些信息会被挂载到 Pod 中。

2. Api-server 的认证机制

不管是通过外部还是集群内部访问 kubernetes 集群,api-server 都是一个入口,所以,k8s 的认证和授权机制几乎都是 api-server 来实现的。

通过上述对用户的阐述,我们总结如下:任一 kubernetes api 的访问都是以下三种方式之一:

  1. 以证书方式访问的普通用户或进程,包括运维人员及 kubectl、kubelets 等进程;
  2. 以 Service Account 方式访问的 Kubernetes 的内部服务进程;
  3. 以匿名方式访问的进程。

认证方面,kubernetes 提供了如下的认证方式:

  • HTTPS 证书认证:基于 CA 根证书签名的双向数字证书认证方式,比如运维人员操作 kubectl 时就是采用这种认证的方式,证书信息在 $HOME/.kube/config 中保存。
  • HTTP Bear Token 认证:通过一个 Bear Token 识别合法用户,比如 Service Account 就是这种认证方式。我们都知道 Service Account 是和一个 Secret 紧密关联的,每一个关联的 Secret 都保存了一个 Token。这个 Token 就是一个 JWT Token。
  • OpenID Connect Token 第三方认证:通过第三方 OIDC 协议进行认证。
  • Webhook Token 认证:通过外部 Webhook 服务进行认证。
  • Authentication Proxy 认证:通过认证代理进程序行认证。

HTTPS 证书认证

kubernetes 需要 PKI(public key infrastructure,公钥基础设施)证书来基于 TLS 的安全的认证。 如果你使用 kubeadm 来初始化的集群,则 kubeadm 会帮助你自动生成集群所需要的各类证书。 kubeadm 会将证书放置在 /etc/kubernetes/pki 目录下,而管理员(用户账户)的证书会放置到 /etc/kubernetes 目录下。

k8s 使用 x509 证书中 CN(Common Name) 以及 O(Organization) 字段对应 k8s 中的 user 和 group,将 Authentication 和 RBAC Authorization 结合到了一起,巧妙地将 Control Plane 中的各个核心 User 和 Group、与操作权限(ClusterRole)进行了绑定(ClusterRoleBinding)。

其实每一个 kubectl 命令背后都有一个 kubeconf 文件在支持:

kubectl get pods --kubeconfig=file1
kubectl get pods --kubeconfig=file2
复制代码
kubectl get nodes \
    --server https://localhost:6443 \
    --user docker-for-desktop \
    --client-certificate my.cert \
    --client-key my.key \
    --insecure-skip-tls-verify
复制代码

3. Api-server 的授权管理

当用户通过认证(Authentication)后,下一步就是授权(Authorization)了。api-server 目前支持以下授权策略:

  • AlwaysDeny:拒绝所有,用于测试。目前这个已经被废弃
  • AlwaysAllow:允许所有,如果集群不需要授权流程,就可以采用该套策略。这个基本不会出现在生产环境。
  • ABAC(Attribute Based Access Control)基于属性的访问控制,它是k8s 1.6之前的默认策略,现在已经被RBAC代替。
  • RBAC(Role Based Access Control)基于角色的访问控制,它是目前(截止到k8s 1.23)默认的授权策略。如果RBAC仍然不满足某些特定需求,则用户还可以自行编写授权逻辑并通过Webhook方式注册为Kubernetes的授权服务,以实现更加复杂的授权规则。
  • Webhook:通过调用外部的REST服务对用户进行授权。
  • Node:是对kubelet进行授权的一种特殊模式。

通过 APIServer 的启动参数--authorization-mode可配置多种授权策略,用逗号分隔即可。在通常情况下,我们会设置授权策略为 Node,RBAC,APIServer 在收到请求后,会读取该请求中的数据,生成一个访问策略对象,APIServer 会将这个访问策略对象和配置的授权模式逐条进行匹配,第一个被满足或拒绝的授权策略决定了该请求的授权结果,如果匹配的结果是禁止访问,则 APIServer 会终止 API 调用流程,并返回客户端的错误调用码。

Node 授权策略

Node 授权策略用于对 kubelet 发出的请求进行访问控制,与用户的应用授权无关,属于 Kubernetes 自身安全的增强功能。简单来说,就是限制每个 Node 只访问它自身运行的 Pod 及相关的 Service、Endpoints 等信息;也只能受限于修改自身 Node 的一些信息,比如 Label;也不能操作其他 Node 上的资源。而之前用 RBAC 这种通用权限模型其实并不能满足 Node 这种特殊的安全要求,所以将其剥离出来定义为新的 Node 授权策略。

RBAC 授权策略

因为 AlwaysDeny、AlwaysAllow 这两个基本不会出现在生产环境,而 ABAC 已经被 RBAC 所替代,所以,我们直接从 RBAC 开始。

RBAC 在 k8s 1.8 版本时升级为 GA 稳定版本,并作为 kubeadm 安装方式下的默认选项。
RBAC具有如下优势:

  1. 对集群中的资源和非资源权限均有完整的覆盖。
  2. RBAC 的权限配置通过几个 API 对象即可完成,同其他 API 对象一样,可以用 kubectl 或 API 进行操作。
  3. 可以在运行时进行调整,无须重新启动 api-server。

在 RBAC 管理体系中,Kubernetes 引入了4个资源对象:Role、ClusterRole、RoleBinding 和ClusterRoleBinding。同其他 API 资源对象一样,用户可以使用 kubectl 或者 API 调用等方式操作这些资源对象。

角色(Role)和集群角色(ClusterRole)

一个角色就是一组权限的集合,在 Role 中设置的权限都是许可(Permissive)形式的,不可以设置拒绝(Deny)形式的规则。Role 设置的权限将会局限于命名空间(namespace)范围内,如果需要在集群级别设置权限,就需要使用 ClusterRole 了。

Role示例

apiVersion: rbac.authorization.k8s.io/v1
kind: Role
metadata:
  name: pod-reader
  namespace: default
  labels:
    role: pod-reader
  annotation:
rules:
- apiGroups: [""]
  resources: ["pods"]
  verbs: ["get","list","watch"]
复制代码

上述定义的 Role 的 name 是 pod-reader,这个 Role 的权限是可以对所有 apigruop 下面的资源名为“pod” 的资源进行 get、list 以及 watch 的操作。但是需要注意的是限定在 default 的 namespace 范围内。

Role 资源对象主要通过 rules 字段来描述它的功能,rules 字段是一个 rule 的 list,每一个 rule 包含如下几个关键字段:

  • apiGroups:api组,比如当我们使用 kubectl api-resources 来查询集群所支持的 api 资源时,会发现比如“apps/v1”这样的 vesion,它的结构是 apiVersion: GROUP_NAME/VERSION,所以,这里的 api 组就是 apps。
  • resouces:和 Role 绑定的资源名称。
  • verbs:和 Role 绑定的动作,比如 get、list 等。

ClusterRole示例

apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRole
metadata:
  # ClusterRole不受命名空间限制,所以不必绑定namespaces
  name: secret-reader
rules:
- apiGroups: [""]
  resources: ["secrets"]
  verbs: ["get","watch","list"]
复制代码

从命名上来看,ClusterRole 的处理范围要比 Role 大,因为 Role 的范围是 namespace,而 ClusterRole的范围是 Cluster。

ClusterRole 主要适用以下场景:

  • 对集群范围内资源的授权,例如 Node。
  • 对非资源型的授权,例如 /healthz。
  • 对包含全部 namespace 资源的授权,例如 pods(用于kubectl get pods --all-namespaces 这样的操作授权)。
  • 对某个命名空间中多种权限的一次性授权。

上述 ClusterRole 定义的是一个 name 为 secret-reader,拥有对所有 apiGroup 下的资源类型为 secrets 的资源进行 get、watch 和 list 的操作。并且他没有限定 namespace。

角色绑定(RoleBinding)和集群角色绑定(ClusterRoleBinding)

RoleBinding示例

RoleBinding 可以与属于相同命名空间的 Role 或者某个集群级别的 ClusterRole 绑定,完成对某个主体的授权。

这里的主体包括用户(User)、组(Group)以及 Service Account

apiVersion: rbac.authorization.k8s.io/v1
kind: RoleBinding
metadata:
  name: read-pods
  namespace: default
  labels:
    roleBinding: read-pods
  annotation:
subjects:
- kind: User
  name: Jackey
  apiGroup: rbac.authorization.k8s.io
roleRef:
  kind: Role
  name: pod-reader
  apiGroup: rbac.authorization.k8s.io
复制代码

RoleBinding 有两个比较重要的根节点,一个是 subjects,描述了需要绑定的主体,有 user、group 和 service account;另一个是 roleRef,描述了要绑定的 Role。

授权目标主体(Subject)的命名规范

在 RBAC 系统中,通过角色绑定(RoleBinding或ClusterRoleBinding)的定义,将在角色(Role或ClusterRole)中设置的授权规则与某个目标主体(Subject)绑定。授权的目标主体可以是用户(User)、用户组(Group)和Service Account三者之一。

  • 用户名由字符串进行标识,例如人名(alice)、Email地址(bob@example.com)、用户ID(1001)等,通常应该在客户端CA证书中进行设置。
  • 需要注意的是,Kubernetes内置了一组系统级别的用户/用户组,以“system:”开头,用户自定义的名称不应该使用这个前缀。
  • 用户组与用户名类似,由字符串进行标识,通常也应该在客户端CA证书中进行设置,并且要求不以“system:”为前缀。
  • Service Account在Kubernetes系统中的用户名会被设置成以“system:serviceaccount:”为前缀的名称,其所属的组名会被设置成以“system:serviceaccounts:”为前缀的名称。

RoleBinding 本身会被 namespace 所影响,用于某个 namespace 内的授权,如果它适合 Role 进行绑定,就需要保持一致的 namespace;而 RoleBinding 除了能够和 Role 绑定,也能和 ClusterRole 绑定,这个操作的含义是:对目标主体在其所在的命名空间授予在 ClusterRole 中定义的权限。

如下这个例子展示了这个情况:

apiVersion: rbac.authorization.k8s.io/v1
kind: RoleBinding
metadata:
  name: rolebinding-with-clusterole
  namespace: development
subjects:
- kind: User
  name: Jackey
  apiGroup: rbac.authorization.k8s.io
roleRef:
  kind: ClusterRole
  name: secret-reader
  apiGroup: rbac.authorization.k8s.io
复制代码

secret-reader 这个 ClusterRole 的定义在前面已给出。虽然 secret-reader 是一个集群角色,但因为 RoleBinding 的作用范围为命名空间 development,所以用户 Jackey 只能读取命名空间 development 中的 secret 资源对象,而不能读取其他命名空间中的 secret 资源对象。

ClusterRoleBinding绑定示例

ClusterRoleBinding 用于进行集群级别或者对所有命名空间都生效的授权。下面的例子允许 manager 组的用户读取任意命名空间中的 secret 资源对象:

apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRoleBinding
metadata:
  name: read-secret-global
subjects:
- kind: Group
  name: manager
  apiGroup: rbac.authorization.k8s.io
roleRef:
  kind: ClusterRole
  name: secret-reader
  apiGroup: rbac.authorization.k8s.io
复制代码

在集群角色绑定(ClusterRoleBinding)中引用的角色只能是集群级别的角色(ClusterRole),而不能是命名空间级别的Role。

一旦通过创建 RoleBinding 或 ClusterRoleBinding 与某个 Role 或 ClusterRole 完成了绑定,用户就无法修改与之绑定的 Role 或 ClusterRole 了。只有删除了 RoleBinding 或 ClusterRoleBinding,才能修改 Role 或 ClusterRole。Kubernetes 限制 roleRef 字段中的内容不可更改,主要有以下两个原因:

  1. 从逻辑上来说,与一个新的 Role 进行绑定实际上是一次全新的授权操作。通过删除或重建的方式更改绑定的 Role,可以确保给主体授予新角色的权限(而不是在不验证所有现有主体的情况下去修改 roleRef)。
  2. 使 roleRef 不变,可以授予某个用户对现有绑定对象(Bindingobject)的更新(update)权限,以便其管理授权主体(subject),同时禁止更改角色中的权限设置。

RBAC 对资源的引用方式

在 RBAC 中引用资源的方式就是资源对象的名称,如 pods、services、deloys 等等。但是有一些资源是具有字资源的,例如 pod 的日志。对于这种资源,RBAC 的引用方式是“资源/子资源”。例如 pods/log。

此外,默认情况下 RBAC 引用的资源能够包含所有该资源的实例,如果只是想针对某个资源的某个特定的实例,可以通过 resourcesName 来指定:

apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRole
metadata:
  # ClusterRole不受命名空间限制,所以不必绑定namespaces
  name: secret-reader
rules:
- apiGroups: [""]
  resources: ["secrets"]
  resourceName: "my-secrets"
  verbs: ["update","get"]
复制代码

resourceName 有一个限制,它对 list、watch、create 或者 deletecollection 这些操作是无效的。这是因为必须要通过 URL 进行鉴权,而资源名称在 list、watch、create 或 deletecollection 请求中只是请求 Body 数据的一部分。

ClusterRole 的聚合

某些情况下需要多个 ClusterRole 合并适用,这种情况下适用聚合 ClusterRole 能够有效的减轻管理员的操作。

这个功能是通过 aggregationRule 这个字段来完成的。aggregationRole 使用 LABEL SElECTOR 选择多个 ClusterRole,由相关的控制器(Controller)来保证能够将多个适合的 ClusterRole 进行整合。

apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRole
metadata:
  name: clusterrole-with-aggregation
aggregationRule:
  clusterRoleSelectors:
  - matchLabels:
      rbac.example.com/aggregate-to-monitoring: "true"
rules: [] #系统自动填充合并的结果
---
apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRole
metadata:
  name: clusterole-demo
  labels: 
    rbac.example.com/aggregate-to-monitoring: "true"
rules:
- apiGroups: [""]
  resources: ["services","endpoints","pods"]
  verbs: ["get","list","watch"]
复制代码

clusterrole-with-aggregation 会由系统进行控制,形成一个 ClusterRole 的聚合,我们可以通过 kubectl describe clusterrole/clusterrole-with-aggregation 来进行验证。

预防权限提升和授权初始化

RBACAPI 防止用户通过编辑 Role 或者 RoleBinding 获得权限的提升。这一限制是在 API 级别生效的,因此即使没有启用 RBAC,也仍然有效。

(一) 创建或更新 Role 或 ClusterRole 的限制用户要对角色(Role 或 ClusterRole)进行创建或更新操作,需要满足下列至少一个条件:

  1. 用户已拥有 Role 中包含的所有权限,且与该角色的生效范围一致(如果是集群角色,则是集群范围;如果是普通角色,则可能是同一个命名空间或者整个集群)。
  2. 用户被显式授予针对 Role 或 ClusterRole 资源的提权(Escalate)操作权限。

例如,用户 user-1 没有列出集群中所有 Secret 资源的权限,就不能创建具有这一权限的集群角色。要让一个用户能够创建或更新角色,则需要:

  1. 为其授予一个允许创建或更新 Role 或 ClusterRole 资源对象的角色。
  2. 为其授予允许创建或更新角色的权限,有隐式和显式两种方法。
  • 隐式:为用户授予这些权限。用户如果尝试使用尚未被授予的权限来创建或修改 Role 或 ClusterRole,则该 API 请求将被禁止。
  • 显式:为用户显式授予 rbac.authorization.k8s.ioAPIGroup 中的 Role 或 ClusterRole 的提权(Escalate)操作权限。

(二) 创建或更新 RoleBinding 或 ClusterRoleBinding 的限制仅当我们已经拥有被引用的角色( Role 或 ClusterRole)中包含的所有权限(与角色绑定的作用域相同)或已被授权对被引用的角色执行绑定(bind)操作时,才能创建或更新角色绑定(RoleBinding 或 ClusterRoleBinding)。

例如,如果用户 user-1 没有列出集群中所有 Secret 资源的权限,就无法为一个具有这样权限的角色创建 ClusterRoleBinding。要使用户能够创建或更新角色绑定,则需要进行以下操作。

  1. 为其授予一个允许创建和更新 RoleBinding 或 ClusterRoleBinding 的角色。
  2. 为其授予绑定特定角色的权限,有隐式或显式两种方法。
  • 隐式:授予其该角色中的所有权限。
  • 显式:授予在特定角色或集群角色中执行绑定(bind)操作的权限。

例如,通过下面的 ClusterRole 和 RoleBinding 设置,将允许用户 user-1 为其他用户在 user-1-namespace 命名空间中授予 admin、edit 及 view 角色的权限:

apiVersion: rbac.authorization.k8s.io/v1
Kind: ClusterRole
metadata:
  name: role-grantor
rules:
- apiGroups: ["rbac.authorization.k8s.io"]
  resources: ["rolebindings"]
  verbs: ["create"]
- apiGroups: ["rbac.authorization.k8s.io"]
  resources: ["clusterroles"]
  verbs: ["bind"]
  resourceNames: ["edit","admin","view"]
  ---
  apiVersion: rbac.authorization.k8s.io
  kind: RoleBinding
  metadata:
    name: role-grantor-binding
    namespace: user-1-namespace
  subjects:
  - kind: User
    name: user-1
    apiGroup: rbac.authorization.k8s.io
  roleRef:
    kind: ClusterRole
    name: role-grantor
    apiGroup: rbac.authorization.k8s.io
复制代码

在系统初始化过程中启用第1个角色和角色绑定时,必须让初始用户具备其尚未被授予的权限。要进行初始的角色和角色绑定设置,有以下两种办法。

  1. 使用属于 system:masters 组的凭据,这个组默认具有 cluster-admin 这个超级用户的权限。
  2. 如果 APIServer 以 --insecure-port 参数运行,则客户端通过这个非安全端口进行接口调用,通过这个非安全端口的访问没有认证鉴权的限制。

4. Admission Control

突破了之前所说的认证和鉴权两道关卡之后,客户端的调用请求就能够得到 APIServer 的真正响应了吗?答案是:不能!这个请求还要通过 AdmissionControl(准入控制)所控制的一个准入控制链的层层考验,才能获得成功的响应。

AdmissionControl 配备了一个准入控制器的插件列表,发送给 APIServer 的任何请求都需要通过列表中每个准入控制器的检查,检查不通过,APIServer 就会拒绝此调用请求。此外,准入控制器插件能够修改请求参数以完成一些自动化任务,比如 Service Account 这个控制器插件。

Service Account

ServiceAccount 也是一种账号,但它并不是给 Kubernetes 集群的用户(系统管理员、运维人员、租户用户等)用的,而是给运行在 Pod 里的进程用的,它为 Pod 里的进程提供了必要的身份证明。

在正常情况下,为了确保 Kubernetes 集群的安全,api-server 都会对客户端进行身份认证,认证失败的客户端无法进行 API 调用。此外,在 Pod 中访问 Kubernetes api-server 服务时,是以 Service 方式访问名为 kubernetes 的这个服务的,而 kubernetes 服务又只在 HTTPS 安全端口 443 上提供。

认证原理是在用一种类似 HTTP Token 的新认证方式 --Service Account Auth,Pod 中的客户端调用 Kubernetes API 时,在 HTTP Header 中传递了一个 Token 字符串,这类似于之前提到的 HTTP Token 认证方式,但有以下几个不同之处:

  • 这个 Token 的内容来自 Pod 里指定路径下的一个文件(/run/secrets/kubernetes.io/serviceaccount/token),该Token是动态生成的,确切地说,是由Kubernetes Controller 进程用 api-server 的私钥(--service-account-private-key-file 指定的私钥)签名生成的一个 JWT Token。
  • 在官方提供的客户端 REST 框架代码里,通过 HTTPS 方式与 api-server 建立连接后,会用 Pod 里指定路径下的一个 CA 证书(/run/secrets/kubernetes.io/serviceaccount/ca.crt)验证 api-server 发来的证书,验证是否为 CA 证书签名的合法证书。
  • api-server 在收到这个 Token 以后,会采用自己的私钥(实际上是使用 service-account-key-file 参数指定的私钥,如果没有设置此参数,则默认采用 tls-private-key-file 指定的参数,即自己的私钥)对 Token 进行合法性验证。

Service Account 与 Secret

每个命名空间中都有一个名为 default 的默认 Service Account对象,在这个 Service Account 里面有一个名为 Tokens 的可以作为 Volume 被挂载到 Pod 里的 Secret,Pod 启动时,这个 Secret 会自动被挂载到 Pod 的指定目录下,用来协助完成 Pod 中的进程访问 api-server 时的身份鉴权。

一个 Service Account 可以包含多个 secret。其中,名为 Tokens 的 Secret 用于访问 APIServer 的 Secret,也被称为 ServiceAccountSecret;名为 imagePullSecrets 的 Secret 用于下载容器镜像时的认证,镜像库通常运行在 Insecure 模式下,所以这个 Secret 为空;用户自定义的其他 Secret 用于用户的进程中。

Service Account 的正常工作离不开以下控制器:Service Account Controller、Token Controller、Admission Controller。

Service Account Controller

Service Account Controller 的工作相对简单,它会监听 Service Account 和 Namespace 这两种资源对象的事件,如果在一个 Namespace 中没有默认的 Service Account,那么它会为该 Namespace 创建一个默认的 ServiceAccount 对象,这就是在每个 Namespace 下都有一个名为 default 的 Service Account 的原因。

Token Controller

Token Controller 也监听 Service Account 的事件,如果发现在新建的 Service Account 里没有对应的 Service Account Secret,则会用 APIServer 私钥(--service-account-private-key-file 指定的文件)创建一个 Token,并用该 Token、api-server 的 CA 证书等三个信息产生一个新的 Secret 对象,然后放入刚才的 Service Account 中。如果监听到的事件是 Service Account 删除事件,则自动删除与该 Service Account 相关的所有 Secret。此外,Token Controller 对象也会同时监听 Secret 的创建和删除事件,确保与对应的 Service Account 的关联关系正确。

Admission Controller

接下来就是 Admission Controller 的重要作用了,当我们在 api-server 的准入控制链中启用了 Service Account 类型的准入控制器时(这也是默认的设置),则针对 Pod 新增或修改的请求,Admission Controller 会验证 Pod 里的 Service Account 是否合法,并做出如下控制操作:

  1. 如果 spec.serviceAccount 域没有被设置,则 Kubernetes 默认为其指定名称为 default 的Serviceaccout。
  2. 如果 Pod 的 spec.serviceAccount 域指定了不存在的 Service Account,则该 Pod 操作会被拒绝。
  3. 如果在 Pod 中没有指定 ImagePullSecrets,那么这个 spec.serviceAccount 域指定的 ServiceAccount 的 ImagePullSecrets 会被加入该 Pod 中。
  4. 给 Pod 添加一个特殊的 volumeSource,在该 Volume 中包含 ServiceAccountSecret 中的 Token。
  5. 给 Pod 里的每个容器都增加对应的 VolumeSource,将包含 Secret 的 Volume 挂载到 Pod 中所有容器的指定目录下(/var/run/secrets/kubernetes.io/serviceaccount)。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/74418.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

计算机毕业设计springboot+vue基本微信小程序的大学生竞赛信息发布与报名小程序

项目介绍 大学生竞赛是提升大学生综合能力和专业素质的重要手段和途径,越来越受到广大高校师生的关注与重视。大学生学科竞赛活动不仅有利于提升大学生的专业素养,也有利于提升大学生的创新、实践能力、竞争意识与团队精神。 各类学科竞赛汇总、信息发布、信息收集、报名、备赛…

针对前端项目node版本问题导致依赖安装异常的处理办法

Mac如何切换版本 前端项目开发过程中,多人开发会遇到由于node版本不同造成的依赖不适配。 比如: node 16.xx 大多都会遇到依赖版本与node版本不适配导致安装报错等问题,并且你不管如何更新还是使用稳定版本的node.js都不起作用,此时你需要修…

看直播怎么录屏?这2个方法,一看就会!

​现在很多人在斗鱼、虎牙、斗鱼、腾讯课堂等平台上直播,有些人在视频聊天平台上,如微信上直播。我们如何保存这些直播视频?看直播怎么录屏?今天小编就分享2个方法,教你如何看直播的同时录屏。 看直播怎么录屏方法一&a…

Font字体属性

Font字体属性 源代码 font font属性用于定义字体系列、大小、粗细、和文字样式(如斜体) font-family font-family属性用于定义文本字体,多个字体用 ” , ” 号隔开,一般情况下,有空格隔开的多个单词组成的字体&a…

Eziriz .NET Reactor保护NET代码

Eziriz .NET Reactor保护NET代码 NET Reactor软件是一个简单而小巧的工具,但对保护NET代码非常强大。会的。编程数据可以通过使用本程序、编写的代码和程序来保护,并禁止复制和使用它们。 Eziriz.NET Reactor软件的功能和特点: -支持收集和模…

MODBUS-ASCII协议

MODBUS协议在RS485总线上有RTU和ASCII两种传输格式。 其中ASCII协议应用比较少,主要还是以RTU格式为主。 相比较于RTU的2进制传输,ASCII使用的是文本传输,整个数据包都是可打印字符。 下面是示例: :010300000001FB\r\n 帧头是冒…

GWAS:使用R,比较GLM和MLM对假阳性的控制差异(复刻Nature genetics 图)

目录 1.数据准备 2.代码 如果想知道横纵坐标设置的原理,移步这篇超级棒的文章! 我们来复刻如下这张2016年发表在Nature genetics上的一篇文章中比较GLM和MLM的QQ plot! 参考文献: Genetic variation in ZmVPP1 contributes t…

Python爬虫项目100例

前言 Python爬虫项目100例(一):入门级 CentOS环境安装 和谐图网站爬取 美空网数据爬取 美空网未登录图片爬取 27270图片爬取 蜂鸟网图片爬取之一 蜂鸟网图片爬取之二 蜂鸟网图片爬取之三 河北阳光理政投诉板块 Pyt图虫网多线程爬取…

【计算机视觉+Tensorflow】SORT目标跟踪算法的讲解(图文解释 超详细)

觉得有帮助请点赞关注收藏~~~ 一、目标跟踪简介 目标跟踪算法可以进行轨迹特征的分析和提取,以弥补目标检测的不足;有效地去除误检,提高检测精度,为进一步的行为分析提供基础。例如,在自动驾驶系统中,目标跟…

Kafka知识点

消费者组 kafka的消费者组里面包含一个或多个消费者实例,它们共享一个公共的 ID,这个 ID 被称为 Group ID。一个消费者组可以订阅多个主题,但是同一个消费者组里面的一个实例只能消费一个主题里面的一个分区。 consumer group A kafka consum…

web课程设计网页规划与设计(HTML+CSS+JavaScript仿悦世界游戏官网 6个页面)

🎉精彩专栏推荐👇🏻👇🏻👇🏻 ✍️ 作者简介: 一个热爱把逻辑思维转变为代码的技术博主 💂 作者主页: 【主页——🚀获取更多优质源码】 🎓 web前端期末大作业…

web前端 基于html实现花店购物网站源码(HTML+CSS+JavaScript) 企业网站制作

🎉精彩专栏推荐 💭文末获取联系 ✍️ 作者简介: 一个热爱把逻辑思维转变为代码的技术博主 💂 作者主页: 【主页——🚀获取更多优质源码】 🎓 web前端期末大作业: 【📚毕设项目精品实战案例 (10…

央国企信创全面加速,如何将数字化转型和信创深度融合?

11月30日,《央国企信创白皮书》正式对外发布,CIO 时代研究院认为:在各行各业深入开展全面数字化转型的背景下,央国企信创工作和数字化转型工作势必并轨同行,既是机遇,亦是挑战。 信创是突破“卡脖子”技术…

Linux权限的理解

主要分为两类讲解:1.对人操作 2.对角色和文件操作 目录 Linux权限管理 文件类型和访问权限(事物属性) 基本权限 文件访问者的分类 文件权限值的表示方法 文件访问权限的设置方法 目录权限 umask 粘滞位 1.对人: Linux下…

JVM 篇之 牛刀小试 (二)(PS:之前请教我的小伙子校招去了字节~)

闲聊 最近在学jvm,当然这是一个持续过程,知行合一对吧,学习之后立马去应用,才能验证你学到是不是有用的东西。大家看过我上一篇写的博客么,之前在csdn请教过我的一个小伙子,最近我看他已经入职字节跳动了&a…

C++ 类与对象(四)构造函数2、static成员、友元、内部类

目录 1. 再谈构造函数 1.1 构造函数体赋值 1.2 初始化列表 1.3 explicit关键字 2. static成员 2.1 概念 2.2 特性 3. 友元 3.1友元函数 3.2 友元类 4. 内部类 1. 再谈构造函数 1.1 构造函数体赋值 在创建对象时,编译器通过调用构造函数,给对…

Nacos配置中心

小结: Nacos:注册中心,解决服务的注册与发现 Ribbon:客户端的负载均衡器,服务集群的负载均衡 OpenFeign:声明式的HTTP客户端,服务远程调用 Nacos:配置中心,中心化管理配置…

窗帘轨道怎么安装?方法有哪些?-江南爱窗帘十大品牌

不论是卧室的装修还是客厅的装修,都离不开窗帘的装饰,窗帘定制也是很多业主的选择,作为定制的窗帘必须具备的器件—窗帘轨道。那么。窗帘轨道的安装方法是什么?窗帘轨道的高低如何选择?下面小编就给大家详细介绍下吧。 01-窗帘轨道安装高度…

SpringBoot实战:整合MyBatis搭建基本骨架

这篇开始,开始进行 SpringBoot 框架功能的具体实现,本篇是 SpringBoot 整合 MyBatis 搭建基本骨架; 项目源码实现后分支地址:https://toscode.gitee.com/li_ziheng/lizhengi-samples/tree/feature%2Fspring-boot-1.0.0/ &#xff…

CSS 实现鼠标hover 展示内容

前言 &#x1f44f;CSS 实现鼠标hover 展示内容&#xff0c;速速来Get吧~ &#x1f947;文末分享源代码。记得点赞关注收藏&#xff01; 1.实现效果 2.实现步骤 定义一个宽高为300px的父容器 <div class"box"></box>.box {position: relative;width: …