123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331 |
- import asyncio
- from datetime import timedelta
- from unittest import mock
- from aioresponses import aioresponses
- from django.core import mail
- from django.core.cache import cache
- from django.urls import reverse
- from django.utils import timezone
- from freezegun import freeze_time
- from model_bakery import baker
- from glitchtip.test_utils.test_case import GlitchTipTestCase
- from organizations_ext.models import OrganizationUserRole
- from projects.models import ProjectAlertStatus
- from ..constants import MonitorType
- from ..models import Monitor, MonitorCheck
- from ..tasks import UPTIME_COUNTER_KEY, bucket_monitors, dispatch_checks
- from ..utils import fetch_all
- from ..webhooks import send_uptime_as_webhook
- class UptimeTestCase(GlitchTipTestCase):
- @mock.patch("glitchtip.uptime.tasks.perform_checks.run")
- def test_dispatch_checks(self, mocked):
- mock.return_value = None
- test_url = "https://example.com"
- with freeze_time("2020-01-01"):
- mon1 = baker.make(Monitor, url=test_url, monitor_type=MonitorType.GET)
- mon2 = baker.make(Monitor, url=test_url, monitor_type=MonitorType.GET)
- baker.make(MonitorCheck, monitor=mon1)
- self.assertEqual(mocked.call_count, 2)
- cache.set(UPTIME_COUNTER_KEY, 59)
- with freeze_time("2020-01-02"):
- baker.make(MonitorCheck, monitor=mon2)
- dispatch_checks()
- self.assertEqual(mocked.call_count, 3)
- @aioresponses()
- def test_fetch_all(self, mocked):
- test_url = "https://example.com"
- mocked.get(test_url, status=200)
- mon1 = baker.make(Monitor, url=test_url, monitor_type=MonitorType.GET)
- mocked.get(test_url, status=200)
- monitors = list(Monitor.objects.all().values())
- results = asyncio.run(fetch_all(monitors))
- self.assertEqual(results[0]["id"], mon1.pk)
- @aioresponses()
- def test_monitor_checks_integration(self, mocked):
- test_url = "https://example.com"
- mocked.get(test_url, status=200)
- with freeze_time("2020-01-01"):
- mon = baker.make(Monitor, url=test_url, monitor_type=MonitorType.GET)
- self.assertEqual(mon.checks.count(), 1)
- mocked.get(test_url, status=200)
- with freeze_time("2020-01-01"):
- dispatch_checks()
- self.assertEqual(mon.checks.count(), 1)
- cache.set(UPTIME_COUNTER_KEY, 59)
- with freeze_time("2020-01-02"):
- with self.assertNumQueries(3):
- dispatch_checks()
- self.assertEqual(mon.checks.count(), 2)
- @aioresponses()
- def test_expected_response(self, mocked):
- test_url = "https://example.com"
- mocked.get(test_url, status=200, body="Status: OK")
- monitor = baker.make(
- Monitor,
- name=test_url,
- url=test_url,
- expected_body="OK",
- monitor_type=MonitorType.GET,
- )
- check = monitor.checks.first()
- self.assertTrue(check.is_up)
- mocked.get(test_url, status=200, body="Status: Failure")
- monitor = baker.make(
- Monitor,
- name=test_url,
- url=test_url,
- expected_body="OK",
- monitor_type=MonitorType.GET,
- )
- check = monitor.checks.first()
- self.assertFalse(check.is_up)
- self.assertEqual(check.data["payload"], "Status: Failure")
- @aioresponses()
- @mock.patch("requests.post")
- def test_monitor_notifications(self, mocked, mock_post):
- self.create_user_and_project()
- test_url = "https://example.com"
- mocked.get(test_url, status=200)
- with freeze_time("2020-01-01"):
- baker.make(
- Monitor,
- name=test_url,
- url=test_url,
- monitor_type=MonitorType.GET,
- project=self.project,
- )
- baker.make(
- "alerts.AlertRecipient",
- alert__uptime=True,
- alert__project=self.project,
- recipient_type="email",
- )
- baker.make(
- "alerts.AlertRecipient",
- alert__uptime=True,
- alert__project=self.project,
- recipient_type="webhook",
- url="https://example.com",
- )
- mocked.get(test_url, status=500)
- cache.set(UPTIME_COUNTER_KEY, 59)
- with freeze_time("2020-01-02"):
- dispatch_checks()
- self.assertEqual(len(mail.outbox), 1)
- self.assertIn("is down", mail.outbox[0].body)
- mock_post.assert_called_once()
- mocked.get(test_url, status=500)
- cache.set(UPTIME_COUNTER_KEY, 59)
- with freeze_time("2020-01-03"):
- dispatch_checks()
- self.assertEqual(len(mail.outbox), 1)
- mocked.get(test_url, status=200)
- cache.set(UPTIME_COUNTER_KEY, 59)
- with freeze_time("2020-01-04"):
- dispatch_checks()
- self.assertEqual(len(mail.outbox), 2)
- self.assertIn("is back up", mail.outbox[1].body)
- @aioresponses()
- @mock.patch("requests.post")
- def test_discord_webhook(self, mocked, mocked_post):
- self.create_user_and_project()
- test_url = "https://example.com"
- mocked.get(test_url, status=200)
- check = baker.make(
- "uptime.MonitorCheck",
- monitor__monitor_type=MonitorType.GET,
- monitor__url=test_url,
- monitor__project=self.project,
- )
- recipient = baker.make("alerts.AlertRecipient", recipient_type="discord")
- send_uptime_as_webhook(recipient, check.pk, True, timezone.now())
- mocked_post.assert_called_once()
- @aioresponses()
- def test_notification_default_scope(self, mocked):
- """Subscribe by default should not result in alert emails for non-team members"""
- self.create_user_and_project()
- test_url = "https://example.com"
- # user2 is an org member but not in a relevant team, should not receive alerts
- user2 = baker.make("users.user")
- org_user2 = self.organization.add_user(user2, OrganizationUserRole.MEMBER)
- team2 = baker.make("teams.Team", organization=self.organization)
- team2.members.add(org_user2)
- # user3 is in team3 which should receive alerts
- user3 = baker.make("users.user")
- org_user3 = self.organization.add_user(user3, OrganizationUserRole.MEMBER)
- self.team.members.add(org_user3)
- team3 = baker.make("teams.Team", organization=self.organization)
- team3.members.add(org_user3)
- team3.projects.add(self.project)
- baker.make(
- "alerts.AlertRecipient",
- alert__uptime=True,
- alert__project=self.project,
- recipient_type="email",
- )
- mocked.get(test_url, status=200)
- with freeze_time("2020-01-01"):
- baker.make(
- Monitor,
- name=test_url,
- url=test_url,
- monitor_type=MonitorType.GET,
- project=self.project,
- )
- mocked.get(test_url, status=500)
- cache.set(UPTIME_COUNTER_KEY, 59)
- with self.assertNumQueries(10):
- with freeze_time("2020-01-02"):
- dispatch_checks()
- self.assertNotIn(user2.email, mail.outbox[0].to)
- self.assertIn(user3.email, mail.outbox[0].to)
- self.assertEqual(len(mail.outbox[0].to), 2)
- @aioresponses()
- def test_user_project_alert_scope(self, mocked):
- """User project alert should not result in alert emails for non-team members"""
- self.create_user_and_project()
- test_url = "https://example.com"
- baker.make(
- "alerts.AlertRecipient",
- alert__uptime=True,
- alert__project=self.project,
- recipient_type="email",
- )
- user2 = baker.make("users.user")
- self.organization.add_user(user2, OrganizationUserRole.MEMBER)
- baker.make(
- "projects.UserProjectAlert",
- user=user2,
- project=self.project,
- status=ProjectAlertStatus.ON,
- )
- mocked.get(test_url, status=200)
- with freeze_time("2020-01-01"):
- baker.make(
- Monitor,
- name=test_url,
- url=test_url,
- monitor_type=MonitorType.GET,
- project=self.project,
- )
- mocked.get(test_url, status=500)
- cache.set(UPTIME_COUNTER_KEY, 59)
- with self.assertNumQueries(10):
- with freeze_time("2020-01-02"):
- dispatch_checks()
- self.assertNotIn(user2.email, mail.outbox[0].to)
- def xtest_heartbeat(self):
- """
- Cannot run due to async code, it doesn't close the DB connection
- Run manually with --keepdb
- """
- self.create_user_and_project()
- with freeze_time("2020-01-01"):
- monitor = baker.make(
- Monitor,
- monitor_type=MonitorType.HEARTBEAT,
- project=self.project,
- )
- baker.make(
- "alerts.AlertRecipient",
- alert__uptime=True,
- alert__project=self.project,
- recipient_type="email",
- )
- url = reverse(
- "heartbeat-check",
- kwargs={
- "organization_slug": monitor.organization.slug,
- "endpoint_id": monitor.endpoint_id,
- },
- )
- self.assertFalse(monitor.checks.exists())
- self.client.post(url)
- self.assertTrue(monitor.checks.filter(is_up=True).exists())
- dispatch_checks()
- self.assertTrue(monitor.checks.filter(is_up=True).exists())
- self.assertEqual(len(mail.outbox), 0)
- cache.set(UPTIME_COUNTER_KEY, 59)
- with freeze_time("2020-01-02"):
- dispatch_checks()
- self.assertEqual(len(mail.outbox), 1)
- cache.set(UPTIME_COUNTER_KEY, 59)
- with freeze_time("2020-01-03"):
- dispatch_checks() # Still down
- self.assertEqual(len(mail.outbox), 1)
- cache.set(UPTIME_COUNTER_KEY, 59)
- with freeze_time("2020-01-04"):
- self.client.post(url) # Back up
- self.assertEqual(len(mail.outbox), 2)
- def test_heartbeat_grace_period(self):
- # Don't alert users when heartbeat check has never come in
- self.create_user_and_project()
- baker.make(Monitor, monitor_type=MonitorType.HEARTBEAT, project=self.project)
- dispatch_checks()
- self.assertEqual(len(mail.outbox), 0)
- @mock.patch("glitchtip.uptime.tasks.perform_checks.run")
- def test_bucket_monitors(self, _):
- interval_timeouts = [
- [1, 10],
- [3, 20],
- [3, None],
- [10, 10],
- [2, 40],
- [3, 50],
- ]
- for interval, timeout in interval_timeouts:
- baker.make(
- Monitor,
- url="http://example.com",
- interval=timedelta(seconds=interval),
- timeout=timeout,
- )
- monitors = Monitor.objects.all()
- result = bucket_monitors(monitors, 1)
- @mock.patch("glitchtip.uptime.utils.asyncio.open_connection")
- def test_port_monitor(self, mocked):
- self.create_user_and_project()
- monitor = baker.make(
- Monitor,
- url="example.com:80",
- monitor_type=MonitorType.PORT,
- project=self.project,
- )
- mocked.assert_called_once()
- self.assertTrue(monitor.checks.filter(is_up=True).exists())
|