2021-12-09 07:51:44 +00:00
|
|
|
import os.path
|
2021-12-02 14:28:55 +00:00
|
|
|
import time
|
|
|
|
from kubernetes import client, config
|
|
|
|
from kubernetes.client.rest import ApiException
|
2022-05-12 10:49:53 +00:00
|
|
|
from common.milvus_sys import MilvusSys
|
2021-12-03 02:23:32 +00:00
|
|
|
from utils.util_log import test_log as log
|
2021-12-02 14:28:55 +00:00
|
|
|
|
|
|
|
|
2021-12-03 09:11:45 +00:00
|
|
|
def wait_pods_ready(namespace, label_selector, expected_num=None, timeout=360):
|
2021-12-03 05:35:59 +00:00
|
|
|
"""
|
|
|
|
wait pods with label selector all ready
|
|
|
|
|
|
|
|
:param namespace: the namespace where the release
|
|
|
|
:type namespace: str
|
|
|
|
|
|
|
|
:param label_selector: labels to restrict which pods are waiting to be ready
|
|
|
|
:type label_selector: str
|
|
|
|
|
2021-12-03 09:11:45 +00:00
|
|
|
:param expected_num: expected the minimum number of pods to be ready if not None
|
|
|
|
:type expected_num: int
|
|
|
|
|
2021-12-03 05:35:59 +00:00
|
|
|
:param timeout: limits the duration of the call
|
|
|
|
:type timeout: int
|
|
|
|
|
|
|
|
:example:
|
2021-12-03 09:11:45 +00:00
|
|
|
>>> wait_pods_ready("default", "app.kubernetes.io/instance=scale-query", expected_num=9)
|
|
|
|
"""
|
2021-12-02 14:28:55 +00:00
|
|
|
config.load_kube_config()
|
|
|
|
api_instance = client.CoreV1Api()
|
|
|
|
try:
|
|
|
|
all_pos_ready_flag = False
|
|
|
|
time_cnt = 0
|
|
|
|
while not all_pos_ready_flag and time_cnt < timeout:
|
|
|
|
api_response = api_instance.list_namespaced_pod(namespace=namespace, label_selector=label_selector)
|
|
|
|
all_pos_ready_flag = True
|
2021-12-03 09:11:45 +00:00
|
|
|
if expected_num is not None and len(api_response.items) < expected_num:
|
|
|
|
all_pos_ready_flag = False
|
|
|
|
else:
|
|
|
|
for item in api_response.items:
|
2021-12-29 12:58:37 +00:00
|
|
|
if item.status.phase != 'Running':
|
|
|
|
all_pos_ready_flag = False
|
2022-01-20 01:11:41 +00:00
|
|
|
break
|
2021-12-03 09:11:45 +00:00
|
|
|
for c in item.status.container_statuses:
|
|
|
|
log.info(f"{c.name} status is {c.ready}")
|
|
|
|
if c.ready is False:
|
|
|
|
all_pos_ready_flag = False
|
|
|
|
break
|
2021-12-02 14:28:55 +00:00
|
|
|
if not all_pos_ready_flag:
|
|
|
|
log.info("all pods are not ready, please wait")
|
|
|
|
time.sleep(30)
|
|
|
|
time_cnt += 30
|
|
|
|
if all_pos_ready_flag:
|
|
|
|
log.info(f"all pods in namespace {namespace} with label {label_selector} are ready")
|
|
|
|
else:
|
2021-12-03 05:35:59 +00:00
|
|
|
log.info(f"timeout for waiting all pods in namespace {namespace} with label {label_selector} ready")
|
2021-12-02 14:28:55 +00:00
|
|
|
except ApiException as e:
|
|
|
|
log.error("Exception when calling CoreV1Api->list_namespaced_pod: %s\n" % e)
|
|
|
|
raise Exception(str(e))
|
|
|
|
|
|
|
|
return all_pos_ready_flag
|
|
|
|
|
|
|
|
|
2021-12-08 09:55:14 +00:00
|
|
|
def get_pod_list(namespace, label_selector):
|
|
|
|
"""
|
|
|
|
get pod list with label selector
|
|
|
|
|
|
|
|
:param namespace: the namespace where the release
|
|
|
|
:type namespace: str
|
|
|
|
|
|
|
|
:param label_selector: labels to restrict which pods to list
|
|
|
|
:type label_selector: str
|
|
|
|
|
|
|
|
:example:
|
|
|
|
>>> get_pod_list("chaos-testing", "app.kubernetes.io/instance=test-proxy-pod-failure, component=proxy")
|
|
|
|
"""
|
|
|
|
config.load_kube_config()
|
|
|
|
api_instance = client.CoreV1Api()
|
|
|
|
try:
|
|
|
|
api_response = api_instance.list_namespaced_pod(namespace=namespace, label_selector=label_selector)
|
|
|
|
return api_response.items
|
|
|
|
except ApiException as e:
|
|
|
|
log.error("Exception when calling CoreV1Api->list_namespaced_pod: %s\n" % e)
|
|
|
|
raise Exception(str(e))
|
|
|
|
|
|
|
|
|
2022-04-13 02:39:34 +00:00
|
|
|
def get_pod_ip_name_pairs(namespace, label_selector):
|
|
|
|
"""
|
|
|
|
get pod ip name pairs with label selector
|
|
|
|
|
|
|
|
:param namespace: the namespace where the release
|
|
|
|
:type namespace: str
|
|
|
|
|
|
|
|
:param label_selector: labels to restrict which pods to list
|
|
|
|
:type label_selector: str
|
|
|
|
|
|
|
|
:example:
|
|
|
|
>>> get_pod_ip_name_pairs("chaos-testing", "app.kubernetes.io/instance=test-proxy-pod-failure, component=querynode")
|
|
|
|
"""
|
|
|
|
m = dict()
|
|
|
|
items = get_pod_list(namespace, label_selector)
|
|
|
|
for item in items:
|
|
|
|
ip = item.status.pod_ip
|
|
|
|
name = item.metadata.name
|
|
|
|
m[ip] = name
|
|
|
|
return m
|
|
|
|
|
|
|
|
|
2022-05-12 10:49:53 +00:00
|
|
|
def get_querynode_id_pod_pairs(namespace, label_selector):
|
|
|
|
"""
|
|
|
|
get milvus node id and corresponding pod name pairs with label selector
|
|
|
|
|
|
|
|
:param namespace: the namespace where the release
|
|
|
|
:type namespace: str
|
|
|
|
|
|
|
|
:param label_selector: labels to restrict which pods to list
|
|
|
|
:type label_selector: str
|
|
|
|
|
|
|
|
:example:
|
|
|
|
>>> querynode_id_pod_pair = get_querynode_id_pod_pairs("chaos-testing", "app.kubernetes.io/instance=milvus-multi-querynode, component=querynode")
|
|
|
|
{
|
|
|
|
5: 'milvus-multi-querynode-querynode-7b8f4b5c5-4pn42',
|
|
|
|
9: 'milvus-multi-querynode-querynode-7b8f4b5c5-99tx7',
|
|
|
|
1: 'milvus-multi-querynode-querynode-7b8f4b5c5-w9sk8',
|
|
|
|
3: 'milvus-multi-querynode-querynode-7b8f4b5c5-xx84j',
|
|
|
|
6: 'milvus-multi-querynode-querynode-7b8f4b5c5-x95dp'
|
|
|
|
}
|
|
|
|
"""
|
|
|
|
# TODO: extend this function to other worker nodes, not only querynode
|
|
|
|
querynode_ip_pod_pair = get_pod_ip_name_pairs(namespace, label_selector)
|
|
|
|
querynode_id_pod_pair = {}
|
|
|
|
ms = MilvusSys()
|
|
|
|
for node in ms.query_nodes:
|
|
|
|
ip = node["infos"]['hardware_infos']["ip"].split(":")[0]
|
|
|
|
querynode_id_pod_pair[node["identifier"]] = querynode_ip_pod_pair[ip]
|
|
|
|
return querynode_id_pod_pair
|
|
|
|
|
|
|
|
|
2022-01-18 01:21:36 +00:00
|
|
|
def export_pod_logs(namespace, label_selector, release_name=None):
|
2021-12-09 07:51:44 +00:00
|
|
|
"""
|
|
|
|
export pod logs with label selector to '/tmp/milvus'
|
|
|
|
|
|
|
|
:param namespace: the namespace where the release
|
|
|
|
:type namespace: str
|
|
|
|
|
|
|
|
:param label_selector: labels to restrict which pods logs to export
|
|
|
|
:type label_selector: str
|
|
|
|
|
2022-01-18 01:21:36 +00:00
|
|
|
:param release_name: use the release name as server logs director name
|
|
|
|
:type label_selector: str
|
|
|
|
|
2021-12-09 07:51:44 +00:00
|
|
|
:example:
|
2021-12-20 11:18:26 +00:00
|
|
|
>>> export_pod_logs("chaos-testing", "app.kubernetes.io/instance=mic-milvus")
|
2021-12-09 07:51:44 +00:00
|
|
|
"""
|
2022-01-18 01:21:36 +00:00
|
|
|
if isinstance(release_name, str):
|
|
|
|
if len(release_name.strip()) == 0:
|
|
|
|
raise ValueError("Got an unexpected space release_name")
|
|
|
|
else:
|
|
|
|
raise TypeError("Got an unexpected non-string release_name")
|
|
|
|
pod_log_path = '/tmp/milvus_logs' if release_name is None else f'/tmp/milvus_logs/{release_name}'
|
|
|
|
|
2021-12-09 07:51:44 +00:00
|
|
|
if not os.path.isdir(pod_log_path):
|
|
|
|
os.makedirs(pod_log_path)
|
|
|
|
|
|
|
|
# get pods and export logs
|
2021-12-20 11:18:26 +00:00
|
|
|
items = get_pod_list(namespace, label_selector=label_selector)
|
2021-12-09 07:51:44 +00:00
|
|
|
try:
|
|
|
|
for item in items:
|
|
|
|
pod_name = item.metadata.name
|
|
|
|
os.system(f'kubectl logs {pod_name} > {pod_log_path}/{pod_name}.log 2>&1')
|
|
|
|
except Exception as e:
|
|
|
|
log.error(f"Exception when export pod {pod_name} logs: %s\n" % e)
|
|
|
|
raise Exception(str(e))
|
|
|
|
|
|
|
|
|
2022-01-20 01:11:41 +00:00
|
|
|
def read_pod_log(namespace, label_selector, release_name):
|
|
|
|
config.load_kube_config()
|
|
|
|
items = get_pod_list(namespace, label_selector=label_selector)
|
|
|
|
|
|
|
|
try:
|
|
|
|
# export log to /tmp/release_name path
|
|
|
|
pod_log_path = f'/tmp/milvus_logs/{release_name}'
|
|
|
|
if not os.path.isdir(pod_log_path):
|
|
|
|
os.makedirs(pod_log_path)
|
|
|
|
|
|
|
|
api_instance = client.CoreV1Api()
|
|
|
|
|
|
|
|
for item in items:
|
|
|
|
pod = item.metadata.name
|
|
|
|
log.debug(f'Start to read {pod} log')
|
|
|
|
logs = api_instance.read_namespaced_pod_log(name=pod, namespace=namespace, async_req=True)
|
|
|
|
with open(f'{pod_log_path}/{pod}.log', "w") as f:
|
|
|
|
f.write(logs.get())
|
|
|
|
|
|
|
|
except ApiException as e:
|
|
|
|
log.error(f"Exception when read pod {pod} logs: %s\n" % e)
|
|
|
|
raise Exception(str(e))
|
|
|
|
|
|
|
|
|
2021-12-02 14:28:55 +00:00
|
|
|
if __name__ == '__main__':
|
2022-04-13 02:39:34 +00:00
|
|
|
label = "app.kubernetes.io/instance=milvus-load-balance, component=querynode"
|
2021-12-08 09:55:14 +00:00
|
|
|
res = get_pod_list("chaos-testing", label_selector=label)
|
2022-04-13 02:39:34 +00:00
|
|
|
m = get_pod_ip_name_pairs("chaos-testing", label_selector=label)
|
2021-12-20 11:18:26 +00:00
|
|
|
export_pod_logs(namespace='chaos-testing', label_selector=label)
|