123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374 |
- from __future__ import unicode_literals
- import copy
- import logging
- import sys
- from unittest import skipUnless
- import warnings
- from django.conf import LazySettings
- from django.core import mail
- from django.test import TestCase, RequestFactory
- from django.test.utils import override_settings, patch_logger
- from django.utils.encoding import force_text
- from django.utils.log import (CallbackFilter, RequireDebugFalse,
- RequireDebugTrue)
- from django.utils.six import StringIO
- from admin_scripts.tests import AdminScriptTestCase
- from .logconfig import MyEmailBackend
- OLD_LOGGING = {
- 'version': 1,
- 'disable_existing_loggers': False,
- 'handlers': {
- 'mail_admins': {
- 'level': 'ERROR',
- 'class': 'django.utils.log.AdminEmailHandler'
- }
- },
- 'loggers': {
- 'django.request': {
- 'handlers': ['mail_admins'],
- 'level': 'ERROR',
- 'propagate': True,
- },
- }
- }
- class LoggingFiltersTest(TestCase):
- def test_require_debug_false_filter(self):
- """
- Test the RequireDebugFalse filter class.
- """
- filter_ = RequireDebugFalse()
- with self.settings(DEBUG=True):
- self.assertEqual(filter_.filter("record is not used"), False)
- with self.settings(DEBUG=False):
- self.assertEqual(filter_.filter("record is not used"), True)
- def test_require_debug_true_filter(self):
- """
- Test the RequireDebugTrue filter class.
- """
- filter_ = RequireDebugTrue()
- with self.settings(DEBUG=True):
- self.assertEqual(filter_.filter("record is not used"), True)
- with self.settings(DEBUG=False):
- self.assertEqual(filter_.filter("record is not used"), False)
- class DefaultLoggingTest(TestCase):
- def setUp(self):
- self.logger = logging.getLogger('django')
- self.old_stream = self.logger.handlers[0].stream
- def tearDown(self):
- self.logger.handlers[0].stream = self.old_stream
- def test_django_logger(self):
- """
- The 'django' base logger only output anything when DEBUG=True.
- """
- output = StringIO()
- self.logger.handlers[0].stream = output
- self.logger.error("Hey, this is an error.")
- self.assertEqual(output.getvalue(), '')
- with self.settings(DEBUG=True):
- self.logger.error("Hey, this is an error.")
- self.assertEqual(output.getvalue(), 'Hey, this is an error.\n')
- class WarningLoggerTests(TestCase):
- """
- Tests that warnings output for DeprecationWarnings is enabled
- and captured to the logging system
- """
- def setUp(self):
-
-
-
-
-
- self._old_capture_state = bool(getattr(logging, '_warnings_showwarning', False))
- logging.captureWarnings(True)
-
-
-
- self.logger = logging.getLogger('py.warnings')
- self.outputs = []
- self.old_streams = []
- for handler in self.logger.handlers:
- self.old_streams.append(handler.stream)
- self.outputs.append(StringIO())
- handler.stream = self.outputs[-1]
- def tearDown(self):
- for i, handler in enumerate(self.logger.handlers):
- self.logger.handlers[i].stream = self.old_streams[i]
-
- logging.captureWarnings(self._old_capture_state)
- @override_settings(DEBUG=True)
- def test_warnings_capture(self):
- warnings.warn('Foo Deprecated', DeprecationWarning)
- output = force_text(self.outputs[0].getvalue())
- self.assertTrue('Foo Deprecated' in output)
- def test_warnings_capture_debug_false(self):
- warnings.warn('Foo Deprecated', DeprecationWarning)
- output = force_text(self.outputs[0].getvalue())
- self.assertFalse('Foo Deprecated' in output)
- class CallbackFilterTest(TestCase):
- def test_sense(self):
- f_false = CallbackFilter(lambda r: False)
- f_true = CallbackFilter(lambda r: True)
- self.assertEqual(f_false.filter("record"), False)
- self.assertEqual(f_true.filter("record"), True)
- def test_passes_on_record(self):
- collector = []
- def _callback(record):
- collector.append(record)
- return True
- f = CallbackFilter(_callback)
- f.filter("a record")
- self.assertEqual(collector, ["a record"])
- class AdminEmailHandlerTest(TestCase):
- logger = logging.getLogger('django.request')
- def get_admin_email_handler(self, logger):
-
-
-
- admin_email_handler = [
- h for h in logger.handlers
- if h.__class__.__name__ == "AdminEmailHandler"
- ][0]
- return admin_email_handler
- def test_fail_silently(self):
- admin_email_handler = self.get_admin_email_handler(self.logger)
- self.assertTrue(admin_email_handler.connection().fail_silently)
- @override_settings(
- ADMINS=(('whatever admin', 'admin@example.com'),),
- EMAIL_SUBJECT_PREFIX='-SuperAwesomeSubject-'
- )
- def test_accepts_args(self):
- """
- Ensure that user-supplied arguments and the EMAIL_SUBJECT_PREFIX
- setting are used to compose the email subject.
- Refs #16736.
- """
- message = "Custom message that says '%s' and '%s'"
- token1 = 'ping'
- token2 = 'pong'
- admin_email_handler = self.get_admin_email_handler(self.logger)
-
- orig_filters = admin_email_handler.filters
- try:
- admin_email_handler.filters = []
- self.logger.error(message, token1, token2)
- self.assertEqual(len(mail.outbox), 1)
- self.assertEqual(mail.outbox[0].to, ['admin@example.com'])
- self.assertEqual(mail.outbox[0].subject,
- "-SuperAwesomeSubject-ERROR: Custom message that says 'ping' and 'pong'")
- finally:
-
- admin_email_handler.filters = orig_filters
- @override_settings(
- ADMINS=(('whatever admin', 'admin@example.com'),),
- EMAIL_SUBJECT_PREFIX='-SuperAwesomeSubject-',
- INTERNAL_IPS=('127.0.0.1',),
- )
- def test_accepts_args_and_request(self):
- """
- Ensure that the subject is also handled if being
- passed a request object.
- """
- message = "Custom message that says '%s' and '%s'"
- token1 = 'ping'
- token2 = 'pong'
- admin_email_handler = self.get_admin_email_handler(self.logger)
-
- orig_filters = admin_email_handler.filters
- try:
- admin_email_handler.filters = []
- rf = RequestFactory()
- request = rf.get('/')
- self.logger.error(message, token1, token2,
- extra={
- 'status_code': 403,
- 'request': request,
- }
- )
- self.assertEqual(len(mail.outbox), 1)
- self.assertEqual(mail.outbox[0].to, ['admin@example.com'])
- self.assertEqual(mail.outbox[0].subject,
- "-SuperAwesomeSubject-ERROR (internal IP): Custom message that says 'ping' and 'pong'")
- finally:
-
- admin_email_handler.filters = orig_filters
- @override_settings(
- ADMINS=(('admin', 'admin@example.com'),),
- EMAIL_SUBJECT_PREFIX='',
- DEBUG=False,
- )
- def test_subject_accepts_newlines(self):
- """
- Ensure that newlines in email reports' subjects are escaped to avoid
- AdminErrorHandler to fail.
- Refs #17281.
- """
- message = 'Message \r\n with newlines'
- expected_subject = 'ERROR: Message \\r\\n with newlines'
- self.assertEqual(len(mail.outbox), 0)
- self.logger.error(message)
- self.assertEqual(len(mail.outbox), 1)
- self.assertFalse('\n' in mail.outbox[0].subject)
- self.assertFalse('\r' in mail.outbox[0].subject)
- self.assertEqual(mail.outbox[0].subject, expected_subject)
- @override_settings(
- ADMINS=(('admin', 'admin@example.com'),),
- EMAIL_SUBJECT_PREFIX='',
- DEBUG=False,
- )
- def test_truncate_subject(self):
- """
- RFC 2822's hard limit is 998 characters per line.
- So, minus "Subject: ", the actual subject must be no longer than 989
- characters.
- Refs #17281.
- """
- message = 'a' * 1000
- expected_subject = 'ERROR: aa' + 'a' * 980
- self.assertEqual(len(mail.outbox), 0)
- self.logger.error(message)
- self.assertEqual(len(mail.outbox), 1)
- self.assertEqual(mail.outbox[0].subject, expected_subject)
- @override_settings(
- ADMINS=(('admin', 'admin@example.com'),),
- DEBUG=False,
- )
- def test_uses_custom_email_backend(self):
- """
- Refs #19325
- """
- message = 'All work and no play makes Jack a dull boy'
- admin_email_handler = self.get_admin_email_handler(self.logger)
- mail_admins_called = {'called': False}
- def my_mail_admins(*args, **kwargs):
- connection = kwargs['connection']
- self.assertIsInstance(connection, MyEmailBackend)
- mail_admins_called['called'] = True
-
- orig_mail_admins = mail.mail_admins
- orig_email_backend = admin_email_handler.email_backend
- mail.mail_admins = my_mail_admins
- admin_email_handler.email_backend = (
- 'logging_tests.logconfig.MyEmailBackend')
- try:
- self.logger.error(message)
- self.assertTrue(mail_admins_called['called'])
- finally:
-
- mail.mail_admins = orig_mail_admins
- admin_email_handler.email_backend = orig_email_backend
- class SettingsConfigTest(AdminScriptTestCase):
- """
- Test that accessing settings in a custom logging handler does not trigger
- a circular import error.
- """
- def setUp(self):
- log_config = """{
- 'version': 1,
- 'handlers': {
- 'custom_handler': {
- 'level': 'INFO',
- 'class': 'logging_tests.logconfig.MyHandler',
- }
- }
- }"""
- self.write_settings('settings.py', sdict={'LOGGING': log_config})
- def tearDown(self):
- self.remove_settings('settings.py')
- def test_circular_dependency(self):
-
- out, err = self.run_manage(['validate'])
- self.assertNoOutput(err)
- self.assertOutput(out, "0 errors found")
- def dictConfig(config):
- dictConfig.called = True
- dictConfig.called = False
- class SettingsConfigureLogging(TestCase):
- """
- Test that calling settings.configure() initializes the logging
- configuration.
- """
- def test_configure_initializes_logging(self):
- settings = LazySettings()
- settings.configure(
- LOGGING_CONFIG='logging_tests.tests.dictConfig')
- self.assertTrue(dictConfig.called)
- class SecurityLoggerTest(TestCase):
- urls = 'logging_tests.urls'
- def test_suspicious_operation_creates_log_message(self):
- with self.settings(DEBUG=True):
- with patch_logger('django.security.SuspiciousOperation', 'error') as calls:
- response = self.client.get('/suspicious/')
- self.assertEqual(len(calls), 1)
- self.assertEqual(calls[0], 'dubious')
- def test_suspicious_operation_uses_sublogger(self):
- with self.settings(DEBUG=True):
- with patch_logger('django.security.DisallowedHost', 'error') as calls:
- response = self.client.get('/suspicious_spec/')
- self.assertEqual(len(calls), 1)
- self.assertEqual(calls[0], 'dubious')
|