matrix-alertbot/matrix_alertbot/command.py

219 lines
7.5 KiB
Python
Raw Normal View History

import logging
2022-07-10 14:09:00 +02:00
from typing import List
2022-07-06 00:54:13 +02:00
from diskcache import Cache
2022-07-10 14:06:36 +02:00
from nio import AsyncClient, MatrixRoom
2019-09-25 14:26:29 +02:00
from matrix_alertbot.alertmanager import AlertmanagerClient
2022-07-10 14:06:36 +02:00
from matrix_alertbot.chat_functions import send_text_to_room
2022-06-13 20:55:01 +02:00
from matrix_alertbot.config import Config
from matrix_alertbot.errors import (
AlertNotFoundError,
AlertmanagerError,
SilenceNotFoundError,
)
2022-07-10 12:51:49 +02:00
from matrix_alertbot.matcher import AlertMatcher, AlertRegexMatcher
logger = logging.getLogger(__name__)
2021-01-10 04:30:07 +01:00
class Command:
def __init__(
self,
client: AsyncClient,
cache: Cache,
alertmanager: AlertmanagerClient,
2021-01-10 04:30:07 +01:00
config: Config,
command: str,
room: MatrixRoom,
2022-07-10 14:06:36 +02:00
sender: str,
event_id: str,
2021-01-10 04:30:07 +01:00
):
"""A command made by a user.
2019-09-25 14:26:29 +02:00
Args:
client: The client to communicate to matrix with.
2019-09-25 14:26:29 +02:00
cache: Bot cache.
2019-09-25 14:26:29 +02:00
config: Bot configuration parameters.
command: The command and arguments.
2019-09-25 14:26:29 +02:00
room: The room the command was sent in.
2019-09-25 14:26:29 +02:00
event: The event describing the command.
2019-09-25 14:26:29 +02:00
"""
self.client = client
self.cache = cache
self.alertmanager = alertmanager
2019-10-26 01:40:05 +02:00
self.config = config
2019-09-25 14:26:29 +02:00
self.command = command
self.room = room
2022-07-10 14:06:36 +02:00
self.sender = sender
self.event_id = event_id
2019-09-25 14:26:29 +02:00
self.args = self.command.split()[1:]
2022-06-14 23:37:54 +02:00
async def process(self) -> None:
2019-09-25 14:26:29 +02:00
"""Process the command"""
if self.command.startswith("ack"):
await self._ack()
2022-07-06 01:26:27 +02:00
elif self.command.startswith("unack") or self.command.startswith("nack"):
2022-07-06 00:54:13 +02:00
await self._unack()
2019-09-25 14:26:29 +02:00
elif self.command.startswith("help"):
await self._show_help()
else:
await self._unknown_command()
async def _ack(self) -> None:
"""Acknowledge an alert and silence it for a certain duration in Alertmanager"""
2022-07-10 12:51:49 +02:00
matchers: List[AlertMatcher] = []
2022-07-10 02:40:04 +02:00
durations = []
for arg in self.args:
if "=~" in arg:
label, regex = arg.split("=~")
regex_matcher = AlertRegexMatcher(label, regex)
matchers.append(regex_matcher)
elif "=" in arg:
label, value = arg.split("=")
matcher = AlertMatcher(label, value)
matchers.append(matcher)
else:
durations.append(arg)
if len(durations) > 0:
duration = " ".join(durations)
else:
duration = "1d"
2022-07-10 02:40:04 +02:00
logger.debug(
2022-07-10 14:06:36 +02:00
f"Receiving a command to create a silence for a duration of {duration}"
)
2022-07-10 14:09:00 +02:00
logger.debug(f"Read alert fingerprints for event {self.event_id} from cache")
2022-07-05 23:35:19 +02:00
2022-07-10 15:11:25 +02:00
if self.event_id not in self.cache:
logger.error(f"Cannot find fingerprints for event {self.event_id} in cache")
return
2022-07-10 15:11:25 +02:00
2022-07-10 14:06:36 +02:00
alert_fingerprints = self.cache[self.event_id]
2022-07-10 15:11:25 +02:00
logger.debug(f"Found {len(alert_fingerprints)} in cache")
count_alert_not_found = 0
2022-07-10 15:11:25 +02:00
count_created_silences = 0
2022-07-05 23:35:19 +02:00
for alert_fingerprint in alert_fingerprints:
logger.debug(
f"Create silence for alert with fingerprint {alert_fingerprint} for a duration of {duration}"
)
2022-07-06 00:54:13 +02:00
try:
2022-07-09 09:56:28 +02:00
await self.alertmanager.create_silence(
2022-07-10 02:40:04 +02:00
alert_fingerprint,
duration,
2022-07-10 14:06:36 +02:00
self.room.user_name(self.sender),
2022-07-10 02:40:04 +02:00
matchers,
2022-07-06 00:54:13 +02:00
)
2022-07-09 09:56:28 +02:00
count_created_silences += 1
except AlertNotFoundError as e:
logger.warning(f"Unable to create silence: {e}")
count_alert_not_found += 1
2022-07-09 10:38:40 +02:00
except AlertmanagerError as e:
2022-07-08 23:22:31 +02:00
logger.exception(f"Unable to create silence: {e}", exc_info=e)
2022-07-06 00:54:13 +02:00
if count_alert_not_found > 0:
await send_text_to_room(
self.client,
self.room.room_id,
2022-07-11 23:45:15 +02:00
f"Sorry, I couldn't find {count_alert_not_found} alerts, therefore I couldn't create their silence.",
)
2022-07-11 23:43:27 +02:00
if count_created_silences > 0:
await send_text_to_room(
self.client,
self.room.room_id,
f"Created {count_created_silences} silences with a duration of {duration}.",
)
2022-07-06 00:54:13 +02:00
async def _unack(self) -> None:
"""Delete an alert's acknowledgement of an alert and remove corresponding silence in Alertmanager"""
2022-07-10 12:51:49 +02:00
matchers: List[AlertMatcher] = []
for arg in self.args:
if "=~" in arg:
label, regex = arg.split("=~")
regex_matcher = AlertRegexMatcher(label, regex)
matchers.append(regex_matcher)
elif "=" in arg:
label, value = arg.split("=")
matcher = AlertMatcher(label, value)
matchers.append(matcher)
2022-07-10 14:06:36 +02:00
logger.debug("Receiving a command to delete a silence")
2022-07-10 14:09:00 +02:00
logger.debug(f"Read alert fingerprints for event {self.event_id} from cache")
2022-07-06 00:54:13 +02:00
2022-07-10 15:11:25 +02:00
if self.event_id not in self.cache:
logger.error(f"Cannot find fingerprints for event {self.event_id} in cache")
return
2022-07-10 15:11:25 +02:00
2022-07-10 14:06:36 +02:00
alert_fingerprints = self.cache[self.event_id]
2022-07-10 15:11:25 +02:00
logger.debug(f"Found {len(alert_fingerprints)} in cache")
count_alert_not_found = 0
2022-07-10 15:11:25 +02:00
count_removed_silences = 0
2022-07-06 00:54:13 +02:00
for alert_fingerprint in alert_fingerprints:
logger.debug(
f"Delete silence for alert with fingerprint {alert_fingerprint}"
2022-07-05 23:35:19 +02:00
)
2022-07-06 00:54:13 +02:00
try:
2022-07-09 09:56:28 +02:00
removed_silences = await self.alertmanager.delete_silences(
2022-07-10 12:51:49 +02:00
alert_fingerprint, matchers
2022-07-09 00:08:51 +02:00
)
2022-07-09 09:56:28 +02:00
count_removed_silences += len(removed_silences)
except (AlertNotFoundError, SilenceNotFoundError) as e:
2022-07-11 23:41:30 +02:00
logger.error(f"Unable to delete silence: {e}")
count_alert_not_found += 1
2022-07-09 10:38:40 +02:00
except AlertmanagerError as e:
2022-07-08 23:22:31 +02:00
logger.exception(f"Unable to delete silence: {e}", exc_info=e)
2022-07-06 00:54:13 +02:00
if count_alert_not_found > 0:
await send_text_to_room(
self.client,
self.room.room_id,
2022-07-11 23:45:15 +02:00
f"Sorry, I couldn't find {count_alert_not_found} alerts, therefore I couldn't remove their silences.",
)
2022-07-11 23:43:27 +02:00
if count_removed_silences > 0:
await send_text_to_room(
self.client,
self.room.room_id,
f"Removed {count_removed_silences} silences.",
)
2019-09-25 14:26:29 +02:00
2022-06-14 23:37:54 +02:00
async def _show_help(self) -> None:
2019-09-25 14:26:29 +02:00
"""Show the help text"""
2022-07-10 14:06:36 +02:00
logger.debug(f"Displaying help to room {self.room.display_name}")
2019-09-25 14:26:29 +02:00
if not self.args:
2020-08-10 00:02:07 +02:00
text = (
"Hello, I am a bot made with matrix-nio! Use `help commands` to view "
"available commands."
)
2019-09-25 14:26:29 +02:00
await send_text_to_room(self.client, self.room.room_id, text)
return
topic = self.args[0]
if topic == "rules":
text = "These are the rules!"
elif topic == "commands":
text = "Available commands: ..."
2019-09-25 14:26:29 +02:00
else:
text = "Unknown help topic!"
await send_text_to_room(self.client, self.room.room_id, text)
2022-06-14 23:37:54 +02:00
async def _unknown_command(self) -> None:
logger.debug(
2022-07-10 14:06:36 +02:00
f"Sending unknown command response to room {self.room.display_name}"
)
2019-09-25 14:26:29 +02:00
await send_text_to_room(
self.client,
self.room.room_id,
f"Unknown command '{self.command}'. Try the 'help' command for more information.",
)