0
0
Fork 0
mirror of https://github.com/healthchecks/healthchecks.git synced 2025-04-04 21:05:26 +00:00

Update MS Teams Workflows notification to include failure reason

cc: 
This commit is contained in:
Pēteris Caune 2024-12-17 09:50:29 +02:00
parent aff41f6688
commit f390d6eece
No known key found for this signature in database
GPG key ID: E28D7679E9A9EDE2
3 changed files with 47 additions and 7 deletions

View file

@ -1,4 +1,3 @@
from __future__ import annotations
import json
@ -18,6 +17,7 @@ class NotifyMsTeamsTestCase(BaseTestCase):
super().setUp()
self.check = Check(project=self.project)
self.check.name = "Foo"
# Transport classes should use flip.new_status,
# so the status "paused" should not appear anywhere
self.check.status = "paused"
@ -39,6 +39,7 @@ class NotifyMsTeamsTestCase(BaseTestCase):
self.flip.created = now()
self.flip.old_status = "new"
self.flip.new_status = "down"
self.flip.reason = "timeout"
def facts(self, payload: Any) -> dict[str, str]:
card = payload["attachments"][0]["content"]
@ -49,8 +50,6 @@ class NotifyMsTeamsTestCase(BaseTestCase):
def test_it_works(self, mock_post: Mock) -> None:
mock_post.return_value.status_code = 200
self.check.name = "_underscores_ & more"
self.channel.notify(self.flip)
assert Notification.objects.count() == 1
@ -60,10 +59,14 @@ class NotifyMsTeamsTestCase(BaseTestCase):
card = payload["attachments"][0]["content"]
# summary and title should be the same, except
# title should have any special HTML characters escaped
self.assertEqual(card["fallbackText"], "_underscores_ & more” is DOWN.")
self.assertEqual(card["fallbackText"], "Foo” is DOWN.")
heading = card["body"][0]
self.assertEqual(heading["text"], "🔴 “_underscores_ & more” is DOWN.")
self.assertEqual(
heading["text"],
"🔴 “Foo” is DOWN (success signal did not"
" arrive on time, grace time passed).",
)
facts = self.facts(payload)
self.assertEqual(facts["Last Ping:"], "Success, 10 minutes ago")
@ -73,6 +76,30 @@ class NotifyMsTeamsTestCase(BaseTestCase):
serialized = json.dumps(payload)
self.assertNotIn(str(self.check.code), serialized)
@patch("hc.api.transports.curl.request", autospec=True)
def test_it_handles_reason_fail(self, mock_post: Mock) -> None:
mock_post.return_value.status_code = 200
self.flip.reason = "fail"
self.channel.notify(self.flip)
payload = mock_post.call_args.kwargs["json"]
text = payload["attachments"][0]["content"]["body"][0]["text"]
self.assertEqual(text, "🔴 “Foo” is DOWN (received a failure signal).")
@patch("hc.api.transports.curl.request", autospec=True)
def test_it_escapes_special_characters(self, mock_post: Mock) -> None:
mock_post.return_value.status_code = 200
self.check.name = "_underscores_ & more"
self.channel.notify(self.flip)
payload = mock_post.call_args.kwargs["json"]
card = payload["attachments"][0]["content"]
self.assertEqual(card["fallbackText"], "“_underscores_ & more” is DOWN.")
heading = card["body"][0]
self.assertIn("🔴 “_underscores_ & more” is DOWN", heading["text"])
@patch("hc.api.transports.curl.request", autospec=True)
def test_it_shows_cron_schedule_and_tz(self, mock_post: Mock) -> None:
mock_post.return_value.status_code = 200

View file

@ -1229,7 +1229,13 @@ class MsTeamsWorkflow(HttpTransport):
check = flip.owner
name = check.name_then_code()
fields = SlackFields()
indicator = "🔴" if flip.new_status == "down" else "🟢"
ctx = {
"flip": flip,
"check": flip.owner,
"status": flip.new_status,
}
text = tmpl("msteamsw_message.html", **ctx)
result: JSONDict = {
"type": "message",
"attachments": [
@ -1244,7 +1250,7 @@ class MsTeamsWorkflow(HttpTransport):
"body": [
{
"type": "TextBlock",
"text": f"{indicator}{escape(name)}” is {flip.new_status.upper()}.",
"text": text,
"weight": "bolder",
"size": "medium",
"wrap": True,

View file

@ -0,0 +1,7 @@
{% load linemode %}{% linemode %}
{% if status == "down" %}
{% line %}🔴 “{{ check.name_then_code }}” is DOWN{% if flip.reason %} ({{ flip.reason_long }}){% endif %}.{% endline %}
{% else %}
{% line %}🟢 “{{ check.name_then_code }}” is UP.{% endline %}
{% endif %}
{% endlinemode %}