mirror of
https://github.com/alerta/alerta.git
synced 2025-01-25 01:39:40 +00:00
248 lines
9.4 KiB
Python
248 lines
9.4 KiB
Python
import json
|
|
import time
|
|
import unittest
|
|
from uuid import uuid4
|
|
|
|
from alerta.app import create_app, db, plugins
|
|
from alerta.models.heartbeat import Heartbeat
|
|
|
|
|
|
class HeartbeatsTestCase(unittest.TestCase):
|
|
|
|
def setUp(self):
|
|
|
|
test_config = {
|
|
'TESTING': True,
|
|
'AUTH_REQUIRED': False,
|
|
'ALERT_TIMEOUT': 2,
|
|
'HEARTBEAT_TIMEOUT': 4,
|
|
'HEARTBEAT_EVENTS': ['Heartbeat', 'Watchdog'],
|
|
'PLUGINS': ['heartbeat']
|
|
}
|
|
self.app = create_app(test_config)
|
|
self.client = self.app.test_client()
|
|
|
|
self.origin = str(uuid4()).upper()[:8]
|
|
|
|
self.heartbeat = {
|
|
'origin': self.origin,
|
|
'tags': ['foo', 'bar', 'baz']
|
|
}
|
|
|
|
self.headers = {
|
|
'Content-type': 'application/json'
|
|
}
|
|
|
|
def tearDown(self):
|
|
plugins.plugins.clear()
|
|
db.destroy()
|
|
|
|
def test_heartbeat(self):
|
|
|
|
# create heartbeat
|
|
response = self.client.post('/heartbeat', data=json.dumps(self.heartbeat), headers=self.headers)
|
|
self.assertEqual(response.status_code, 201)
|
|
data = json.loads(response.data.decode('utf-8'))
|
|
self.assertEqual(data['heartbeat']['origin'], self.origin)
|
|
self.assertListEqual(data['heartbeat']['tags'], self.heartbeat['tags'])
|
|
|
|
heartbeat_id = data['id']
|
|
|
|
# create duplicate heartbeat
|
|
response = self.client.post('/heartbeat', data=json.dumps(self.heartbeat), headers=self.headers)
|
|
self.assertEqual(response.status_code, 201)
|
|
data = json.loads(response.data.decode('utf-8'))
|
|
self.assertEqual(heartbeat_id, data['heartbeat']['id'])
|
|
|
|
# get heartbeat
|
|
response = self.client.get('/heartbeat/' + heartbeat_id)
|
|
self.assertEqual(response.status_code, 200)
|
|
data = json.loads(response.data.decode('utf-8'))
|
|
self.assertEqual(heartbeat_id, data['heartbeat']['id'])
|
|
self.assertEqual(data['heartbeat']['status'], 'ok', data)
|
|
|
|
# delete heartbeat
|
|
response = self.client.delete('/heartbeat/' + heartbeat_id)
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
def test_heartbeat_not_found(self):
|
|
|
|
response = self.client.get('/heartbeat/doesnotexist')
|
|
self.assertEqual(response.status_code, 404)
|
|
|
|
def test_get_heartbeats(self):
|
|
|
|
# create heartbeat
|
|
response = self.client.post('/heartbeat', data=json.dumps(self.heartbeat), headers=self.headers)
|
|
self.assertEqual(response.status_code, 201)
|
|
|
|
response = self.client.get('/heartbeats')
|
|
self.assertEqual(response.status_code, 200)
|
|
data = json.loads(response.data.decode('utf-8'))
|
|
self.assertGreater(data['total'], 0, 'total heartbeats > 0')
|
|
|
|
def test_timeout(self):
|
|
|
|
# create heartbeat with default timeout
|
|
response = self.client.post('/heartbeat', data=json.dumps(self.heartbeat), headers=self.headers)
|
|
self.assertEqual(response.status_code, 201)
|
|
data = json.loads(response.data.decode('utf-8'))
|
|
self.assertEqual(data['heartbeat']['timeout'], 4)
|
|
|
|
# resend heartbeat with different timeout
|
|
self.heartbeat['timeout'] = 20
|
|
response = self.client.post('/heartbeat', data=json.dumps(self.heartbeat), headers=self.headers)
|
|
self.assertEqual(response.status_code, 201)
|
|
data = json.loads(response.data.decode('utf-8'))
|
|
self.assertEqual(data['heartbeat']['timeout'], 20)
|
|
|
|
# resend heartbeat with timeout disabled (ie. 0)
|
|
self.heartbeat['timeout'] = 0
|
|
response = self.client.post('/heartbeat', data=json.dumps(self.heartbeat), headers=self.headers)
|
|
self.assertEqual(response.status_code, 201)
|
|
data = json.loads(response.data.decode('utf-8'))
|
|
self.assertEqual(data['heartbeat']['timeout'], 0)
|
|
|
|
# send heartbeat with different timeout
|
|
self.heartbeat['timeout'] = 10
|
|
response = self.client.post('/heartbeat', data=json.dumps(self.heartbeat), headers=self.headers)
|
|
self.assertEqual(response.status_code, 201)
|
|
data = json.loads(response.data.decode('utf-8'))
|
|
self.assertEqual(data['heartbeat']['timeout'], 10)
|
|
|
|
# send heartbeat with very short timeout
|
|
self.heartbeat['timeout'] = 2
|
|
response = self.client.post('/heartbeat', data=json.dumps(self.heartbeat), headers=self.headers)
|
|
self.assertEqual(response.status_code, 201)
|
|
data = json.loads(response.data.decode('utf-8'))
|
|
self.assertEqual(data['heartbeat']['timeout'], 2)
|
|
|
|
heartbeat_id = data['id']
|
|
|
|
time.sleep(4)
|
|
|
|
# get heartbeat
|
|
response = self.client.get('/heartbeat/' + heartbeat_id)
|
|
self.assertEqual(response.status_code, 200)
|
|
data = json.loads(response.data.decode('utf-8'))
|
|
self.assertEqual(heartbeat_id, data['heartbeat']['id'])
|
|
self.assertEqual(data['heartbeat']['timeout'], 2)
|
|
self.assertEqual(data['heartbeat']['status'], 'expired')
|
|
|
|
with self.app.app_context():
|
|
hb = Heartbeat.parse(data['heartbeat'])
|
|
|
|
payload = {
|
|
'resource': hb.origin,
|
|
'event': 'HeartbeatFail',
|
|
'environment': 'Production',
|
|
'severity': 'major',
|
|
'correlate': ['HeartbeatFail', 'HeartbeatSlow', 'HeartbeatOK'],
|
|
'service': ['Test'],
|
|
'group': 'Alerta',
|
|
'value': f'{hb.since}',
|
|
'text': f'Heartbeat not received in {hb.timeout} seconds',
|
|
'tags': hb.tags,
|
|
'attributes': hb.attributes,
|
|
# 'timeout': None
|
|
}
|
|
|
|
# create heartbeat alert
|
|
response = self.client.post('/alert', json=payload)
|
|
self.assertEqual(response.status_code, 201, response.data)
|
|
data = json.loads(response.data.decode('utf-8'))
|
|
self.assertEqual(data['alert']['resource'], hb.origin)
|
|
self.assertEqual(data['alert']['status'], 'open')
|
|
self.assertEqual(data['alert']['service'], ['Test'])
|
|
self.assertEqual(data['alert']['timeout'], 2)
|
|
self.assertEqual(data['alert']['duplicateCount'], 0)
|
|
self.assertEqual(data['alert']['trendIndication'], 'moreSevere')
|
|
self.assertEqual(data['alert']['history'][0]['user'], None)
|
|
|
|
def test_heartbeat_from_alert(self):
|
|
|
|
heartbeat_alert = {
|
|
'event': 'Watchdog',
|
|
'resource': 'net01',
|
|
'environment': 'Production',
|
|
'service': ['Svc1'],
|
|
'origin': 'test/hb',
|
|
'timeout': 500
|
|
}
|
|
|
|
response = self.client.post('/alert', data=json.dumps(heartbeat_alert), headers=self.headers)
|
|
self.assertEqual(response.status_code, 202, response.data)
|
|
data = json.loads(response.data.decode('utf-8'))
|
|
self.assertEqual(data['message'], 'Alert converted to Heartbeat')
|
|
|
|
response = self.client.get('/heartbeats?origin=test/hb')
|
|
self.assertEqual(response.status_code, 200)
|
|
data = json.loads(response.data.decode('utf-8'))
|
|
self.assertEqual(data['heartbeats'][0]['origin'], 'test/hb')
|
|
self.assertEqual(data['heartbeats'][0]['timeout'], 500)
|
|
|
|
prometheus_alert = r"""
|
|
{
|
|
"receiver": "alerta",
|
|
"status": "firing",
|
|
"alerts": [
|
|
{
|
|
"status": "firing",
|
|
"labels": {
|
|
"alertname": "Heartbeat",
|
|
"dc": "eu-west-1",
|
|
"environment": "Development",
|
|
"group": "Healthchecks",
|
|
"monitor": "codelab",
|
|
"region": "EU",
|
|
"service": "Monitoring",
|
|
"severity": "warning",
|
|
"timeout": "8"
|
|
},
|
|
"annotations": {},
|
|
"startsAt": "2020-07-05T18:51:22.012771244Z",
|
|
"endsAt": "0001-01-01T00:00:00Z",
|
|
"generatorURL": "http://9b924908440c:9090/graph?g0.expr=vector%281%29\u0026g0.tab=1",
|
|
"fingerprint": "fc635810a7f52b9d"
|
|
}
|
|
],
|
|
"groupLabels": {
|
|
"alertname": "Heartbeat"
|
|
},
|
|
"commonLabels": {
|
|
"alertname": "Heartbeat",
|
|
"dc": "eu-west-1",
|
|
"environment": "Development",
|
|
"group": "Healthchecks",
|
|
"monitor": "codelab",
|
|
"region": "EU",
|
|
"service": "Monitoring",
|
|
"severity": "warning"
|
|
},
|
|
"commonAnnotations": {},
|
|
"externalURL": "http://ca39969851f4:9093",
|
|
"version": "4",
|
|
"groupKey": "{}:{alertname=\"Heartbeat\"}",
|
|
"truncatedAlerts": 0
|
|
}
|
|
"""
|
|
|
|
response = self.client.post('/webhooks/prometheus', data=prometheus_alert, headers=self.headers)
|
|
self.assertEqual(response.status_code, 202)
|
|
data = json.loads(response.data.decode('utf-8'))
|
|
self.assertEqual(data['message'], 'Alert converted to Heartbeat')
|
|
|
|
response = self.client.get('/heartbeats?origin=prometheus/codelab')
|
|
self.assertEqual(response.status_code, 200)
|
|
data = json.loads(response.data.decode('utf-8'))
|
|
self.assertEqual(sorted(data['heartbeats'][0]['tags']), sorted(['dc=eu-west-1', 'region=EU']))
|
|
self.assertDictEqual(
|
|
data['heartbeats'][0]['attributes'],
|
|
dict(
|
|
environment='Development',
|
|
severity='warning',
|
|
service=['Monitoring'],
|
|
group='Healthchecks'
|
|
)
|
|
)
|
|
self.assertEqual(data['heartbeats'][0]['timeout'], 8)
|