tests.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382
  1. # -*- coding: utf-8 -*-
  2. from __future__ import absolute_import, print_function
  3. import datetime
  4. import json
  5. import logging
  6. import mock
  7. import zlib
  8. from django.conf import settings
  9. from django.core.urlresolvers import reverse
  10. from django.test.utils import override_settings
  11. from django.utils import timezone
  12. from gzip import GzipFile
  13. from exam import fixture
  14. from raven import Client
  15. from sentry.models import Group, Event
  16. from sentry.testutils import TestCase, TransactionTestCase
  17. from sentry.testutils.helpers import get_auth_header
  18. from sentry.utils.compat import StringIO
  19. from sentry.utils.settings import (
  20. validate_settings, ConfigurationError, import_string)
  21. DEPENDENCY_TEST_DATA = {
  22. "postgresql": ('DATABASES', 'psycopg2.extensions', "database engine", "django.db.backends.postgresql_psycopg2", {
  23. 'default': {
  24. 'ENGINE': "django.db.backends.postgresql_psycopg2",
  25. 'NAME': 'test',
  26. 'USER': 'root',
  27. 'PASSWORD': '',
  28. 'HOST': 'localhost',
  29. 'PORT': ''
  30. }
  31. }),
  32. "mysql": ('DATABASES', 'MySQLdb', "database engine", "django.db.backends.mysql", {
  33. 'default': {
  34. 'ENGINE': "django.db.backends.mysql",
  35. 'NAME': 'test',
  36. 'USER': 'root',
  37. 'PASSWORD': '',
  38. 'HOST': 'localhost',
  39. 'PORT': ''
  40. }
  41. }),
  42. "oracle": ('DATABASES', 'cx_Oracle', "database engine", "django.db.backends.oracle", {
  43. 'default': {
  44. 'ENGINE': "django.db.backends.oracle",
  45. 'NAME': 'test',
  46. 'USER': 'root',
  47. 'PASSWORD': '',
  48. 'HOST': 'localhost',
  49. 'PORT': ''
  50. }
  51. }),
  52. "memcache": ('CACHES', 'memcache', "caching backend", "django.core.cache.backends.memcached.MemcachedCache", {
  53. 'default': {
  54. 'BACKEND': "django.core.cache.backends.memcached.MemcachedCache",
  55. 'LOCATION': '127.0.0.1:11211',
  56. }
  57. }),
  58. "pylibmc": ('CACHES', 'pylibmc', "caching backend", "django.core.cache.backends.memcached.PyLibMCCache", {
  59. 'default': {
  60. 'BACKEND': "django.core.cache.backends.memcached.PyLibMCCache",
  61. 'LOCATION': '127.0.0.1:11211',
  62. }
  63. }),
  64. }
  65. class AssertHandler(logging.Handler):
  66. def emit(self, entry):
  67. raise AssertionError(entry.message)
  68. class RavenIntegrationTest(TransactionTestCase):
  69. """
  70. This mocks the test server and specifically tests behavior that would
  71. happen between Raven <--> Sentry over HTTP communication.
  72. """
  73. def setUp(self):
  74. self.user = self.create_user('coreapi@example.com')
  75. self.project = self.create_project()
  76. self.pm = self.project.team.member_set.get_or_create(user=self.user)[0]
  77. self.pk = self.project.key_set.get_or_create()[0]
  78. self.configure_sentry_errors()
  79. def configure_sentry_errors(self):
  80. assert_handler = AssertHandler()
  81. sentry_errors = logging.getLogger('sentry.errors')
  82. sentry_errors.addHandler(assert_handler)
  83. sentry_errors.setLevel(logging.DEBUG)
  84. def remove_handler():
  85. sentry_errors.handlers.pop(sentry_errors.handlers.index(assert_handler))
  86. self.addCleanup(remove_handler)
  87. def sendRemote(self, url, data, headers={}):
  88. content_type = headers.pop('Content-Type', None)
  89. headers = dict(('HTTP_' + k.replace('-', '_').upper(), v) for k, v in headers.iteritems())
  90. resp = self.client.post(
  91. reverse('sentry-api-store', args=[self.pk.project_id]),
  92. data=data,
  93. content_type=content_type,
  94. **headers)
  95. assert resp.status_code == 200, resp.content
  96. @mock.patch('raven.base.Client.send_remote')
  97. def test_basic(self, send_remote):
  98. send_remote.side_effect = self.sendRemote
  99. client = Client(
  100. dsn='http://%s:%s@localhost:8000/%s' % (
  101. self.pk.public_key, self.pk.secret_key, self.pk.project_id)
  102. )
  103. with self.tasks():
  104. client.capture('Message', message='foo')
  105. send_remote.assert_called_once()
  106. assert Group.objects.count() == 1
  107. group = Group.objects.get()
  108. assert group.event_set.count() == 1
  109. instance = group.event_set.get()
  110. assert instance.message == 'foo'
  111. class SentryRemoteTest(TestCase):
  112. @fixture
  113. def path(self):
  114. return reverse('sentry-api-store')
  115. def test_minimal(self):
  116. kwargs = {'message': 'hello'}
  117. resp = self._postWithHeader(kwargs)
  118. assert resp.status_code == 200, resp.content
  119. event_id = json.loads(resp.content)['id']
  120. instance = Event.objects.get(event_id=event_id)
  121. assert instance.message == 'hello'
  122. def test_timestamp(self):
  123. timestamp = timezone.now().replace(microsecond=0, tzinfo=timezone.utc) - datetime.timedelta(hours=1)
  124. kwargs = {u'message': 'hello', 'timestamp': timestamp.strftime('%s.%f')}
  125. resp = self._postWithSignature(kwargs)
  126. assert resp.status_code == 200, resp.content
  127. instance = Event.objects.get()
  128. assert instance.message == 'hello'
  129. assert instance.datetime == timestamp
  130. group = instance.group
  131. assert group.first_seen == timestamp
  132. assert group.last_seen == timestamp
  133. def test_timestamp_as_iso(self):
  134. timestamp = timezone.now().replace(microsecond=0, tzinfo=timezone.utc) - datetime.timedelta(hours=1)
  135. kwargs = {u'message': 'hello', 'timestamp': timestamp.strftime('%Y-%m-%dT%H:%M:%S.%f')}
  136. resp = self._postWithSignature(kwargs)
  137. assert resp.status_code == 200, resp.content
  138. instance = Event.objects.get()
  139. assert instance.message == 'hello'
  140. assert instance.datetime == timestamp
  141. group = instance.group
  142. assert group.first_seen == timestamp
  143. assert group.last_seen == timestamp
  144. def test_ungzipped_data(self):
  145. kwargs = {'message': 'hello'}
  146. resp = self._postWithSignature(kwargs)
  147. assert resp.status_code == 200
  148. instance = Event.objects.get()
  149. assert instance.message == 'hello'
  150. @override_settings(SENTRY_ALLOW_ORIGIN='getsentry.com')
  151. def test_correct_data_with_get(self):
  152. kwargs = {'message': 'hello'}
  153. resp = self._getWithReferer(kwargs)
  154. assert resp.status_code == 200, resp.content
  155. instance = Event.objects.get()
  156. assert instance.message == 'hello'
  157. @override_settings(SENTRY_ALLOW_ORIGIN='getsentry.com')
  158. def test_get_without_referer(self):
  159. self.project.update_option('sentry:origins', '')
  160. kwargs = {'message': 'hello'}
  161. resp = self._getWithReferer(kwargs, referer=None, protocol='4')
  162. assert resp.status_code == 403, (resp.status_code, resp.get('X-Sentry-Error'))
  163. @override_settings(SENTRY_ALLOW_ORIGIN='*')
  164. def test_get_without_referer_allowed(self):
  165. self.project.update_option('sentry:origins', '')
  166. kwargs = {'message': 'hello'}
  167. resp = self._getWithReferer(kwargs, referer=None, protocol='4')
  168. assert resp.status_code == 200, (resp.status_code, resp.get('X-Sentry-Error'))
  169. def test_signature(self):
  170. kwargs = {'message': 'hello'}
  171. resp = self._postWithSignature(kwargs)
  172. assert resp.status_code == 200, resp.content
  173. instance = Event.objects.get()
  174. assert instance.message == 'hello'
  175. def test_content_encoding_deflate(self):
  176. kwargs = {'message': 'hello'}
  177. message = zlib.compress(json.dumps(kwargs))
  178. key = self.projectkey.public_key
  179. secret = self.projectkey.secret_key
  180. with self.tasks():
  181. resp = self.client.post(
  182. self.path, message,
  183. content_type='application/octet-stream',
  184. HTTP_CONTENT_ENCODING='deflate',
  185. HTTP_X_SENTRY_AUTH=get_auth_header('_postWithHeader', key, secret),
  186. )
  187. assert resp.status_code == 200, resp.content
  188. event_id = json.loads(resp.content)['id']
  189. instance = Event.objects.get(event_id=event_id)
  190. assert instance.message == 'hello'
  191. def test_content_encoding_gzip(self):
  192. kwargs = {'message': 'hello'}
  193. message = json.dumps(kwargs)
  194. fp = StringIO()
  195. try:
  196. f = GzipFile(fileobj=fp, mode='w')
  197. f.write(message)
  198. finally:
  199. f.close()
  200. key = self.projectkey.public_key
  201. secret = self.projectkey.secret_key
  202. with self.tasks():
  203. resp = self.client.post(
  204. self.path, fp.getvalue(),
  205. content_type='application/octet-stream',
  206. HTTP_CONTENT_ENCODING='gzip',
  207. HTTP_X_SENTRY_AUTH=get_auth_header('_postWithHeader', key, secret),
  208. )
  209. assert resp.status_code == 200, resp.content
  210. event_id = json.loads(resp.content)['id']
  211. instance = Event.objects.get(event_id=event_id)
  212. assert instance.message == 'hello'
  213. def test_protocol_v2_0_without_secret_key(self):
  214. kwargs = {'message': 'hello'}
  215. resp = self._postWithHeader(
  216. data=kwargs,
  217. key=self.projectkey.public_key,
  218. protocol='2.0',
  219. )
  220. assert resp.status_code == 200, resp.content
  221. event_id = json.loads(resp.content)['id']
  222. instance = Event.objects.get(event_id=event_id)
  223. assert instance.message == 'hello'
  224. def test_protocol_v3(self):
  225. kwargs = {'message': 'hello'}
  226. resp = self._postWithHeader(
  227. data=kwargs,
  228. key=self.projectkey.public_key,
  229. secret=self.projectkey.secret_key,
  230. protocol='3',
  231. )
  232. assert resp.status_code == 200, resp.content
  233. event_id = json.loads(resp.content)['id']
  234. instance = Event.objects.get(event_id=event_id)
  235. assert instance.message == 'hello'
  236. def test_protocol_v4(self):
  237. kwargs = {'message': 'hello'}
  238. resp = self._postWithHeader(
  239. data=kwargs,
  240. key=self.projectkey.public_key,
  241. secret=self.projectkey.secret_key,
  242. protocol='4',
  243. )
  244. assert resp.status_code == 200, resp.content
  245. event_id = json.loads(resp.content)['id']
  246. instance = Event.objects.get(event_id=event_id)
  247. assert instance.message == 'hello'
  248. def test_protocol_v5(self):
  249. kwargs = {'message': 'hello'}
  250. resp = self._postWithHeader(
  251. data=kwargs,
  252. key=self.projectkey.public_key,
  253. secret=self.projectkey.secret_key,
  254. protocol='5',
  255. )
  256. assert resp.status_code == 200, resp.content
  257. event_id = json.loads(resp.content)['id']
  258. instance = Event.objects.get(event_id=event_id)
  259. assert instance.message == 'hello'
  260. def test_protocol_v6(self):
  261. kwargs = {'message': 'hello'}
  262. resp = self._postWithHeader(
  263. data=kwargs,
  264. key=self.projectkey.public_key,
  265. secret=self.projectkey.secret_key,
  266. protocol='6',
  267. )
  268. assert resp.status_code == 200, resp.content
  269. event_id = json.loads(resp.content)['id']
  270. instance = Event.objects.get(event_id=event_id)
  271. assert instance.message == 'hello'
  272. class DepdendencyTest(TestCase):
  273. def raise_import_error(self, package):
  274. def callable(package_name):
  275. if package_name != package:
  276. return import_string(package_name)
  277. raise ImportError("No module named %s" % (package,))
  278. return callable
  279. @mock.patch('django.conf.settings', mock.Mock())
  280. @mock.patch('sentry.utils.settings.import_string')
  281. def validate_dependency(self, key, package, dependency_type, dependency,
  282. setting_value, import_string):
  283. import_string.side_effect = self.raise_import_error(package)
  284. with self.settings(**{key: setting_value}):
  285. with self.assertRaises(ConfigurationError):
  286. validate_settings(settings)
  287. def test_validate_fails_on_postgres(self):
  288. self.validate_dependency(*DEPENDENCY_TEST_DATA['postgresql'])
  289. def test_validate_fails_on_mysql(self):
  290. self.validate_dependency(*DEPENDENCY_TEST_DATA['mysql'])
  291. def test_validate_fails_on_oracle(self):
  292. self.validate_dependency(*DEPENDENCY_TEST_DATA['oracle'])
  293. def test_validate_fails_on_memcache(self):
  294. self.validate_dependency(*DEPENDENCY_TEST_DATA['memcache'])
  295. def test_validate_fails_on_pylibmc(self):
  296. self.validate_dependency(*DEPENDENCY_TEST_DATA['pylibmc'])