Skip to content

Commit

Permalink
refactor logic
Browse files Browse the repository at this point in the history
  • Loading branch information
IbraAoad committed Jun 3, 2024
1 parent 4105cc6 commit 6eb7120
Show file tree
Hide file tree
Showing 4 changed files with 81 additions and 54 deletions.
2 changes: 1 addition & 1 deletion CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0

## [0.9.0] - 2024-05-30

- Adding PagerDuty native support (#76).
- Added PagerDuty native support (#76).


## [0.8.0] - 2024-03-07
Expand Down
73 changes: 46 additions & 27 deletions cos_alerter/alerter.py
Original file line number Diff line number Diff line change
Expand Up @@ -215,12 +215,7 @@ def reset_alert_timeout(self):
"""Set the "last alert time" to right now."""
# In case an instance was down, resolve the PagerDuty incident before resetting the last alert time
if self.is_down():
pagerduty_destinations, _ = split_destinations()
handle_pagerduty_incidents(
incident_type="resolve",
dedup_key=f"{self.clientid}-{self.last_alert_datetime()}",
destinations_list=pagerduty_destinations,
)
self.resolve_existing_alerts()
logger.debug("Resetting alert timeout for %s.", self.clientid)
self.data["alert_time"] = time.monotonic()

Expand Down Expand Up @@ -284,15 +279,26 @@ def notify(self):
# Sending notifications can be a long operation so handle that in a separate thread.
# This avoids interfering with the execution of the main loop.
notify_thread = threading.Thread(
target=send_notifications,
target=send_all_notifications,
kwargs={
"title": title,
"body": body,
"destinations": split_destinations(config["notify"]["destinations"]),
"incident_type": "trigger",
"dedup_key": f"{self.clientid}-{self.last_alert_datetime()}",
},
)
notify_thread.start()

def resolve_existing_alerts(self):
"""Resolves the current alerts."""
categorized_destinations = split_destinations(config["notify"]["destinations"])
handle_pagerduty_incidents(
incident_type="resolve",
dedup_key=f"{self.clientid}-{self.last_alert_datetime()}",
destinations=categorized_destinations["pagerduty"],
)


def now_datetime():
"""Return the current datetime using the monotonic clock."""
Expand All @@ -305,48 +311,59 @@ def up_time():
return time.monotonic() - state["start_time"]


def split_destinations():
"""Split destinations into PagerDuty and non-PagerDuty lists."""
pagerduty_destinations = [
source for source in config["notify"]["destinations"] if source.startswith("pagerduty")
]
non_pagerduty_destinations = [
source for source in config["notify"]["destinations"] if not source.startswith("pagerduty")
]
return pagerduty_destinations, non_pagerduty_destinations
def split_destinations(destinations: list):
"""Split destinations into categorized lists."""
categorized_destinations = {"standard": [], "pagerduty": []}

for source in destinations:
if source.startswith("pagerduty"):
categorized_destinations["pagerduty"].append(source)
else:
categorized_destinations["standard"].append(source)

return categorized_destinations


def send_notifications(title: str, body: str, dedup_key: str):
def send_all_notifications(
title: str, body: str, destinations: list, incident_type: str, dedup_key: str
):
"""Send a notification to all receivers."""
send_standard_notifications(title=title, body=body, destinations=destinations["standard"])
handle_pagerduty_incidents(
incident_type=incident_type,
dedup_key=dedup_key,
destinations=destinations["pagerduty"],
incident_summary=body,
)


def send_standard_notifications(title: str, body: str, destinations: list):
"""Send a notification to all standard receivers."""
# TODO: Since this is run in its own thread, we have to make sure we properly
# log failures here.
pagerduty_destinations, non_pagerduty_destinations = split_destinations()

# Send notifications to non-PagerDuty destinations
sender = apprise.Apprise()
for source in non_pagerduty_destinations:
for source in destinations:
sender.add(source)
sender.notify(title=title, body=body)

# Send notifications to PagerDuty destinations
handle_pagerduty_incidents("trigger", dedup_key, pagerduty_destinations, body)


def handle_pagerduty_incidents(
incident_type: str,
dedup_key: str,
destinations_list: list,
destinations: list,
incident_summary: Optional[str] = None,
):
"""Handles PagerDuty incidents by triggering or resolving incidents based on the specified incident type.
Args:
incident_type (str): The type of incident action to perform. Should be either 'trigger' or 'resolve'.
dedup_key (str): The deduplication key to uniquely identify the incident.
destinations_list (list): List of destinations to handle PagerDuty incidents for.
destinations (list): List of destinations to handle PagerDuty incidents for.
incident_summary (str, optional): A summary of the incident, used only when triggering an incident. Defaults to None.
"""
for source in destinations_list:
for source in destinations:
integration_key = source.split("//")[1].split("@")[0]
session = EventsAPISession(integration_key)

Expand All @@ -359,8 +376,10 @@ def handle_pagerduty_incidents(
def send_test_notification():
"""Signal handler which sends a test email to all configured receivers."""
logger.info("Sending test notifications.")
send_notifications(
send_all_notifications(
title="COS-Alerter test email.",
body="This is a test email automatically generated by COS-alerter.",
dedup_key="testkey",
destinations=split_destinations(config["notify"]["destinations"]),
incident_type="trigger",
dedup_key="test-dedup-key",
)
1 change: 1 addition & 0 deletions tests/helpers.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
DESTINATIONS = [
"mailtos://user:pass@domain/[email protected],[email protected]",
"slack://xoxb-1234-1234-4ddbc191d40ee098cbaae6f3523ada2d/#general",
"pagerduty://integration-key@api-key",
]

CONFIG = {
Expand Down
59 changes: 33 additions & 26 deletions tests/test_alerter.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,13 +10,24 @@
import freezegun
import yaml
from helpers import DESTINATIONS
from pdpyras import EventsAPISession

from cos_alerter.alerter import AlerterState, config, send_test_notification, up_time
from cos_alerter.alerter import (
AlerterState,
config,
send_test_notification,
split_destinations,
up_time,
)


def assert_notifications(notify_mock, add_mock, title, body):
add_mock.assert_has_calls([unittest.mock.call(x) for x in DESTINATIONS])
def assert_notifications(notify_mock, add_mock, pd_mock, title, body, dedup_key):
categorized_destinations = split_destinations(DESTINATIONS)
add_mock.assert_has_calls(
[unittest.mock.call(x) for x in categorized_destinations["standard"]]
)
notify_mock.assert_called_with(title=title, body=body)
pd_mock.assert_called_with(source="cos-alerter", summary=body, dedup_key=dedup_key)


def test_config_gets_item(fake_fs):
Expand Down Expand Up @@ -142,7 +153,8 @@ def test_is_down_from_initialize(monotonic_mock, fake_fs):

@freezegun.freeze_time("2023-01-01")
@unittest.mock.patch("time.monotonic")
def test_is_down_with_reset_alert_timeout(monotonic_mock, fake_fs):
@unittest.mock.patch.object(EventsAPISession, "resolve")
def test_is_down_with_reset_alert_timeout(pd_mock, monotonic_mock, fake_fs):
monotonic_mock.return_value = 1000
AlerterState.initialize()
state = AlerterState(clientid="clientid1")
Expand All @@ -153,6 +165,7 @@ def test_is_down_with_reset_alert_timeout(monotonic_mock, fake_fs):
assert state.is_down() is False
monotonic_mock.return_value = 2330 # Five and a half minutes have passed
assert state.is_down() is True
pd_mock.assert_called_with(f"{state.clientid}-None")


@freezegun.freeze_time("2023-01-01")
Expand Down Expand Up @@ -201,21 +214,6 @@ def test_is_down_from_graceful_shutdown(monotonic_mock, fake_fs):
assert state.is_down() is True


@freezegun.freeze_time("2023-01-01")
@unittest.mock.patch("time.monotonic")
def test_is_down(monotonic_mock, fake_fs):
monotonic_mock.return_value = 1000
AlerterState.initialize()
state = AlerterState(clientid="clientid1")
with state:
monotonic_mock.return_value = 2000
state.reset_alert_timeout()
monotonic_mock.return_value = 2180 # Three minutes have passed
assert state.is_down() is False
monotonic_mock.return_value = 2330 # Five and a half minutes have passed
assert state.is_down() is True


@freezegun.freeze_time("2023-01-01")
@unittest.mock.patch("time.monotonic")
def test_recently_notified(monotonic_mock, fake_fs):
Expand All @@ -234,46 +232,55 @@ def test_recently_notified(monotonic_mock, fake_fs):
@unittest.mock.patch("time.monotonic")
@unittest.mock.patch.object(apprise.Apprise, "add")
@unittest.mock.patch.object(apprise.Apprise, "notify")
def test_notify(notify_mock, add_mock, monotonic_mock, fake_fs):
@unittest.mock.patch.object(EventsAPISession, "trigger")
def test_notify(pd_mock, notify_mock, add_mock, monotonic_mock, fake_fs):
monotonic_mock.return_value = 1000
AlerterState.initialize()
state = AlerterState(clientid="clientid1")

dedup_key = f"{state.clientid}-{state.last_alert_datetime()}"
with state:
state.notify()
for thread in threading.enumerate():
if thread != threading.current_thread():
thread.join()

assert_notifications(
notify_mock,
add_mock,
notify_mock=notify_mock,
add_mock=add_mock,
pd_mock=pd_mock,
title="**Alertmanager is Down!**",
body=textwrap.dedent(
"""
Your Alertmanager instance: clientid1 seems to be down!
It has not alerted COS-Alerter ever.
"""
),
dedup_key=dedup_key,
)

# Make sure if we try again, nothing is sent
notify_mock.reset_mock()
pd_mock.reset_mock()

with state:
state.notify()
for thread in threading.enumerate():
if thread != threading.current_thread():
thread.join()
notify_mock.assert_not_called()
pd_mock.assert_not_called()


@unittest.mock.patch.object(apprise.Apprise, "add")
@unittest.mock.patch.object(apprise.Apprise, "notify")
def test_send_test_notification(notify_mock, add_mock, fake_fs):
@unittest.mock.patch.object(EventsAPISession, "trigger")
def test_send_test_notification(pd_mock, notify_mock, add_mock, fake_fs):
send_test_notification()
assert_notifications(
notify_mock,
add_mock,
notify_mock=notify_mock,
add_mock=add_mock,
pd_mock=pd_mock,
title="COS-Alerter test email.",
body="This is a test email automatically generated by COS-alerter.",
dedup_key="test-dedup-key",
)

0 comments on commit 6eb7120

Please sign in to comment.