2022-07-11 23:18:57 +02:00
|
|
|
import unittest
|
|
|
|
from typing import Dict
|
2022-07-26 19:33:04 +02:00
|
|
|
from unittest.mock import Mock, call, patch
|
2022-07-11 23:18:57 +02:00
|
|
|
|
|
|
|
import aiohttp.test_utils
|
|
|
|
import nio
|
|
|
|
from aiohttp import web
|
|
|
|
from diskcache import Cache
|
|
|
|
from nio import LocalProtocolError, RoomSendResponse
|
|
|
|
|
|
|
|
import matrix_alertbot.webhook
|
|
|
|
from matrix_alertbot.config import Config
|
|
|
|
from matrix_alertbot.webhook import Webhook
|
|
|
|
|
|
|
|
|
|
|
|
def send_text_to_room_raise_error(
|
|
|
|
client: nio.AsyncClient, room_id: str, plaintext: str, html: str, notice: bool
|
|
|
|
) -> RoomSendResponse:
|
|
|
|
raise LocalProtocolError()
|
|
|
|
|
|
|
|
|
|
|
|
class WebhookApplicationTestCase(aiohttp.test_utils.AioHTTPTestCase):
|
|
|
|
async def get_application(self) -> web.Application:
|
|
|
|
self.fake_client = Mock(spec=nio.AsyncClient)
|
|
|
|
self.fake_cache = Mock(spec=Cache)
|
|
|
|
|
|
|
|
self.fake_config = Mock(spec=Config)
|
|
|
|
self.fake_config.port = aiohttp.test_utils.unused_port()
|
|
|
|
self.fake_config.address = "localhost"
|
|
|
|
self.fake_config.socket = "webhook.sock"
|
|
|
|
self.fake_config.room_id = "!abcdefg:example.com"
|
|
|
|
self.fake_config.cache_expire_time = 0
|
|
|
|
|
|
|
|
self.fake_alerts = {
|
|
|
|
"alerts": [
|
|
|
|
{
|
|
|
|
"fingerprint": "fingerprint1",
|
|
|
|
"generatorURL": "http://example.com/alert1",
|
|
|
|
"status": "firing",
|
|
|
|
"labels": {
|
|
|
|
"alertname": "alert1",
|
|
|
|
"severity": "critical",
|
|
|
|
"job": "job1",
|
|
|
|
},
|
|
|
|
"annotations": {"description": "some description1"},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"fingerprint": "fingerprint2",
|
|
|
|
"generatorURL": "http://example.com/alert2",
|
|
|
|
"status": "resolved",
|
|
|
|
"labels": {
|
|
|
|
"alertname": "alert2",
|
|
|
|
"severity": "warning",
|
|
|
|
"job": "job2",
|
|
|
|
},
|
|
|
|
"annotations": {"description": "some description2"},
|
|
|
|
},
|
|
|
|
]
|
|
|
|
}
|
|
|
|
|
|
|
|
webhook = Webhook(self.fake_client, self.fake_cache, self.fake_config)
|
|
|
|
return webhook.app
|
|
|
|
|
|
|
|
@patch.object(matrix_alertbot.webhook, "send_text_to_room")
|
2022-07-26 19:33:04 +02:00
|
|
|
async def test_post_alerts(self, fake_send_text_to_room: Mock) -> None:
|
2022-07-11 23:18:57 +02:00
|
|
|
data = self.fake_alerts
|
2022-07-12 18:22:11 +02:00
|
|
|
async with self.client.request("POST", "/alerts", json=data) as response:
|
2022-07-11 23:18:57 +02:00
|
|
|
self.assertEqual(200, response.status)
|
2022-07-26 19:33:04 +02:00
|
|
|
fake_send_text_to_room.assert_has_calls(
|
|
|
|
[
|
|
|
|
call(
|
|
|
|
self.fake_client,
|
|
|
|
self.fake_config.room_id,
|
|
|
|
"[🔥 CRITICAL] alert1: some description1",
|
|
|
|
"<font color='#dc3545'><b>[🔥 CRITICAL]</b></font> "
|
|
|
|
"<a href='http://example.com/alert1'>alert1</a> (job1)<br/>"
|
|
|
|
"some description1",
|
|
|
|
notice=False,
|
|
|
|
),
|
|
|
|
call(
|
|
|
|
self.fake_client,
|
|
|
|
self.fake_config.room_id,
|
|
|
|
"[🥦 RESOLVED] alert2: some description2",
|
|
|
|
"<font color='#33cc33'><b>[🥦 RESOLVED]</b></font> "
|
|
|
|
"<a href='http://example.com/alert2'>alert2</a> (job2)<br/>"
|
|
|
|
"some description2",
|
|
|
|
notice=False,
|
|
|
|
),
|
|
|
|
]
|
2022-07-11 23:18:57 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
@patch.object(matrix_alertbot.webhook, "send_text_to_room")
|
2022-07-26 19:33:04 +02:00
|
|
|
async def test_post_alerts_with_empty_data(
|
2022-07-11 23:18:57 +02:00
|
|
|
self, fake_send_text_to_room: Mock
|
|
|
|
) -> None:
|
2022-07-12 18:22:11 +02:00
|
|
|
async with self.client.request("POST", "/alerts", json={}) as response:
|
2022-07-11 23:18:57 +02:00
|
|
|
self.assertEqual(400, response.status)
|
|
|
|
error_msg = await response.text()
|
|
|
|
self.assertEqual("Data must contain 'alerts' key.", error_msg)
|
|
|
|
fake_send_text_to_room.assert_not_called()
|
|
|
|
|
|
|
|
@patch.object(matrix_alertbot.webhook, "send_text_to_room")
|
2022-07-26 19:33:04 +02:00
|
|
|
async def test_post_empty_alerts(self, fake_send_text_to_room: Mock) -> None:
|
2022-07-11 23:18:57 +02:00
|
|
|
data: Dict = {"alerts": []}
|
2022-07-12 18:22:11 +02:00
|
|
|
async with self.client.request("POST", "/alerts", json=data) as response:
|
2022-07-11 23:18:57 +02:00
|
|
|
self.assertEqual(400, response.status)
|
|
|
|
error_msg = await response.text()
|
|
|
|
self.assertEqual("Alerts cannot be empty.", error_msg)
|
|
|
|
fake_send_text_to_room.assert_not_called()
|
|
|
|
|
|
|
|
@patch.object(matrix_alertbot.webhook, "send_text_to_room")
|
2022-07-26 19:33:04 +02:00
|
|
|
async def test_post_invalid_alerts(self, fake_send_text_to_room: Mock) -> None:
|
2022-07-11 23:18:57 +02:00
|
|
|
data = {"alerts": "invalid"}
|
2022-07-12 18:22:11 +02:00
|
|
|
async with self.client.request("POST", "/alerts", json=data) as response:
|
2022-07-11 23:18:57 +02:00
|
|
|
self.assertEqual(400, response.status)
|
|
|
|
error_msg = await response.text()
|
|
|
|
self.assertEqual("Alerts must be a list.", error_msg)
|
|
|
|
fake_send_text_to_room.assert_not_called()
|
|
|
|
|
|
|
|
@patch.object(matrix_alertbot.webhook, "send_text_to_room")
|
2022-07-26 19:33:04 +02:00
|
|
|
async def test_post_alerts_with_empty_items(
|
2022-07-11 23:18:57 +02:00
|
|
|
self, fake_send_text_to_room: Mock
|
|
|
|
) -> None:
|
|
|
|
data: Dict = {"alerts": [{}]}
|
2022-07-12 18:22:11 +02:00
|
|
|
async with self.client.request("POST", "/alerts", json=data) as response:
|
2022-07-11 23:18:57 +02:00
|
|
|
self.assertEqual(400, response.status)
|
|
|
|
error_msg = await response.text()
|
|
|
|
self.assertEqual("Invalid alert: {}.", error_msg)
|
|
|
|
fake_send_text_to_room.assert_not_called()
|
|
|
|
|
|
|
|
@patch.object(
|
|
|
|
matrix_alertbot.webhook,
|
|
|
|
"send_text_to_room",
|
|
|
|
side_effect=send_text_to_room_raise_error,
|
|
|
|
)
|
2022-07-26 19:33:04 +02:00
|
|
|
async def test_post_alerts_raise_send_error(
|
2022-07-11 23:18:57 +02:00
|
|
|
self, fake_send_text_to_room: Mock
|
|
|
|
) -> None:
|
|
|
|
data = self.fake_alerts
|
2022-07-12 18:22:11 +02:00
|
|
|
async with self.client.request("POST", "/alerts", json=data) as response:
|
2022-07-11 23:18:57 +02:00
|
|
|
self.assertEqual(500, response.status)
|
|
|
|
error_msg = await response.text()
|
|
|
|
self.assertEqual(
|
2022-07-26 19:33:04 +02:00
|
|
|
"An error occured when sending alert with fingerprint 'fingerprint1' to Matrix room.",
|
|
|
|
error_msg,
|
2022-07-11 23:18:57 +02:00
|
|
|
)
|
|
|
|
fake_send_text_to_room.assert_called_once()
|
|
|
|
|
2022-07-12 18:19:52 +02:00
|
|
|
async def test_health(self) -> None:
|
|
|
|
async with self.client.request("GET", "/health") as response:
|
|
|
|
self.assertEqual(200, response.status)
|
|
|
|
|
2022-07-11 23:18:57 +02:00
|
|
|
|
|
|
|
class WebhookServerTestCase(unittest.IsolatedAsyncioTestCase):
|
|
|
|
async def asyncSetUp(self) -> None:
|
|
|
|
self.fake_client = Mock(spec=nio.AsyncClient)
|
|
|
|
self.fake_cache = Mock(spec=Cache)
|
|
|
|
|
|
|
|
self.fake_config = Mock(spec=Config)
|
|
|
|
self.fake_config.port = aiohttp.test_utils.unused_port()
|
|
|
|
self.fake_config.address = "localhost"
|
|
|
|
self.fake_config.socket = "webhook.sock"
|
|
|
|
self.fake_config.room_id = "!abcdefg:example.com"
|
|
|
|
self.fake_config.cache_expire_time = 0
|
|
|
|
|
|
|
|
@patch.object(matrix_alertbot.webhook.web, "TCPSite", autospec=True)
|
|
|
|
async def test_webhook_start_address_port(self, fake_tcp_site: Mock) -> None:
|
|
|
|
webhook = Webhook(self.fake_client, self.fake_cache, self.fake_config)
|
|
|
|
await webhook.start()
|
|
|
|
|
|
|
|
fake_tcp_site.assert_called_once_with(
|
|
|
|
webhook.runner, self.fake_config.address, self.fake_config.port
|
|
|
|
)
|
|
|
|
|
|
|
|
await webhook.close()
|
|
|
|
|
|
|
|
@patch.object(matrix_alertbot.webhook.web, "UnixSite", autospec=True)
|
|
|
|
async def test_webhook_start_unix_socket(self, fake_unix_site: Mock) -> None:
|
|
|
|
self.fake_config.address = None
|
|
|
|
self.fake_config.port = None
|
|
|
|
|
|
|
|
webhook = Webhook(self.fake_client, self.fake_cache, self.fake_config)
|
|
|
|
await webhook.start()
|
|
|
|
|
|
|
|
fake_unix_site.assert_called_once_with(webhook.runner, self.fake_config.socket)
|
|
|
|
|
|
|
|
await webhook.close()
|