|
|
- # coding: utf-8
-
- from datetime import timedelta as td
- import json
- from unittest.mock import patch
-
- from django.core import mail
- from django.utils.timezone import now
- from hc.api.models import Channel, Check, Notification
- from hc.test import BaseTestCase
-
-
- class NotifySmsTestCase(BaseTestCase):
- def setUp(self):
- super().setUp()
-
- self.check = Check(project=self.project)
- self.check.status = "down"
- self.check.last_ping = now() - td(minutes=61)
- self.check.save()
-
- spec = {"value": "+1234567890", "up": False, "down": True}
- self.channel = Channel(project=self.project, kind="sms")
- self.channel.value = json.dumps(spec)
- self.channel.save()
- self.channel.checks.add(self.check)
-
- @patch("hc.api.transports.requests.request")
- def test_it_works(self, mock_post):
- self.check.last_ping = now() - td(hours=2)
- mock_post.return_value.status_code = 200
-
- self.channel.notify(self.check)
-
- args, kwargs = mock_post.call_args
- payload = kwargs["data"]
- self.assertEqual(payload["To"], "+1234567890")
- self.assertNotIn("\xa0", payload["Body"])
- self.assertIn("is DOWN", payload["Body"])
-
- n = Notification.objects.get()
- callback_path = f"/api/v1/notifications/{n.code}/status"
- self.assertTrue(payload["StatusCallback"].endswith(callback_path))
-
- # sent SMS counter should go up
- self.profile.refresh_from_db()
- self.assertEqual(self.profile.sms_sent, 1)
-
- @patch("hc.api.transports.requests.request")
- def test_it_enforces_limit(self, mock_post):
- # At limit already:
- self.profile.last_sms_date = now()
- self.profile.sms_sent = 50
- self.profile.save()
-
- self.channel.notify(self.check)
- self.assertFalse(mock_post.called)
-
- n = Notification.objects.get()
- self.assertTrue("Monthly SMS limit exceeded" in n.error)
-
- # And email should have been sent
- self.assertEqual(len(mail.outbox), 1)
-
- email = mail.outbox[0]
- self.assertEqual(email.to[0], "[email protected]")
- self.assertEqual(email.subject, "Monthly SMS Limit Reached")
-
- @patch("hc.api.transports.requests.request")
- def test_it_resets_limit_next_month(self, mock_post):
- # At limit, but also into a new month
- self.profile.sms_sent = 50
- self.profile.last_sms_date = now() - td(days=100)
- self.profile.save()
-
- mock_post.return_value.status_code = 200
-
- self.channel.notify(self.check)
- self.assertTrue(mock_post.called)
-
- @patch("hc.api.transports.requests.request")
- def test_it_does_not_escape_special_characters(self, mock_post):
- self.check.name = "Foo > Bar & Co"
- self.check.last_ping = now() - td(hours=2)
-
- mock_post.return_value.status_code = 200
-
- self.channel.notify(self.check)
-
- args, kwargs = mock_post.call_args
- payload = kwargs["data"]
- self.assertIn("Foo > Bar & Co", payload["Body"])
-
- @patch("hc.api.transports.requests.request")
- def test_it_handles_disabled_down_notification(self, mock_post):
- payload = {"value": "+123123123", "up": True, "down": False}
- self.channel.value = json.dumps(payload)
-
- self.channel.notify(self.check)
- self.assertFalse(mock_post.called)
-
- @patch("hc.api.transports.requests.request")
- def test_it_sends_up_notification(self, mock_post):
- payload = {"value": "+123123123", "up": True, "down": False}
- self.channel.value = json.dumps(payload)
-
- self.check.last_ping = now()
- self.check.status = "up"
- mock_post.return_value.status_code = 200
-
- self.channel.notify(self.check)
-
- args, kwargs = mock_post.call_args
- payload = kwargs["data"]
- self.assertIn("is UP", payload["Body"])
|