目录
1.前提条件
2.开始创建核心组件Pod的Webhook
2.1.什么是Webhook
2.2.在本地k8s集群安装cert-manager
2.3.创建一个空的文件夹
2.4. 生成工程框架
2.5. 生成核心组件Pod的API
2.6.生成Webhook
2.7.开始实现Webhook相关代码
2.7.1.修改相关配置
2.7.2.修改代码
2.7.3.按照最新配置更新yaml
2.7.4.集群运行测试
2.7.4.1.修改Makefile文件
2.7.4.2.将应用部署到k8s集群上
2.7.5.测试
2.7.5.1.准备测试Deployment的yaml配置
2.7.5.2.部署这个Pod demo
2.7.5.3.停止测试
参考文章
1.前提条件
可以按之前的文章配置,配置好之后,我们会准备好以下内容:
- 本地多节点集群
- kubectl 客户端命令工具
- Lens k8s dashboard 可视化客户端工具
- golang开发语言
- VScode Linux 版
- kubebuilder
2.开始创建核心组件Pod的Webhook
K8s的核心组件就那些几个:Deployment、Pod、Service、Ingress、ConfigMap、Secret、……
在Kubebuilder的官方文档中,也有提到webhook的内容,但是比较简单,而且有些参考文章是基于CRD做的Webhook示例,此处我将演示核心组件Pod的Webhook。
2.1.什么是Webhook
从我的角度看,Webhook就是一个回调动作,举个例子方便理解:例如,我们希望在pod创建的时候,在annotation中增加一个标签,如author:geoff之类的,就可以利用这个webhook实现,当创建调用链完成之后,利用“回调”在这个创建动作之后丝滑地加入“增加annotation的额外动作”。可以结合Js的回调、java的AOP切片来理解。我们看看其中的原理:
从图中可以看到,Webhook是在Mutating Admission或者Vaildating Admission阶段往Webhook发送一个“请求”并接受来自Webhook的“响应”。我们此时不妨大胆推测一下,实现这两点,要做些什么呢?我可以不负责任推测一下步骤:
- 在XXX Admission Controller配置一个请求,请求会往Webhook接受、处理并返回
- Webhook开发一个接收这个请求的Handler(类似Java的controller)
事实也正如我们瞎想的哪样,确实核心就是这两个动作。这里我们看到,一共有两种Admission Controller,有点点区别:
- Mutating Admission Controller # 可以修改资源
- Vaildating Admission Controller # 验证资源,可以通过或者不通过
2.2.在本地k8s集群安装cert-manager
这个算是提前准备了,需要注意的是,k8s集群的版本和cert-manage需要注意下,不然版本差别过多,可能会有些意想不到的问题
# 查看本地集群的版本
kind version
# 在集群上安装cert-maneger,其中v1.7.1就是cert-manager版本。
kubectl apply -f https://github.com/jetstack/cert-manager/releases/download/v1.7.1/cert-manager.yaml
# 验证对应的资源实例是否安装成功
kubectl get all -n cert-manager
# 看对应的Pod是否被成功创建,可能一开始pod的replicas=0,因为集群节点是image container,下载image可能比较慢
kubectl get pods -n cert-manager
事实上,有时候,get pods可能会发现无资源,这可能是local cluster有问题,建议先删除,后新建:
# 先删除本地的集群
kind delete cluster --name k8s-local-dev
# 再新建
## 创建集群配置文件,1个master node,1个worker node。
cat << EOF > kind-clusters-mutil-config.yaml
kind: Cluster
apiVersion: kind.x-k8s.io/v1alpha4
nodes:
- role: control-plane
- role: worker
EOF
## start to create cluster
kind create cluster --name k8s-local-dev --config ./kind-clusters-mutil-config.yaml
## copy kubeconfig to Lens kubecofig
## 不然Lens在Kind集群更新后,配置文件没更新,会打不开
cp ~/.kube/config /mnt/c/Users/${current_user}/.kube/config
2.3.创建一个空的文件夹
# 创建一个空的文件节夹
mkdir create-pod-webhook-demo
# 进入空文件夹
cd create-pod-webhook-demo
2.4. 生成工程框架
# 生成工程框架
kubebuilder init --domain geoff.wh.demo --repo k8s-operator/kubebuilder-webhook-demo
生成文件目录如下, 此处,我不赘述了,上一篇文章已经有比较清晰的介绍了。
➜ create-pod-webhook-demo tree
.
├── Dockerfile
├── Makefile
├── PROJECT
├── README.md
├── config
│ ├── default
│ │ ├── kustomization.yaml
│ │ ├── manager_auth_proxy_patch.yaml
│ │ └── manager_config_patch.yaml
│ ├── manager
│ │ ├── controller_manager_config.yaml
│ │ ├── kustomization.yaml
│ │ └── manager.yaml
│ ├── prometheus
│ │ ├── kustomization.yaml
│ │ └── monitor.yaml
│ └── rbac
│ ├── auth_proxy_client_clusterrole.yaml
│ ├── auth_proxy_role.yaml
│ ├── auth_proxy_role_binding.yaml
│ ├── auth_proxy_service.yaml
│ ├── kustomization.yaml
│ ├── leader_election_role.yaml
│ ├── leader_election_role_binding.yaml
│ ├── role_binding.yaml
│ └── service_account.yaml
├── go.mod
├── go.sum
├── hack
│ └── boilerplate.go.txt
└── main.go
6 directories, 25 files
2.5. 生成核心组件Pod的API
从这里之前,和创建Operator的步骤基本是一致的,从这里开始,要开始有点不一样了
# 定义API,此处要实现Pod的Webhook,因此直接按Pod的Api创建
kubebuilder create api --group core --version v1 --kind Pod
# 之后的option我们不创建Resource和Controller,因此都选择n
之后的option我们不创建Resource和Controller,因此都选择n
2.6.生成Webhook
# --Kind这里,可以指定现存的PAAS组件,也可以
kubebuilder create webhook --group core --version v1 --kind Pod --defaulting --webhook-version v1
生成webhook的代码结构如下:
➜ create-pod-webhook-demo tree
.
├── Dockerfile
├── Makefile
├── PROJECT
├── README.md
├── api
│ └── v1
│ ├── pod_webhook.go
│ └── webhook_suite_test.go
├── bin
│ └── controller-gen
├── config
│ ├── certmanager
│ │ ├── certificate.yaml
│ │ ├── kustomization.yaml
│ │ └── kustomizeconfig.yaml
│ ├── default
│ │ ├── kustomization.yaml
│ │ ├── manager_auth_proxy_patch.yaml
│ │ ├── manager_config_patch.yaml
│ │ ├── manager_webhook_patch.yaml
│ │ └── webhookcainjection_patch.yaml
│ ├── manager
│ │ ├── controller_manager_config.yaml
│ │ ├── kustomization.yaml
│ │ └── manager.yaml
│ ├── prometheus
│ │ ├── kustomization.yaml
│ │ └── monitor.yaml
│ ├── rbac
│ │ ├── auth_proxy_client_clusterrole.yaml
│ │ ├── auth_proxy_role.yaml
│ │ ├── auth_proxy_role_binding.yaml
│ │ ├── auth_proxy_service.yaml
│ │ ├── kustomization.yaml
│ │ ├── leader_election_role.yaml
│ │ ├── leader_election_role_binding.yaml
│ │ ├── role_binding.yaml
│ │ └── service_account.yaml
│ └── webhook
│ ├── kustomization.yaml
│ ├── kustomizeconfig.yaml
│ └── service.yaml
├── go.mod
├── go.sum
├── hack
│ └── boilerplate.go.txt
└── main.go
11 directories, 36 files
到这一步之后,我们直接转转入VScode开发
# 当前工程目录打开VScode
code .
2.7.开始实现Webhook相关代码
2.7.1.修改相关配置
修改config/default/kustomize.yaml:
- 注释-crd 相关内容,因为我们没有CRD,只是实现Pod webhook
- 打开webhook、cert-manager相关配置,如下图所示
- 但是要注意属性缩进对齐,不然会报错
修改config/rdbc/kustomize.yaml:
1.注释 -role,这个也是和CRD相关,因为我们没有,所以注释掉
修改config/default/webhookcainjection_patch.yaml:
1.我们为了不搞这么复杂,我们只关心MutatingAdmision,因此可以直接注释掉ValidateAdmission
2.7.2.修改代码
因为核心组件Pod的Webhook和一般的CRD的webhook不一样,此处生成的pod_webhook.go只有Default()这个function,因此,我们需要直接重写整个代码,最重要的是Handle()方法。
修改api/v1/pod_webhook.go文件:
/*
Copyright 2023.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package v1
import (
"context"
"encoding/json"
"fmt"
"net/http"
corev1 "k8s.io/api/core/v1"
"sigs.k8s.io/controller-runtime/pkg/client"
logf "sigs.k8s.io/controller-runtime/pkg/log"
"sigs.k8s.io/controller-runtime/pkg/webhook/admission"
)
// log is for logging in this package.
var podlog = logf.Log.WithName("pod-resource")
// 定义核心组件pod的webhook的主struct,类似于java的Class
type PodWebhookMutate struct {
Client client.Client
decoder *admission.Decoder
}
// +kubebuilder:webhook:path=/mutate-core-v1-pod,mutating=true,failurePolicy=fail,sideEffects=None,groups=core,resources=pods,verbs=create;update,versions=v1,name=mpod.kb.io,admissionReviewVersions=v1
func (a *PodWebhookMutate) Handle(ctx context.Context, req admission.Request) admission.Response {
pod := &corev1.Pod{}
err := a.decoder.Decode(req, pod)
if err != nil {
return admission.Errored(http.StatusBadRequest, err)
}
// TODO: 变量marshaledPod是一个Map,可以直接修改pod的一些属性
marshaledPod, err := json.Marshal(pod)
if err != nil {
return admission.Errored(http.StatusInternalServerError, err)
}
// 打印
fmt.Println("======================================================")
fmt.Println(string(marshaledPod))
return admission.PatchResponseFromRaw(req.Object.Raw, marshaledPod)
}
func (a *PodWebhookMutate) InjectDecoder(d *admission.Decoder) error {
a.decoder = d
return nil
}
// 注释掉一开始生成的内容
// func (r *Pod) SetupWebhookWithManager(mgr ctrl.Manager) error {
// return ctrl.NewWebhookManagedBy(mgr).
// For(r).
// Complete()
// }
// // TODO(user): EDIT THIS FILE! THIS IS SCAFFOLDING FOR YOU TO OWN!
// var _ webhook.Defaulter = &Pod{}
// // Default implements webhook.Defaulter so a webhook will be registered for the type
// func (r *Pod) Default() {
// podlog.Info("default", "name", r.Name)
// // TODO(user): fill in your defaulting logic.
// }
修改api/v1/xxx_suite_test.go
全部注释掉,替换一个简单的测试Function。这个测试也是有讲究的,有兴趣可以研究一下。
/*
Copyright 2023.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package v1
import (
"fmt"
"testing"
)
func TestFunc(t *testing.T) {
// 打印
fmt.Println("this is test function......")
}
修改main.go文件:
import (
......
# 增加一个依赖包
v1 "k8s-operator/kubebuilder-webhook-demo/api/v1"
......
)
......
// 注释掉
// if err = (&corev1.Pod{}).SetupWebhookWithManager(mgr); err != nil {
// setupLog.Error(err, "unable to create webhook", "webhook", "Pod")
// os.Exit(1)
// }
// 增加webhook注册
mgr.GetWebhookServer().Register("/mutate-core-v1-pod", &webhook.Admission{Handler: &v1.PodWebhookMutate{Client: mgr.GetClient()}})
......
修改Dockerfile
1.和上面一样,因为我们是创建Pod的Webhook,不需要自定义Controller,注释掉。
修改Makefile文件
增加一个kind-load,方便将image上传到本地集群中
......
# local k8s cluster name
KUBE_CLUSTER = k8s-local-dev
......
.PHONY: kind-load
kind-load: ## load the local image to the kind cluster
kind load docker-image ${IMG} --name ${KUBE_CLUSTER}
......
2.7.3.按照最新配置更新yaml
# 更新yaml,这里和之前main.go和xxx_webhook.go的注释有点关,生成的代码是依据注释实现的
make mainfests generate
这一步比较重要,执行后会生成config/webhook/mainfests.yaml文件,切记执行
2.7.4.集群运行测试
2.7.4.1.修改Makefile文件
# 增加一个kind-load,方便将本地image上传到本地k8s集群中
......
# local k8s cluster name
KUBE_CLUSTER = k8s-local-dev
......
.PHONY: kind-load
kind-load: ## load the local image to the kind cluster
kind load docker-image ${IMG} --name ${KUBE_CLUSTER}
......
2.7.4.2.将应用部署到k8s集群上
以下是部署到集群的步骤:构建镜像、上传到集群容器中、部署
其中IMG是一个可以自行设置镜像名的变量,此处为:k8s-podwebhook-demo:1.0。
按如下命令执行后,即可在k8s集群中看到部署的CRD controller应用。
# 构建镜像(有时候会失败,可能是网络问题,多试几遍),IMG需要指定,不然后面部署还是有问题
make docker-build IMG=k8s-podwebhook-demo:1.0
# local镜像上传到Kind创建的k8s集群所在的所有node中(如果本地是)
make kind-load IMG=k8s-podwebhook-demo:1.0
# 部署controller
make deploy IMG=k8s-podwebhook-demo:1.0
在这个过程中,bin/目录存在一些二进制的工具包,可以先删除,是之前make deploy时下载的,如果已存在,有可能会报错。
这个MutatingAdmission的作用是注册一个endpoint,使得Pod在被创建后,往这个endpoint发送一个https请求(这也是为什么需要证书的原因)。事实上main.go中
// 增加webhook注册
mgr.GetWebhookServer().Register("/mutate-core-v1-pod", &webhook.Admission{Handler: &v1.PodWebhookMutate{Client: mgr.GetClient()}})
这句就类似java controller,接受这个https的请求。
2.7.5.测试
2.7.5.1.准备测试Deployment的yaml配置
这里需要说明,因为我们这里创建的是Pod Webhook,但是一般而言,我们不直接创建Pod,而是创建一个deployment,因为deployment最后也是会创建pod的。当然直接创建Pod也是可以的,这里创建一个pod-demo-ngnix.yaml文件去创建一个简单的pod,这个Pod只是一个基本的Ngnix镜像。
cat <<EOF >pod-demo-ngnix.yaml
apiVersion: v1
kind: Pod
metadata:
name: pod-demo-ngnix
labels:
role: myrole
spec:
containers:
- name: web
image: nginx
ports:
- name: web
containerPort: 80
protocol: TCP
EOF
2.7.5.2.部署这个Pod demo
# 部署这个pod demo实例
kubectl apply -f ./pod-demo-ngnix.yaml
2.7.5.3.停止测试
# 删除测试Pod
kubectl delete -f ./pod-demo-ngnix.yaml
# undeploy这个webhook
make undeploy
参考文章
8. kubebuilder 进阶: webhook - Mohuishou
使用kubebuilder开发kubernetes核心资源的webhook | 老 宋
调试运行中的 Pod | Kubernetes