-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmetacontroller-setup-kubenete-custom-controlle-with-python-business-logic.txt
199 lines (154 loc) · 5.97 KB
/
metacontroller-setup-kubenete-custom-controlle-with-python-business-logic.txt
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
#ref https://metacontroller.github.io/metacontroller/guide/create.html
1) Meta controller creation using kustomization
kubectl apply -k https://github.com/metacontroller/metacontroller/manifests/production
2)name space for crd
kubectl create namespace hello
3)custom resource defination - Here we specified group name code-desk-ubuntu2204-4-lts.test for resource api version, apiextensions.k8s.io/v1 required when you define custom resource defination. Our custom resource is kind HelloWorld
#crd.yaml
apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
name: helloworlds.code-desk-ubuntu2204-4-lts.test
spec:
group: code-desk-ubuntu2204-4-lts.test
names:
kind: HelloWorld
plural: helloworlds
singular: helloworld
scope: Namespaced
versions:
- name: v1
served: true
storage: true
schema:
openAPIV3Schema:
type: object
properties:
spec:
type: object
properties:
who:
type: string
subresources:
status: {}
#kubectl apply -f crd.yaml
4)Define custom controller = hello-controller with api version code-desk-ubuntu2204-4-lts.test/v1. Here we created child resource as pods (busybox echo hello (who) run). we are using webhook to forward traffic to pod having sync.py(hello-controller deployment/service). This below creates our custom controller to respond our custom resource kind helloworld (here hello-controller is service name and hello is name space and calling sync url: http://hello-controller.hello/sync ) - Here parentResource is kind helloworld (take pod name from it in sync.py and childResources is pods(busybox in sync.py)
apiVersion: metacontroller.k8s.io/v1alpha1
kind: CompositeController
metadata:
name: hello-controller
spec:
generateSelector: true
parentResource:
apiVersion: code-desk-ubuntu2204-4-lts.test/v1
resource: helloworlds
childResources:
- apiVersion: v1
resource: pods
updateStrategy:
method: Recreate
hooks:
sync:
webhook:
url: http://hello-controller.hello/sync
#kubectl apply -f controller.yaml
5)sync.py as configmap to mount inside webhook pod to run it. Sync is main business logic where you serve http request for custom resource defination creation like HelloWorld kind. Here in these code we are running as HTTP server which take desire state as pod busybox creation with command "echo hello with kind helloworld input (who)".Here pod name is name taken from kind HelloWorld. sync.py respond to Kubernetes API to create desired pod with busybox when triggered by HelloWorld kind
from http.server import BaseHTTPRequestHandler, HTTPServer
import json
class Controller(BaseHTTPRequestHandler):
def sync(self, parent, children):
# Compute status based on observed state.
desired_status = {
"pods": len(children["Pod.v1"])
}
# Generate the desired child object(s).
who = parent.get("spec", {}).get("who", "World")
desired_pods = [
{
"apiVersion": "v1",
"kind": "Pod",
"metadata": {
"name": parent["metadata"]["name"]
},
"spec": {
"restartPolicy": "OnFailure",
"containers": [
{
"name": "hello",
"image": "busybox",
"command": ["echo", "Hello, %s!" % who]
}
]
}
}
]
return {"status": desired_status, "children": desired_pods}
def do_POST(self):
# Serve the sync() function as a JSON webhook.
observed = json.loads(self.rfile.read(int(self.headers.get("content-length"))))
desired = self.sync(observed["parent"], observed["children"])
self.send_response(200)
self.send_header("Content-type", "application/json")
self.end_headers()
self.wfile.write(json.dumps(desired).encode())
HTTPServer(("", 80), Controller).serve_forever()
#configmap creation from sync.py in hello name space
#kubectl -n hello create configmap hello-controller --from-file=sync.py
6)webhook deployment and service to run pod with sync.py with output which is busybox echo hello with kind HelloWorld input name (hello-controller is pod name and service name in name space hello)
apiVersion: apps/v1
kind: Deployment
metadata:
name: hello-controller
spec:
replicas: 1
selector:
matchLabels:
app: hello-controller
template:
metadata:
labels:
app: hello-controller
spec:
containers:
- name: controller
image: python:3
command: ["python3", "/hooks/sync.py"]
volumeMounts:
- name: hooks
mountPath: /hooks
volumes:
- name: hooks
configMap:
name: hello-controller
---
apiVersion: v1
kind: Service
metadata:
name: hello-controller
spec:
selector:
app: hello-controller
ports:
- port: 80
#webhook creation
#kubectl -n hello apply -f webhook.yaml
7)kind type Helloworld creation with api version code-desk-ubuntu2204-4-lts.test/v1 #hello.yaml, name must be small case letter, here who is what you want to display in busybox echo hello pintu (who), name amit is kind resource name as pod
apiVersion: code-desk-ubuntu2204-4-lts.test/v1
kind: HelloWorld
metadata:
name: amit
spec:
who: pintu
#hello yaml creation
#kubectl -n hello apply -f hello.yaml
8)get pods
#kubectl -n hello get pods
NAME READY STATUS RESTARTS AGE
amit 0/1 Completed 0 28s
hello-controller-5454565fc8-sqxrx 1/1 Running 0 9m52s
Here hello-controller-5454565fc8-sqxrx is our webhook pod and amit is our kind:HelloWorld pod having busybox image and echo "Hello, pintu!" command (echo "Hello, (who)!")
9)check pod amit logs
kubectl -n hello logs amit
Hello, pintu!
Flow will be
Kubernetes API -> Custom Metacontroller hello-controller -> Webhook service http://hello-controller.hello/sync -> hello-controller service -> hello-controller pod -> Custom Metacontroller hello-controller-> Kubernetes API.