debian-python-raven/tests/contrib/django/tests.py

802 lines
30 KiB
Python

# -*- coding: utf-8 -*-
from __future__ import absolute_import
from __future__ import with_statement
from __future__ import unicode_literals
import datetime
import django
import logging
import mock
import pytest
import re
import sys # NOQA
from exam import fixture
from django.conf import settings
from django.core.exceptions import SuspiciousOperation
from django.core.urlresolvers import reverse
from django.core.signals import got_request_exception
from django.core.handlers.wsgi import WSGIRequest
from django.http import QueryDict
from django.template import TemplateSyntaxError
from django.test import TestCase
from raven.base import Client
from raven.contrib.django.client import DjangoClient
from raven.contrib.django.celery import CeleryClient
from raven.contrib.django.handlers import SentryHandler
from raven.contrib.django.models import client, get_client, sentry_exception_handler
from raven.contrib.django.middleware.wsgi import Sentry
from raven.contrib.django.templatetags.raven import sentry_public_dsn
from raven.contrib.django.views import is_valid_origin
from raven.utils.serializer import transform
from raven.utils import six
from raven.utils.six import StringIO
from django.test.client import Client as TestClient, ClientHandler as TestClientHandler
from .models import TestModel
settings.SENTRY_CLIENT = 'tests.contrib.django.tests.TempStoreClient'
DJANGO_15 = django.VERSION >= (1, 5, 0)
def make_request():
return WSGIRequest(environ={
'wsgi.input': StringIO(),
'REQUEST_METHOD': 'POST',
'SERVER_NAME': 'testserver',
'SERVER_PORT': '80',
'CONTENT_TYPE': 'text/html',
'ACCEPT': 'text/html',
})
class MockClientHandler(TestClientHandler):
def __call__(self, environ, start_response=[]):
# this pretends doesn't require start_response
return super(MockClientHandler, self).__call__(environ)
class MockSentryMiddleware(Sentry):
def __call__(self, environ, start_response=[]):
# this pretends doesn't require start_response
return list(super(MockSentryMiddleware, self).__call__(environ, start_response))
class TempStoreClient(DjangoClient):
def __init__(self, *args, **kwargs):
self.events = []
super(TempStoreClient, self).__init__(*args, **kwargs)
def send(self, **kwargs):
self.events.append(kwargs)
def is_enabled(self, **kwargs):
return True
class DisabledTempStoreClient(TempStoreClient):
def is_enabled(self, **kwargs):
return False
class Settings(object):
"""
Allows you to define settings that are required for this function to work.
>>> with Settings(SENTRY_LOGIN_URL='foo'): #doctest: +SKIP
>>> print settings.SENTRY_LOGIN_URL #doctest: +SKIP
"""
NotDefined = object()
def __init__(self, **overrides):
self.overrides = overrides
self._orig = {}
def __enter__(self):
for k, v in six.iteritems(self.overrides):
self._orig[k] = getattr(settings, k, self.NotDefined)
setattr(settings, k, v)
def __exit__(self, exc_type, exc_value, traceback):
for k, v in six.iteritems(self._orig):
if v is self.NotDefined:
delattr(settings, k)
else:
setattr(settings, k, v)
class ClientProxyTest(TestCase):
def test_proxy_responds_as_client(self):
self.assertEquals(get_client(), client)
@mock.patch.object(TempStoreClient, 'captureMessage')
def test_basic(self, captureMessage):
client.captureMessage(message='foo')
captureMessage.assert_called_once_with(message='foo')
class DjangoClientTest(TestCase):
## Fixture setup/teardown
urls = 'tests.contrib.django.urls'
def setUp(self):
self.raven = get_client()
def test_basic(self):
self.raven.captureMessage(message='foo')
self.assertEquals(len(self.raven.events), 1)
event = self.raven.events.pop(0)
self.assertTrue('sentry.interfaces.Message' in event)
message = event['sentry.interfaces.Message']
self.assertEquals(message['message'], 'foo')
self.assertEquals(event['level'], logging.ERROR)
self.assertEquals(event['message'], 'foo')
self.assertEquals(type(event['timestamp']), datetime.datetime)
def test_signal_integration(self):
try:
int('hello')
except:
got_request_exception.send(sender=self.__class__, request=None)
else:
self.fail('Expected an exception.')
self.assertEquals(len(self.raven.events), 1)
event = self.raven.events.pop(0)
assert 'exception' in event
exc = event['exception']['values'][0]
self.assertEquals(exc['type'], 'ValueError')
self.assertEquals(exc['value'], "invalid literal for int() with base 10: 'hello'")
self.assertEquals(event['level'], logging.ERROR)
self.assertEquals(event['message'], "ValueError: invalid literal for int() with base 10: 'hello'")
self.assertEquals(event['culprit'], 'tests.contrib.django.tests in test_signal_integration')
def test_view_exception(self):
self.assertRaises(Exception, self.client.get, reverse('sentry-raise-exc'))
self.assertEquals(len(self.raven.events), 1)
event = self.raven.events.pop(0)
assert 'exception' in event
exc = event['exception']['values'][0]
self.assertEquals(exc['type'], 'Exception')
self.assertEquals(exc['value'], 'view exception')
self.assertEquals(event['level'], logging.ERROR)
self.assertEquals(event['message'], 'Exception: view exception')
self.assertEquals(event['culprit'], 'tests.contrib.django.views in raise_exc')
def test_user_info(self):
from django.contrib.auth.models import User
user = User(username='admin', email='admin@example.com')
user.set_password('admin')
user.save()
self.assertRaises(Exception, self.client.get, reverse('sentry-raise-exc'))
assert len(self.raven.events) == 1
event = self.raven.events.pop(0)
assert 'user' not in event
assert self.client.login(username='admin', password='admin')
self.assertRaises(Exception, self.client.get, reverse('sentry-raise-exc'))
self.assertEquals(len(self.raven.events), 1)
event = self.raven.events.pop(0)
assert 'user' in event
user_info = event['user']
assert user_info == {
'is_authenticated': True,
'username': user.username,
'id': user.id,
'email': user.email,
}
@pytest.mark.skipif(str('not DJANGO_15'))
def test_get_user_info_abstract_user(self):
from django.db import models
from django.contrib.auth.models import AbstractBaseUser
class MyUser(AbstractBaseUser):
USERNAME_FIELD = 'username'
username = models.CharField(max_length=32)
email = models.EmailField()
user = MyUser(
username='admin',
email='admin@example.com',
id=1,
)
user_info = self.raven.get_user_info(user)
assert user_info == {
'is_authenticated': True,
'username': user.username,
'id': user.id,
'email': user.email,
}
def test_request_middleware_exception(self):
with Settings(MIDDLEWARE_CLASSES=['tests.contrib.django.middleware.BrokenRequestMiddleware']):
self.assertRaises(ImportError, self.client.get, reverse('sentry-raise-exc'))
self.assertEquals(len(self.raven.events), 1)
event = self.raven.events.pop(0)
assert 'exception' in event
exc = event['exception']['values'][0]
self.assertEquals(exc['type'], 'ImportError')
self.assertEquals(exc['value'], 'request')
self.assertEquals(event['level'], logging.ERROR)
self.assertEquals(event['message'], 'ImportError: request')
self.assertEquals(event['culprit'], 'tests.contrib.django.middleware in process_request')
def test_response_middlware_exception(self):
if django.VERSION[:2] < (1, 3):
return
with Settings(MIDDLEWARE_CLASSES=['tests.contrib.django.middleware.BrokenResponseMiddleware']):
self.assertRaises(ImportError, self.client.get, reverse('sentry-no-error'))
self.assertEquals(len(self.raven.events), 1)
event = self.raven.events.pop(0)
assert 'exception' in event
exc = event['exception']['values'][0]
self.assertEquals(exc['type'], 'ImportError')
self.assertEquals(exc['value'], 'response')
self.assertEquals(event['level'], logging.ERROR)
self.assertEquals(event['message'], 'ImportError: response')
self.assertEquals(event['culprit'], 'tests.contrib.django.middleware in process_response')
def test_broken_500_handler_with_middleware(self):
with Settings(BREAK_THAT_500=True, INSTALLED_APPS=['raven.contrib.django']):
client = TestClient(REMOTE_ADDR='127.0.0.1')
client.handler = MockSentryMiddleware(MockClientHandler())
self.assertRaises(Exception, client.get, reverse('sentry-raise-exc'))
assert len(self.raven.events) == 2
event = self.raven.events.pop(0)
assert 'exception' in event
exc = event['exception']['values'][0]
self.assertEquals(exc['type'], 'Exception')
self.assertEquals(exc['value'], 'view exception')
self.assertEquals(event['level'], logging.ERROR)
self.assertEquals(event['message'], 'Exception: view exception')
self.assertEquals(event['culprit'], 'tests.contrib.django.views in raise_exc')
event = self.raven.events.pop(0)
assert 'exception' in event
exc = event['exception']['values'][0]
self.assertEquals(exc['type'], 'ValueError')
self.assertEquals(exc['value'], 'handler500')
self.assertEquals(event['level'], logging.ERROR)
self.assertEquals(event['message'], 'ValueError: handler500')
self.assertEquals(event['culprit'], 'tests.contrib.django.urls in handler500')
def test_view_middleware_exception(self):
with Settings(MIDDLEWARE_CLASSES=['tests.contrib.django.middleware.BrokenViewMiddleware']):
self.assertRaises(ImportError, self.client.get, reverse('sentry-raise-exc'))
self.assertEquals(len(self.raven.events), 1)
event = self.raven.events.pop(0)
assert 'exception' in event
exc = event['exception']['values'][0]
self.assertEquals(exc['type'], 'ImportError')
self.assertEquals(exc['value'], 'view')
self.assertEquals(event['level'], logging.ERROR)
self.assertEquals(event['message'], 'ImportError: view')
self.assertEquals(event['culprit'], 'tests.contrib.django.middleware in process_view')
def test_exclude_modules_view(self):
exclude_paths = self.raven.exclude_paths
self.raven.exclude_paths = ['tests.views']
self.assertRaises(Exception, self.client.get, reverse('sentry-raise-exc-decor'))
self.assertEquals(len(self.raven.events), 1)
event = self.raven.events.pop(0)
self.assertEquals(event['culprit'], 'tests.contrib.django.views in raise_exc')
self.raven.exclude_paths = exclude_paths
def test_include_modules(self):
include_paths = self.raven.include_paths
self.raven.include_paths = ['django.shortcuts']
self.assertRaises(Exception, self.client.get, reverse('sentry-django-exc'))
self.assertEquals(len(self.raven.events), 1)
event = self.raven.events.pop(0)
assert event['culprit'].startswith('django.shortcuts in ')
self.raven.include_paths = include_paths
def test_template_name_as_view(self):
self.assertRaises(TemplateSyntaxError, self.client.get, reverse('sentry-template-exc'))
self.assertEquals(len(self.raven.events), 1)
event = self.raven.events.pop(0)
self.assertEquals(event['culprit'], 'error.html')
# def test_request_in_logging(self):
# resp = self.client.get(reverse('sentry-log-request-exc'))
# self.assertEquals(resp.status_code, 200)
# self.assertEquals(len(self.raven.events), 1)
# event = self.raven.events.pop(0)
# self.assertEquals(event['culprit'], 'tests.contrib.django.views in logging_request_exc')
# self.assertEquals(event['data']['META']['REMOTE_ADDR'], '127.0.0.1')
# TODO: Python bug #10805
@pytest.mark.skipif(str('six.PY3'))
def test_record_none_exc_info(self):
# sys.exc_info can return (None, None, None) if no exception is being
# handled anywhere on the stack. See:
# http://docs.python.org/library/sys.html#sys.exc_info
record = logging.LogRecord(
'foo',
logging.INFO,
pathname=None,
lineno=None,
msg='test',
args=(),
exc_info=(None, None, None),
)
handler = SentryHandler()
handler.emit(record)
self.assertEquals(len(self.raven.events), 1)
event = self.raven.events.pop(0)
self.assertEquals(event['message'], 'test')
def test_404_middleware(self):
with Settings(MIDDLEWARE_CLASSES=['raven.contrib.django.middleware.Sentry404CatchMiddleware']):
resp = self.client.get('/non-existent-page')
self.assertEquals(resp.status_code, 404)
self.assertEquals(len(self.raven.events), 1)
event = self.raven.events.pop(0)
self.assertEquals(event['level'], logging.INFO)
self.assertEquals(event['logger'], 'http404')
assert 'request' in event
http = event['request']
self.assertEquals(http['url'], 'http://testserver/non-existent-page')
self.assertEquals(http['method'], 'GET')
self.assertEquals(http['query_string'], '')
self.assertEquals(http['data'], None)
def test_404_middleware_when_disabled(self):
extra_settings = {
'MIDDLEWARE_CLASSES': ['raven.contrib.django.middleware.Sentry404CatchMiddleware'],
'SENTRY_CLIENT': 'tests.contrib.django.tests.DisabledTempStoreClient',
}
with Settings(**extra_settings):
resp = self.client.get('/non-existent-page')
assert resp.status_code == 404
assert self.raven.events == []
def test_invalid_client(self):
extra_settings = {
'SENTRY_CLIENT': 'raven.contrib.django.DjangoClient', # default
}
# Should return fallback client (TempStoreClient)
client = get_client('nonexistent.and.invalid')
# client should be valid, and the same as with the next call.
assert client is get_client()
with Settings(**extra_settings):
assert isinstance(get_client(), DjangoClient)
def test_response_error_id_middleware(self):
# TODO: test with 500s
with Settings(MIDDLEWARE_CLASSES=['raven.contrib.django.middleware.SentryResponseErrorIdMiddleware',
'raven.contrib.django.middleware.Sentry404CatchMiddleware']):
resp = self.client.get('/non-existent-page')
self.assertEquals(resp.status_code, 404)
headers = dict(resp.items())
self.assertTrue('X-Sentry-ID' in headers)
self.assertEquals(len(self.raven.events), 1)
event = self.raven.events.pop(0)
assert event['event_id'] == headers['X-Sentry-ID']
def test_get_client(self):
self.assertEquals(get_client(), get_client())
self.assertEquals(get_client('raven.base.Client').__class__, Client)
self.assertEquals(get_client(), self.raven)
self.assertEquals(get_client('%s.%s' % (self.raven.__class__.__module__, self.raven.__class__.__name__)),
self.raven)
self.assertEquals(get_client(), self.raven)
def test_raw_post_data_partial_read(self):
v = '{"foo": "bar"}'
request = make_request()
request.environ.update({
'wsgi.input': StringIO(v + '\r\n\r\n'),
'CONTENT_TYPE': 'application/octet-stream',
'CONTENT_LENGTH': len(v),
})
request.read(1)
self.raven.captureMessage(message='foo', request=request)
self.assertEquals(len(self.raven.events), 1)
event = self.raven.events.pop(0)
assert 'request' in event
http = event['request']
self.assertEquals(http['method'], 'POST')
self.assertEquals(http['data'], '<unavailable>')
def test_read_post_data(self):
request = make_request()
request.POST = QueryDict("foo=bar&ham=spam")
request.read(1)
self.raven.captureMessage(message='foo', request=request)
self.assertEquals(len(self.raven.events), 1)
event = self.raven.events.pop(0)
assert 'request' in event
http = event['request']
self.assertEquals(http['method'], 'POST')
self.assertEquals(http['data'], {'foo': 'bar', 'ham': 'spam'})
def test_request_capture(self):
request = make_request()
request.read(1)
self.raven.captureMessage(message='foo', request=request)
self.assertEquals(len(self.raven.events), 1)
event = self.raven.events.pop(0)
assert 'request' in event
http = event['request']
self.assertEquals(http['method'], 'POST')
self.assertEquals(http['data'], '<unavailable>')
self.assertTrue('headers' in http)
headers = http['headers']
self.assertTrue('Content-Type' in headers, headers.keys())
self.assertEquals(headers['Content-Type'], 'text/html')
env = http['env']
self.assertTrue('SERVER_NAME' in env, env.keys())
self.assertEquals(env['SERVER_NAME'], 'testserver')
self.assertTrue('SERVER_PORT' in env, env.keys())
self.assertEquals(env['SERVER_PORT'], '80')
def test_marks_django_frames_correctly(self):
self.assertRaises(TemplateSyntaxError, self.client.get, reverse('sentry-template-exc'))
self.assertEquals(len(self.raven.events), 1)
event = self.raven.events.pop(0)
frames = event['exception']['values'][0]['stacktrace']['frames']
for frame in frames:
if frame['module'].startswith('django.'):
assert frame.get('in_app') is False
def test_adds_site_to_tags(self):
self.assertRaises(TemplateSyntaxError, self.client.get, reverse('sentry-template-exc'))
self.assertEquals(len(self.raven.events), 1)
event = self.raven.events.pop(0)
tags = event['tags']
assert 'site' in event['tags']
assert tags['site'] == 'example.com'
def test_adds_site_to_tags_fallback(self):
with Settings(SITE_ID=12345): # nonexistent site, should fallback to SITE_ID
self.assertRaises(TemplateSyntaxError, self.client.get, reverse('sentry-template-exc'))
self.assertEquals(len(self.raven.events), 1)
event = self.raven.events.pop(0)
tags = event['tags']
assert 'site' in event['tags']
assert tags['site'] == 12345
def test_settings_site_overrides_contrib(self):
self.raven.site = 'FOO'
self.assertRaises(TemplateSyntaxError, self.client.get, reverse('sentry-template-exc'))
self.assertEquals(len(self.raven.events), 1)
event = self.raven.events.pop(0)
tags = event['tags']
assert 'site' in event['tags']
assert tags['site'] == 'FOO'
@mock.patch.object(WSGIRequest, 'build_absolute_uri')
def test_suspicious_operation_in_build_absolute_uri(self, build_absolute_uri):
build_absolute_uri.side_effect = SuspiciousOperation()
request = make_request()
request.META['HTTP_HOST'] = 'example.com'
result = self.raven.get_data_from_request(request)
build_absolute_uri.assert_called_once_with()
assert 'request' in result
assert result['request']['url'] == 'http://example.com/'
class DjangoTemplateTagTest(TestCase):
@mock.patch('raven.contrib.django.DjangoClient.get_public_dsn')
def test_sentry_public_dsn_no_args(self, get_public_dsn):
sentry_public_dsn()
get_public_dsn.assert_called_once_with(None)
@mock.patch('raven.contrib.django.DjangoClient.get_public_dsn')
def test_sentry_public_dsn_with_https(self, get_public_dsn):
sentry_public_dsn('https')
get_public_dsn.assert_called_once_with('https')
class DjangoLoggingTest(TestCase):
def setUp(self):
self.logger = logging.getLogger(__name__)
self.raven = get_client()
def test_request_kwarg(self):
handler = SentryHandler()
logger = self.logger
logger.handlers = []
logger.addHandler(handler)
logger.error('This is a test error', extra={
'request': WSGIRequest(environ={
'wsgi.input': StringIO(),
'REQUEST_METHOD': 'POST',
'SERVER_NAME': 'testserver',
'SERVER_PORT': '80',
'CONTENT_TYPE': 'application/octet-stream',
'ACCEPT': 'application/json',
})
})
self.assertEquals(len(self.raven.events), 1)
event = self.raven.events.pop(0)
assert 'request' in event
http = event['request']
self.assertEquals(http['method'], 'POST')
class CeleryIsolatedClientTest(TestCase):
def setUp(self):
self.client = CeleryClient(
servers=['http://example.com'],
public_key='public',
secret_key='secret',
)
@mock.patch('raven.contrib.django.celery.send_raw')
def test_send_encoded(self, send_raw):
self.client.send_encoded('foo')
send_raw.delay.assert_called_once_with('foo')
@mock.patch('raven.contrib.django.celery.send_raw')
def test_without_eager(self, send_raw):
"""
Integration test to ensure it propagates all the way down
and calls delay on the task.
"""
self.client.captureMessage(message='test')
self.assertEquals(send_raw.delay.call_count, 1)
class CeleryIntegratedClientTest(TestCase):
def setUp(self):
self.client = CeleryClient()
@mock.patch('raven.contrib.django.celery.send_raw_integrated')
def test_send_encoded(self, send_raw):
with Settings(INSTALLED_APPS=tuple(settings.INSTALLED_APPS) + ('sentry',)):
self.client.send_integrated('foo')
send_raw.delay.assert_called_once_with('foo')
@mock.patch('raven.contrib.django.celery.send_raw_integrated')
def test_without_eager(self, send_raw):
"""
Integration test to ensure it propagates all the way down
and calls delay on the task.
"""
with Settings(INSTALLED_APPS=tuple(settings.INSTALLED_APPS) + ('sentry',)):
self.client.captureMessage(message='test')
self.assertEquals(send_raw.delay.call_count, 1)
class IsValidOriginTestCase(TestCase):
def test_setting_empty(self):
with Settings(SENTRY_ALLOW_ORIGIN=None):
self.assertFalse(is_valid_origin('http://example.com'))
def test_setting_all(self):
with Settings(SENTRY_ALLOW_ORIGIN='*'):
self.assertTrue(is_valid_origin('http://example.com'))
def test_setting_uri(self):
with Settings(SENTRY_ALLOW_ORIGIN=['http://example.com']):
self.assertTrue(is_valid_origin('http://example.com'))
def test_setting_regexp(self):
with Settings(SENTRY_ALLOW_ORIGIN=[re.compile('https?\://(.*\.)?example\.com')]):
self.assertTrue(is_valid_origin('http://example.com'))
class ReportViewTest(TestCase):
urls = 'raven.contrib.django.urls'
def setUp(self):
self.path = reverse('raven-report')
@mock.patch('raven.contrib.django.views.is_valid_origin')
def test_calls_is_valid_origin_with_header(self, is_valid_origin):
self.client.post(self.path, HTTP_ORIGIN='http://example.com')
is_valid_origin.assert_called_once_with('http://example.com')
@mock.patch('raven.contrib.django.views.is_valid_origin')
def test_calls_is_valid_origin_with_header_as_get(self, is_valid_origin):
self.client.get(self.path, HTTP_ORIGIN='http://example.com')
is_valid_origin.assert_called_once_with('http://example.com')
@mock.patch('raven.contrib.django.views.is_valid_origin', mock.Mock(return_value=False))
def test_fails_on_invalid_origin(self):
resp = self.client.post(self.path, HTTP_ORIGIN='http://example.com')
self.assertEquals(resp.status_code, 403)
@mock.patch('raven.contrib.django.views.is_valid_origin', mock.Mock(return_value=True))
def test_options_call_sends_headers(self):
resp = self.client.options(self.path, HTTP_ORIGIN='http://example.com')
self.assertEquals(resp.status_code, 200)
self.assertEquals(resp['Access-Control-Allow-Origin'], 'http://example.com')
self.assertEquals(resp['Access-Control-Allow-Methods'], 'GET, POST, OPTIONS')
@mock.patch('raven.contrib.django.views.is_valid_origin', mock.Mock(return_value=True))
def test_missing_data(self):
resp = self.client.post(self.path, HTTP_ORIGIN='http://example.com')
self.assertEquals(resp.status_code, 400)
@mock.patch('raven.contrib.django.views.is_valid_origin', mock.Mock(return_value=True))
def test_invalid_data(self):
resp = self.client.post(self.path, HTTP_ORIGIN='http://example.com',
data='[1', content_type='application/octet-stream')
self.assertEquals(resp.status_code, 400)
@mock.patch('raven.contrib.django.views.is_valid_origin', mock.Mock(return_value=True))
def test_sends_data(self):
resp = self.client.post(self.path, HTTP_ORIGIN='http://example.com',
data='{}', content_type='application/octet-stream')
self.assertEquals(resp.status_code, 200)
event = client.events.pop(0)
self.assertEquals(event, {'auth_header': None})
@mock.patch('raven.contrib.django.views.is_valid_origin', mock.Mock(return_value=True))
def test_sends_authorization_header(self):
resp = self.client.post(self.path, HTTP_ORIGIN='http://example.com',
HTTP_AUTHORIZATION='Sentry foo/bar', data='{}', content_type='application/octet-stream')
self.assertEquals(resp.status_code, 200)
event = client.events.pop(0)
self.assertEquals(event, {'auth_header': 'Sentry foo/bar'})
@mock.patch('raven.contrib.django.views.is_valid_origin', mock.Mock(return_value=True))
def test_sends_x_sentry_auth_header(self):
resp = self.client.post(self.path, HTTP_ORIGIN='http://example.com',
HTTP_X_SENTRY_AUTH='Sentry foo/bar', data='{}',
content_type='application/octet-stream')
self.assertEquals(resp.status_code, 200)
event = client.events.pop(0)
self.assertEquals(event, {'auth_header': 'Sentry foo/bar'})
class PromiseSerializerTestCase(TestCase):
def test_basic(self):
from django.utils.functional import lazy
obj = lazy(lambda: 'bar', six.text_type)()
res = transform(obj)
expected = "'bar'" if six.PY3 else "u'bar'"
self.assertEquals(res, expected)
def test_handles_gettext_lazy(self):
from django.utils.functional import lazy
def fake_gettext(to_translate):
return 'Igpay Atinlay'
fake_gettext_lazy = lazy(fake_gettext, six.text_type)
result = transform(fake_gettext_lazy("something"))
self.assertTrue(isinstance(result, six.string_types))
expected = "'Igpay Atinlay'" if six.PY3 else "u'Igpay Atinlay'"
self.assertEquals(result, expected)
class ModelInstanceSerializerTestCase(TestCase):
def test_basic(self):
instance = TestModel()
result = transform(instance)
self.assertTrue(isinstance(result, six.string_types))
self.assertEquals(result, '<TestModel: TestModel object>')
class QuerySetSerializerTestCase(TestCase):
def test_basic(self):
from django.db.models.query import QuerySet
obj = QuerySet(model=TestModel)
result = transform(obj)
self.assertTrue(isinstance(result, six.string_types))
self.assertEquals(result, '<QuerySet: model=TestModel>')
class SentryExceptionHandlerTest(TestCase):
@fixture
def request(self):
return make_request()
@fixture
def exc_info(self):
return (ValueError, ValueError('lol world'), None)
@mock.patch.object(TempStoreClient, 'captureException')
@mock.patch('sys.exc_info')
def test_does_capture_exception(self, exc_info, captureException):
exc_info.return_value = self.exc_info
sentry_exception_handler(request=self.request)
captureException.assert_called_once_with(exc_info=self.exc_info, request=self.request)
@mock.patch.object(TempStoreClient, 'captureException')
@mock.patch('sys.exc_info')
@mock.patch('raven.contrib.django.models.get_option')
def test_does_exclude_filtered_types(self, get_option, exc_info, captureException):
exc_info.return_value = self.exc_info
get_option.return_value = ['ValueError']
sentry_exception_handler(request=self.request)
assert not captureException.called
@mock.patch.object(TempStoreClient, 'captureException')
@mock.patch('sys.exc_info')
@mock.patch('raven.contrib.django.models.get_option')
def test_ignore_exceptions_with_expression_match(self, get_option, exc_info, captureException):
exc_info.return_value = self.exc_info
get_option.return_value = ['builtins.*']
if not six.PY3:
get_option.return_value = ['exceptions.*']
sentry_exception_handler(request=self.request)
assert not captureException.called
@mock.patch.object(TempStoreClient, 'captureException')
@mock.patch('sys.exc_info')
@mock.patch('raven.contrib.django.models.get_option')
def test_ignore_exceptions_with_module_match(self, get_option, exc_info, captureException):
exc_info.return_value = self.exc_info
get_option.return_value = ['builtins.ValueError']
if not six.PY3:
get_option.return_value = ['exceptions.ValueError']
sentry_exception_handler(request=self.request)
assert not captureException.called