-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathk8s-discovery.py
executable file
·239 lines (210 loc) · 7.2 KB
/
k8s-discovery.py
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
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
#!/usr/bin/env python3
# vim: set ts=4 sw=4 et:
#
# k8s-discovery.py
"""
Talk to K8s APIs to discover environment
Save discovered information in classes that reflect G-X attributes
These can then be dumped as YAML or as Gaia-X JSON-LD Self-Description.
(c) Tecnalia: Raúl Miñón, Ana I. Torre, Gorka Benguria, Gorka Zarate, Juan López de Armentia <[email protected]>, 3/2023 - 4/2023
SPDX-License-Identifier: EPL-2.0
"""
import os
import sys
from kubernetes import client
from kubernetes import config as cfg
# Global variables
if "KUBECONFIG" in os.environ:
config = os.environ["KUBECONFIG"]
else:
config = ""
context = None
debug = False
outjson = False
ofile = "/dev/stdout"
indent = " "
class ContainerVolume:
def __init__(self, mount_path, v_name, read_only):
self.mount_path = mount_path
self.v_name = v_name
self.read_only = read_only
class ContainerPort:
def __init__(self, container_port, port_name, protocol, port_host_ip, host_port):
self.container_port = container_port
self.port_name = port_name
self.protocol = protocol
self.port_host_ip = port_host_ip
self.port_host_port = host_port
class KubeContainer:
def __init__(
self, env, c_image, command, image_pull_policy, c_name, ports, volumes
):
self.env = env
self.c_image = c_image
self.command = command
self.image_pull_policy = image_pull_policy
self.c_name = c_name
self.ports = ports
self.volumes = volumes
class KubeNodeCapacity:
def __init__(
self, cpu, ephemeral_storage, hugepages_1Gi, hugepages_2Mi, memory, pods
):
self.cpu = cpu
self.ephemeral_storage = ephemeral_storage
self.hugepages_1Gi = hugepages_1Gi
self.hugepages_2Mi = hugepages_2Mi
self.memory = memory
self.pods = pods
class KubeNode:
def __init__(self, hostname, internal_ip, capacity):
self.hostname = hostname
self.internal_ip = internal_ip
self.capacity = capacity
class KubePod:
def __init__(
self,
host_ip,
pod_i_ps,
phase,
start_time,
qos_class,
image,
image_id,
name,
namespace,
labels,
uid,
containers,
):
self.host_ip = host_ip
self.pod_i_ps = pod_i_ps
self.phase = phase
self.start_time = start_time
self.qos_class = qos_class
self.image = image
self.image_id = image_id
self.name = name
self.namespace = namespace
self.labels = labels
self.uid = uid
self.containers = containers
class KubeCluster:
"Abstraction for K8s with all its pods and nodes"
def __init__(self, conn):
self.conn = conn
nodes = conn.list_node()
api_version = nodes.api_version
metadata = nodes.metadata
self.metadata = metadata
self.api_version = api_version
self.kube_nodes = []
for node in nodes.items:
addresses = node.status.addresses
cap = node.status.capacity
capacity = KubeNodeCapacity(
cap["cpu"],
cap["ephemeral-storage"],
cap["hugepages-1Gi"],
cap["hugepages-2Mi"],
cap["memory"],
cap["pods"],
)
hostname = None
internal_ip = None
for address in addresses:
if address.type == "Hostname":
hostname = address.address
elif address.type == "InternalIP":
internal_ip = address.address
self.kube_nodes.append(KubeNode(hostname, internal_ip, capacity))
pods = conn.list_pod_for_all_namespaces(watch=False)
self.kube_pods = []
for pod in pods.items:
host_ip = pod.status.host_ip
pod_i_ps = pod.status.pod_i_ps
phase = pod.status.phase
start_time = pod.status.start_time
qos_class = pod.status.qos_class
image = pod.status.container_statuses[0].image
image_id = pod.status.container_statuses[0].image_id
name = pod.metadata.name
namespace = pod.metadata.namespace
labels = pod.metadata.labels
uid = pod.metadata.uid
containers = []
for container in pod.spec.containers:
env = container.env
c_image = container.image
command = container.command
image_pull_policy = container.image_pull_policy
c_name = container.name
ports = []
if container.ports:
for port in container.ports:
container_port = port.container_port
port_name = port.name
protocol = port.protocol
port_host_ip = port.host_ip
host_port = port.host_port
ports.append(
ContainerPort(
container_port,
port_name,
protocol,
port_host_ip,
host_port,
)
)
volumes = []
if container.volume_mounts:
for volume in container.volume_mounts:
mount_path = volume.mount_path
v_name = volume.name
read_only = volume.read_only
volumes.append(ContainerVolume(mount_path, v_name, read_only))
containers.append(
KubeContainer(
env, c_image, command, image_pull_policy, c_name, ports, volumes
)
)
self.kube_pods.append(
KubePod(
host_ip,
pod_i_ps,
phase,
start_time,
qos_class,
image,
image_id,
name,
namespace,
labels,
uid,
containers,
)
)
def values(self):
"dict representing stored data"
inner = {"kube_nodes": list(map(lambda x: x, self.kube_nodes))}
inner["kube_pods"] = {"kube_pods": list(map(lambda x: x, self.kube_pods))}
inner["api_version"] = self.api_version
inner["metadata"] = self.metadata
return inner
def kubeconn(config_file, context, timeout):
"Establish connection to K8s cluster"
cfg.load_kube_config(config_file=config_file, context=context)
return client.CoreV1Api()
def main(argv):
"Entry point for main program"
global cloud, outjson, indent
global debug, ofile
# timeout = 12
conn = kubeconn(None, None, None)
my_kube = KubeCluster(conn)
if ofile == "/dev/stdout":
print(my_kube, file=sys.stdout)
else:
print(my_kube, file=open(ofile, mode="a", encoding="UTF-8"))
if __name__ == "__main__":
main(sys.argv)