generated from cern-sis/errbot-plugin-template
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmonitor.py
131 lines (114 loc) · 4.63 KB
/
monitor.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
import logging
import os
import kopf
import requests
from errbot import BotPlugin
from kubernetes import client, config
config.load_incluster_config()
v1 = client.AppsV1Api()
class KopfHandler:
def configure_kopf(self, settings: kopf.OperatorSettings, **_):
settings.posting.enabled = True
self.logger = logging.getLogger(__name__)
self.logger.info("test message")
def _prepare_message_for_deployment(
self, replicas_unavailable, deployment, **kwargs
):
BOT_API_KEY = os.environ["BOT_K8S_KEY"]
BOT_EMAIL = "[email protected]"
namespace = kwargs["body"]["metadata"]["namespace"]
if namespace not in os.environ.get("ACCEPTED_NAMESPACES", "").split(","):
return
deployment_name = deployment.metadata.name
zulip_msg_content = f"""
:double_exclamation: Detected deployment
**{deployment_name}** is failing.\n
\n{replicas_unavailable} replica(s) is/are unavailable
"""
zulip_request_payload = {
"type": "stream",
"to": namespace.split("-")[0],
"topic": namespace,
"content": zulip_msg_content,
}
headers = {
"Content-Type": "application/json",
"Authorization": f"Basic {BOT_EMAIL}:{BOT_API_KEY}",
}
response = requests.post(
"https://cern-rcs-sis.zulipchat.com/api/v1/messages",
json=zulip_request_payload,
headers=headers,
)
self.log.info(response.status_code)
def _prepare_message_for_container_restart(self, pod, container_name, **kwargs):
BOT_API_KEY = os.environ["BOT_K8S_KEY"]
BOT_EMAIL = "[email protected]"
namespace = kwargs["body"]["metadata"]["namespace"]
if namespace not in os.environ.get("ACCEPTED_NAMESPACES", "").split(","):
return
pod_name = pod.metadata.name
zulip_msg_content = f"""
:warning: Detected container restarted
**{container_name} has restarted in pod **{pod_name}**\n
"""
zulip_request_payload = {
"type": "stream",
"to": "test",
"topic": "test",
"content": zulip_msg_content,
}
headers = {
"Content-Type": "application/json",
"Authorization": f"Basic {BOT_EMAIL}:{BOT_API_KEY}",
}
response = requests.post(
"https://cern-rcs-sis.zulipchat.com/api/v1/messages",
json=zulip_request_payload,
headers=headers,
)
self.log.info(response.status_code)
# send notification if the container restarts
@kopf.on.event("v1", "pods")
def container_restart_notification_handler(self, body, **kwargs):
self.log.info("container restarted")
for container_status in body.status.container_statuses:
if container_status.state.running and container_status.restart_count > 0:
container = next(
(
c
for c in body.spec.containers
if c.name == container_status.name
),
None,
)
if container:
self._prepare_message_for_container_restart(
body, container, **kwargs
)
# send notification if the deployment is failing
@kopf.on.field("apps", "v1", "deployments", field="status.replicas")
def pod_phase_notification_handler(self, old, new, body, **kwargs):
if not old:
return
deployment = v1.read_namespaced_deployment(
name=body["metadata"]["name"], namespace=body["metadata"]["namespace"]
)
max_unavailable = deployment.spec.strategy.rolling_update.max_unavailable
replicas_unavailable = old - new
if replicas_unavailable > max_unavailable:
self.prepare_message_for_deployment(
replicas_unavailable, deployment, **kwargs
)
class Monitor(BotPlugin):
def activate(self):
super().activate()
self.log.info("activating plugin monitor")
instance = KopfHandler()
kopf.on.startup()(instance.configure_kopf)
self.log.info(instance)
# @kopf.on.startup()
# def configure_kopf(self, settings: kopf.OperatorSettings, **_):
# self.log.info("kopf startup")
# self.log.info(settings)
# settings.posting.enabled = False