2014-01-26 19:32:48 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
2014-01-05 17:24:04 +00:00
|
|
|
import requests
|
|
|
|
from cabotapp.alert import _send_hipchat_alert
|
|
|
|
from django.utils import timezone
|
2014-02-01 11:31:59 +00:00
|
|
|
from django.core.urlresolvers import reverse
|
2014-01-05 17:24:04 +00:00
|
|
|
from django.test import TestCase
|
2014-02-01 11:31:59 +00:00
|
|
|
from django.core.exceptions import ValidationError
|
2014-01-05 17:24:04 +00:00
|
|
|
from django.contrib.auth.models import User
|
2014-02-01 11:31:59 +00:00
|
|
|
from django.test.client import Client
|
2014-01-28 00:53:34 +00:00
|
|
|
from cabotapp.models import (
|
|
|
|
StatusCheck, GraphiteStatusCheck, JenkinsStatusCheck,
|
2014-01-05 17:24:04 +00:00
|
|
|
HttpStatusCheck, Service, StatusCheckResult)
|
|
|
|
from mock import Mock, patch
|
|
|
|
from twilio import rest
|
|
|
|
from django.core import mail
|
|
|
|
from datetime import timedelta
|
|
|
|
import json
|
|
|
|
import os
|
|
|
|
|
2014-01-28 00:53:34 +00:00
|
|
|
|
2014-01-05 17:24:04 +00:00
|
|
|
def get_content(fname):
|
2014-01-28 00:53:34 +00:00
|
|
|
path = os.path.join(os.path.dirname(__file__), 'fixtures/%s' % fname)
|
|
|
|
with open(path) as f:
|
|
|
|
return f.read()
|
|
|
|
|
2014-01-05 17:24:04 +00:00
|
|
|
|
|
|
|
class LocalTestCase(TestCase):
|
2014-01-28 00:53:34 +00:00
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
requests.get = Mock()
|
|
|
|
requests.post = Mock()
|
|
|
|
rest.TwilioRestClient = Mock()
|
|
|
|
mail.send_mail = Mock()
|
2014-02-01 11:31:59 +00:00
|
|
|
self.create_dummy_data()
|
2014-01-28 00:53:34 +00:00
|
|
|
super(LocalTestCase, self).setUp()
|
|
|
|
|
2014-02-01 11:31:59 +00:00
|
|
|
def create_dummy_data(self):
|
|
|
|
self.username = 'testuser'
|
|
|
|
self.password = 'testuserpassword'
|
|
|
|
self.user = User.objects.create(username=self.username)
|
|
|
|
self.user.set_password(self.password)
|
|
|
|
self.user.save()
|
2014-01-28 00:53:34 +00:00
|
|
|
self.graphite_check = GraphiteStatusCheck.objects.create(
|
|
|
|
name='Graphite Check',
|
|
|
|
metric='stats.fake.value',
|
|
|
|
check_type='>',
|
|
|
|
value='9.0',
|
|
|
|
created_by=self.user,
|
|
|
|
importance=Service.ERROR_STATUS,
|
|
|
|
)
|
|
|
|
self.jenkins_check = JenkinsStatusCheck.objects.create(
|
|
|
|
name='Jenkins Check',
|
|
|
|
created_by=self.user,
|
|
|
|
importance=Service.ERROR_STATUS,
|
|
|
|
max_queued_build_time=10,
|
|
|
|
)
|
|
|
|
self.http_check = HttpStatusCheck.objects.create(
|
|
|
|
name='Http Check',
|
|
|
|
created_by=self.user,
|
|
|
|
importance=Service.CRITICAL_STATUS,
|
|
|
|
endpoint='http://arachnys.com',
|
|
|
|
timeout=10,
|
|
|
|
status_code='200',
|
|
|
|
text_match=None,
|
|
|
|
)
|
|
|
|
self.service = Service.objects.create(
|
|
|
|
name='Service',
|
|
|
|
)
|
|
|
|
self.service.status_checks.add(
|
|
|
|
self.graphite_check, self.jenkins_check, self.http_check)
|
|
|
|
# Passing is most recent
|
|
|
|
self.most_recent_result = StatusCheckResult(
|
|
|
|
check=self.graphite_check,
|
|
|
|
time=timezone.now() - timedelta(seconds=1),
|
|
|
|
time_complete=timezone.now(),
|
|
|
|
succeeded=True
|
|
|
|
)
|
|
|
|
self.most_recent_result.save()
|
|
|
|
# failing is second most recent
|
|
|
|
self.older_result = StatusCheckResult(
|
|
|
|
check=self.graphite_check,
|
|
|
|
time=timezone.now() - timedelta(seconds=60),
|
|
|
|
time_complete=timezone.now() - timedelta(seconds=59),
|
|
|
|
succeeded=False
|
|
|
|
)
|
|
|
|
self.older_result.save()
|
|
|
|
self.graphite_check.save() # Will recalculate status
|
|
|
|
|
2014-02-01 11:31:59 +00:00
|
|
|
|
|
|
|
def fake_graphite_response(*args, **kwargs):
|
|
|
|
resp = Mock()
|
|
|
|
resp.json = json.loads(get_content('graphite_response.json'))
|
|
|
|
resp.status_code = 200
|
|
|
|
return resp
|
|
|
|
|
|
|
|
|
|
|
|
def fake_jenkins_response(*args, **kwargs):
|
|
|
|
resp = Mock()
|
|
|
|
resp.json = json.loads(get_content('jenkins_response.json'))
|
|
|
|
resp.status_code = 200
|
|
|
|
return resp
|
|
|
|
|
|
|
|
|
|
|
|
def jenkins_blocked_response(*args, **kwargs):
|
|
|
|
resp = Mock()
|
|
|
|
resp.json = json.loads(get_content('jenkins_blocked_response.json'))
|
|
|
|
resp.status_code = 200
|
|
|
|
return resp
|
|
|
|
|
|
|
|
|
|
|
|
def fake_http_200_response(*args, **kwargs):
|
|
|
|
resp = Mock()
|
|
|
|
resp.content = get_content('http_response.html')
|
|
|
|
resp.status_code = 200
|
|
|
|
return resp
|
|
|
|
|
|
|
|
|
|
|
|
def fake_http_404_response(*args, **kwargs):
|
|
|
|
resp = Mock()
|
|
|
|
resp.content = get_content('http_response.html')
|
|
|
|
resp.status_code = 404
|
|
|
|
return resp
|
|
|
|
|
|
|
|
|
|
|
|
def throws_timeout(*args, **kwargs):
|
|
|
|
raise requests.RequestException(u'фиктивная ошибка innit')
|
|
|
|
|
|
|
|
|
|
|
|
class TestCheckRun(LocalTestCase):
|
|
|
|
|
|
|
|
|
2014-01-28 00:53:34 +00:00
|
|
|
def test_calculate_service_status(self):
|
|
|
|
self.assertEqual(self.graphite_check.calculated_status,
|
|
|
|
Service.CALCULATED_PASSING_STATUS)
|
|
|
|
self.assertEqual(self.jenkins_check.calculated_status,
|
|
|
|
Service.CALCULATED_PASSING_STATUS)
|
|
|
|
self.assertEqual(self.http_check.calculated_status,
|
|
|
|
Service.CALCULATED_PASSING_STATUS)
|
|
|
|
self.service.update_status()
|
|
|
|
self.assertEqual(self.service.overall_status, Service.PASSING_STATUS)
|
|
|
|
|
|
|
|
# Now two most recent are failing
|
|
|
|
self.most_recent_result.succeeded = False
|
|
|
|
self.most_recent_result.save()
|
|
|
|
self.graphite_check.save()
|
|
|
|
self.assertEqual(self.graphite_check.calculated_status,
|
|
|
|
Service.CALCULATED_FAILING_STATUS)
|
|
|
|
self.service.update_status()
|
|
|
|
self.assertEqual(self.service.overall_status, Service.ERROR_STATUS)
|
|
|
|
|
|
|
|
# Will fail even if second one is working
|
|
|
|
self.older_result.succeeded = True
|
|
|
|
self.older_result.save()
|
|
|
|
self.graphite_check.save()
|
|
|
|
self.assertEqual(self.graphite_check.calculated_status,
|
|
|
|
Service.CALCULATED_FAILING_STATUS)
|
|
|
|
self.service.update_status()
|
|
|
|
self.assertEqual(self.service.overall_status, Service.ERROR_STATUS)
|
|
|
|
|
|
|
|
# Changing debounce will change it up
|
|
|
|
self.graphite_check.debounce = 1
|
|
|
|
self.graphite_check.save()
|
|
|
|
self.assertEqual(self.graphite_check.calculated_status,
|
|
|
|
Service.CALCULATED_PASSING_STATUS)
|
|
|
|
self.service.update_status()
|
|
|
|
self.assertEqual(self.service.overall_status, Service.PASSING_STATUS)
|
|
|
|
|
|
|
|
@patch('cabotapp.graphite.requests.get', fake_graphite_response)
|
|
|
|
def test_graphite_run(self):
|
|
|
|
checkresults = self.graphite_check.statuscheckresult_set.all()
|
|
|
|
self.assertEqual(len(checkresults), 2)
|
|
|
|
self.graphite_check.run()
|
|
|
|
checkresults = self.graphite_check.statuscheckresult_set.all()
|
|
|
|
self.assertEqual(len(checkresults), 3)
|
|
|
|
# Most recent check failed
|
|
|
|
self.assertFalse(self.graphite_check.last_result().succeeded)
|
|
|
|
self.assertEqual(self.graphite_check.calculated_status,
|
|
|
|
Service.CALCULATED_FAILING_STATUS)
|
|
|
|
# This should now pass
|
|
|
|
self.graphite_check.value = '11.0'
|
|
|
|
self.graphite_check.save()
|
|
|
|
checkresults = self.graphite_check.statuscheckresult_set.all()
|
|
|
|
self.assertEqual(len(checkresults), 3)
|
|
|
|
self.graphite_check.run()
|
|
|
|
checkresults = self.graphite_check.statuscheckresult_set.all()
|
|
|
|
self.assertEqual(len(checkresults), 4)
|
|
|
|
self.assertEqual(self.graphite_check.calculated_status,
|
|
|
|
Service.CALCULATED_PASSING_STATUS)
|
|
|
|
|
|
|
|
@patch('cabotapp.jenkins.requests.get', fake_jenkins_response)
|
|
|
|
def test_jenkins_run(self):
|
|
|
|
checkresults = self.jenkins_check.statuscheckresult_set.all()
|
|
|
|
self.assertEqual(len(checkresults), 0)
|
|
|
|
self.jenkins_check.run()
|
|
|
|
checkresults = self.jenkins_check.statuscheckresult_set.all()
|
2014-01-30 17:44:10 +00:00
|
|
|
self.assertEqual(len(checkresults), 1)
|
|
|
|
self.assertFalse(self.jenkins_check.last_result().succeeded)
|
|
|
|
|
|
|
|
@patch('cabotapp.jenkins.requests.get', jenkins_blocked_response)
|
|
|
|
def test_jenkins_blocked_build(self):
|
|
|
|
checkresults = self.jenkins_check.statuscheckresult_set.all()
|
|
|
|
self.assertEqual(len(checkresults), 0)
|
|
|
|
self.jenkins_check.run()
|
|
|
|
checkresults = self.jenkins_check.statuscheckresult_set.all()
|
2014-01-28 00:53:34 +00:00
|
|
|
self.assertEqual(len(checkresults), 1)
|
|
|
|
self.assertFalse(self.jenkins_check.last_result().succeeded)
|
|
|
|
|
|
|
|
@patch('cabotapp.models.requests.get', throws_timeout)
|
|
|
|
def test_timeout_handling_in_jenkins(self):
|
|
|
|
checkresults = self.jenkins_check.statuscheckresult_set.all()
|
|
|
|
self.assertEqual(len(checkresults), 0)
|
|
|
|
self.jenkins_check.run()
|
|
|
|
checkresults = self.jenkins_check.statuscheckresult_set.all()
|
|
|
|
self.assertEqual(len(checkresults), 1)
|
|
|
|
self.assertTrue(self.jenkins_check.last_result().succeeded)
|
|
|
|
self.assertIn(u'Error fetching from Jenkins - фиктивная ошибка',
|
|
|
|
self.jenkins_check.last_result().error)
|
|
|
|
|
|
|
|
@patch('cabotapp.models.requests.get', fake_http_200_response)
|
|
|
|
def test_http_run(self):
|
|
|
|
checkresults = self.http_check.statuscheckresult_set.all()
|
|
|
|
self.assertEqual(len(checkresults), 0)
|
|
|
|
self.http_check.run()
|
|
|
|
checkresults = self.http_check.statuscheckresult_set.all()
|
|
|
|
self.assertEqual(len(checkresults), 1)
|
|
|
|
self.assertTrue(self.http_check.last_result().succeeded)
|
|
|
|
self.assertEqual(self.http_check.calculated_status,
|
|
|
|
Service.CALCULATED_PASSING_STATUS)
|
|
|
|
self.http_check.text_match = u'blah blah'
|
|
|
|
self.http_check.save()
|
|
|
|
self.http_check.run()
|
|
|
|
self.assertFalse(self.http_check.last_result().succeeded)
|
|
|
|
self.assertEqual(self.http_check.calculated_status,
|
|
|
|
Service.CALCULATED_FAILING_STATUS)
|
|
|
|
# Unicode
|
|
|
|
self.http_check.text_match = u'как закалялась сталь'
|
|
|
|
self.http_check.save()
|
|
|
|
self.http_check.run()
|
|
|
|
self.assertFalse(self.http_check.last_result().succeeded)
|
|
|
|
self.assertEqual(self.http_check.calculated_status,
|
|
|
|
Service.CALCULATED_FAILING_STATUS)
|
|
|
|
|
|
|
|
@patch('cabotapp.models.requests.get', throws_timeout)
|
|
|
|
def test_timeout_handling_in_http(self):
|
|
|
|
checkresults = self.http_check.statuscheckresult_set.all()
|
|
|
|
self.assertEqual(len(checkresults), 0)
|
|
|
|
self.http_check.run()
|
|
|
|
checkresults = self.http_check.statuscheckresult_set.all()
|
|
|
|
self.assertEqual(len(checkresults), 1)
|
|
|
|
self.assertFalse(self.http_check.last_result().succeeded)
|
|
|
|
self.assertIn(u'Request error occurred: фиктивная ошибка innit',
|
|
|
|
self.http_check.last_result().error)
|
|
|
|
|
|
|
|
@patch('cabotapp.models.requests.get', fake_http_404_response)
|
|
|
|
def test_http_run_bad_resp(self):
|
|
|
|
checkresults = self.http_check.statuscheckresult_set.all()
|
|
|
|
self.assertEqual(len(checkresults), 0)
|
|
|
|
self.http_check.run()
|
|
|
|
checkresults = self.http_check.statuscheckresult_set.all()
|
|
|
|
self.assertEqual(len(checkresults), 1)
|
|
|
|
self.assertFalse(self.http_check.last_result().succeeded)
|
|
|
|
self.assertEqual(self.http_check.calculated_status,
|
|
|
|
Service.CALCULATED_FAILING_STATUS)
|
2014-02-01 11:31:59 +00:00
|
|
|
|
|
|
|
|
|
|
|
class TestWebInterface(LocalTestCase):
|
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
super(TestWebInterface, self).setUp()
|
|
|
|
self.client = Client()
|
|
|
|
|
|
|
|
def test_set_recovery_instructions(self):
|
|
|
|
# Get service page - will get 200 from login page
|
|
|
|
resp = self.client.get(reverse('update-service', kwargs={'pk':self.service.id}), follow=True)
|
|
|
|
self.assertEqual(resp.status_code, 200)
|
|
|
|
self.assertIn('username', resp.content)
|
|
|
|
|
|
|
|
# Log in
|
|
|
|
self.client.login(username=self.username, password=self.password)
|
|
|
|
resp = self.client.get(reverse('update-service', kwargs={'pk':self.service.id}))
|
|
|
|
self.assertEqual(resp.status_code, 200)
|
|
|
|
self.assertNotIn('username', resp.content)
|
|
|
|
|
|
|
|
snippet_link = 'https://sub.hackpad.com/wiki-7YaNlsC11bB.js'
|
|
|
|
self.assertEqual(self.service.hackpad_id, None)
|
|
|
|
resp = self.client.post(
|
|
|
|
reverse('update-service', kwargs={'pk': self.service.id}),
|
|
|
|
data={
|
|
|
|
'name': self.service.name,
|
|
|
|
'hackpad_id': snippet_link,
|
|
|
|
},
|
|
|
|
follow=True,
|
|
|
|
)
|
|
|
|
self.assertEqual(resp.status_code, 200)
|
|
|
|
reloaded = Service.objects.get(id=self.service.id)
|
|
|
|
self.assertEqual(reloaded.hackpad_id, snippet_link)
|
|
|
|
# Now one on the blacklist
|
|
|
|
blacklist_link = 'https://unapproved_link.domain.com/wiki-7YaNlsC11bB.js'
|
|
|
|
resp = self.client.post(
|
|
|
|
reverse('update-service', kwargs={'pk': self.service.id}),
|
|
|
|
data={
|
|
|
|
'name': self.service.name,
|
|
|
|
'hackpad_id': blacklist_link,
|
|
|
|
},
|
|
|
|
follow=True,
|
|
|
|
)
|
|
|
|
self.assertEqual(resp.status_code, 200)
|
|
|
|
self.assertIn('valid JS snippet link', resp.content)
|
|
|
|
reloaded = Service.objects.get(id=self.service.id)
|
|
|
|
# Still the same
|
|
|
|
self.assertEqual(reloaded.hackpad_id, snippet_link)
|