2017-05-25 17:59:42 +08:00
|
|
|
# -*- coding: utf8 -*-
|
|
|
|
from __future__ import unicode_literals
|
|
|
|
|
2017-05-26 13:12:02 +08:00
|
|
|
import sys
|
|
|
|
|
2016-11-21 18:26:43 +08:00
|
|
|
import pytest
|
|
|
|
|
|
|
|
|
2017-02-25 23:02:48 +08:00
|
|
|
WARNINGS_SUMMARY_HEADER = 'warnings summary'
|
|
|
|
|
2017-07-17 07:25:09 +08:00
|
|
|
|
2016-11-21 18:26:43 +08:00
|
|
|
@pytest.fixture
|
2017-03-05 07:53:42 +08:00
|
|
|
def pyfile_with_warnings(testdir, request):
|
|
|
|
"""
|
|
|
|
Create a test file which calls a function in a module which generates warnings.
|
|
|
|
"""
|
|
|
|
testdir.syspathinsert()
|
|
|
|
test_name = request.function.__name__
|
|
|
|
module_name = test_name.lstrip('test_') + '_module'
|
|
|
|
testdir.makepyfile(**{
|
|
|
|
module_name: '''
|
|
|
|
import warnings
|
|
|
|
def foo():
|
2017-05-30 05:59:34 +08:00
|
|
|
warnings.warn(UserWarning("user warning"))
|
|
|
|
warnings.warn(RuntimeWarning("runtime warning"))
|
2017-03-05 07:53:42 +08:00
|
|
|
return 1
|
|
|
|
''',
|
|
|
|
test_name: '''
|
|
|
|
import {module_name}
|
|
|
|
def test_func():
|
|
|
|
assert {module_name}.foo() == 1
|
|
|
|
'''.format(module_name=module_name)
|
|
|
|
})
|
2016-11-21 18:26:43 +08:00
|
|
|
|
|
|
|
|
2017-07-21 10:11:14 +08:00
|
|
|
@pytest.mark.filterwarnings('always')
|
2016-11-21 18:26:43 +08:00
|
|
|
def test_normal_flow(testdir, pyfile_with_warnings):
|
2017-03-05 07:53:42 +08:00
|
|
|
"""
|
|
|
|
Check that the warnings section is displayed, containing test node ids followed by
|
|
|
|
all warnings generated by that test node.
|
|
|
|
"""
|
2017-02-18 22:57:26 +08:00
|
|
|
result = testdir.runpytest()
|
2016-11-21 18:26:43 +08:00
|
|
|
result.stdout.fnmatch_lines([
|
2017-02-25 23:02:48 +08:00
|
|
|
'*== %s ==*' % WARNINGS_SUMMARY_HEADER,
|
2016-11-21 18:26:43 +08:00
|
|
|
|
2017-03-05 07:53:42 +08:00
|
|
|
'*test_normal_flow.py::test_func',
|
|
|
|
|
2017-05-30 05:59:34 +08:00
|
|
|
'*normal_flow_module.py:3: UserWarning: user warning',
|
|
|
|
'* warnings.warn(UserWarning("user warning"))',
|
2016-11-21 18:26:43 +08:00
|
|
|
|
2017-05-30 05:59:34 +08:00
|
|
|
'*normal_flow_module.py:4: RuntimeWarning: runtime warning',
|
|
|
|
'* warnings.warn(RuntimeWarning("runtime warning"))',
|
2017-02-25 23:02:48 +08:00
|
|
|
'* 1 passed, 2 warnings*',
|
2016-11-21 18:26:43 +08:00
|
|
|
])
|
2017-03-05 07:53:42 +08:00
|
|
|
assert result.stdout.str().count('test_normal_flow.py::test_func') == 1
|
2016-11-21 18:26:43 +08:00
|
|
|
|
|
|
|
|
2017-07-21 10:11:14 +08:00
|
|
|
@pytest.mark.filterwarnings('always')
|
2017-03-05 03:07:37 +08:00
|
|
|
def test_setup_teardown_warnings(testdir, pyfile_with_warnings):
|
|
|
|
testdir.makepyfile('''
|
|
|
|
import warnings
|
|
|
|
import pytest
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def fix():
|
|
|
|
warnings.warn(UserWarning("warning during setup"))
|
|
|
|
yield
|
|
|
|
warnings.warn(UserWarning("warning during teardown"))
|
|
|
|
|
|
|
|
def test_func(fix):
|
|
|
|
pass
|
|
|
|
''')
|
|
|
|
result = testdir.runpytest()
|
|
|
|
result.stdout.fnmatch_lines([
|
|
|
|
'*== %s ==*' % WARNINGS_SUMMARY_HEADER,
|
|
|
|
|
|
|
|
'*test_setup_teardown_warnings.py:6: UserWarning: warning during setup',
|
2017-03-17 08:55:03 +08:00
|
|
|
'*warnings.warn(UserWarning("warning during setup"))',
|
2017-03-05 03:07:37 +08:00
|
|
|
|
|
|
|
'*test_setup_teardown_warnings.py:8: UserWarning: warning during teardown',
|
2017-03-17 08:55:03 +08:00
|
|
|
'*warnings.warn(UserWarning("warning during teardown"))',
|
2017-03-05 03:07:37 +08:00
|
|
|
'* 1 passed, 2 warnings*',
|
|
|
|
])
|
|
|
|
|
|
|
|
|
2016-11-21 18:26:43 +08:00
|
|
|
@pytest.mark.parametrize('method', ['cmdline', 'ini'])
|
|
|
|
def test_as_errors(testdir, pyfile_with_warnings, method):
|
|
|
|
args = ('-W', 'error') if method == 'cmdline' else ()
|
|
|
|
if method == 'ini':
|
|
|
|
testdir.makeini('''
|
|
|
|
[pytest]
|
|
|
|
filterwarnings= error
|
|
|
|
''')
|
2017-02-18 22:57:26 +08:00
|
|
|
result = testdir.runpytest(*args)
|
2016-11-21 18:26:43 +08:00
|
|
|
result.stdout.fnmatch_lines([
|
2017-05-30 05:59:34 +08:00
|
|
|
'E UserWarning: user warning',
|
|
|
|
'as_errors_module.py:3: UserWarning',
|
2016-11-21 18:26:43 +08:00
|
|
|
'* 1 failed in *',
|
|
|
|
])
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('method', ['cmdline', 'ini'])
|
|
|
|
def test_ignore(testdir, pyfile_with_warnings, method):
|
|
|
|
args = ('-W', 'ignore') if method == 'cmdline' else ()
|
|
|
|
if method == 'ini':
|
|
|
|
testdir.makeini('''
|
|
|
|
[pytest]
|
|
|
|
filterwarnings= ignore
|
|
|
|
''')
|
|
|
|
|
2017-02-18 22:57:26 +08:00
|
|
|
result = testdir.runpytest(*args)
|
2016-11-21 18:26:43 +08:00
|
|
|
result.stdout.fnmatch_lines([
|
|
|
|
'* 1 passed in *',
|
|
|
|
])
|
2017-02-25 23:02:48 +08:00
|
|
|
assert WARNINGS_SUMMARY_HEADER not in result.stdout.str()
|
2016-11-21 18:26:43 +08:00
|
|
|
|
2017-05-25 17:59:42 +08:00
|
|
|
|
2017-05-26 13:12:02 +08:00
|
|
|
@pytest.mark.skipif(sys.version_info < (3, 0),
|
|
|
|
reason='warnings message is unicode is ok in python3')
|
2017-07-21 10:11:14 +08:00
|
|
|
@pytest.mark.filterwarnings('always')
|
2017-05-25 17:59:42 +08:00
|
|
|
def test_unicode(testdir, pyfile_with_warnings):
|
|
|
|
testdir.makepyfile('''
|
|
|
|
# -*- coding: utf8 -*-
|
|
|
|
import warnings
|
|
|
|
import pytest
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def fix():
|
|
|
|
warnings.warn(u"测试")
|
|
|
|
yield
|
|
|
|
|
|
|
|
def test_func(fix):
|
|
|
|
pass
|
|
|
|
''')
|
|
|
|
result = testdir.runpytest()
|
|
|
|
result.stdout.fnmatch_lines([
|
|
|
|
'*== %s ==*' % WARNINGS_SUMMARY_HEADER,
|
2017-05-30 05:59:34 +08:00
|
|
|
'*test_unicode.py:8: UserWarning: \u6d4b\u8bd5*',
|
2017-05-25 17:59:42 +08:00
|
|
|
'* 1 passed, 1 warnings*',
|
|
|
|
])
|
|
|
|
|
2017-05-26 13:12:02 +08:00
|
|
|
|
|
|
|
@pytest.mark.skipif(sys.version_info >= (3, 0),
|
|
|
|
reason='warnings message is broken as it is not str instance')
|
|
|
|
def test_py2_unicode(testdir, pyfile_with_warnings):
|
|
|
|
testdir.makepyfile('''
|
|
|
|
# -*- coding: utf8 -*-
|
|
|
|
import warnings
|
|
|
|
import pytest
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def fix():
|
|
|
|
warnings.warn(u"测试")
|
|
|
|
yield
|
|
|
|
|
2017-07-21 10:11:14 +08:00
|
|
|
@pytest.mark.filterwarnings('always')
|
2017-05-26 13:12:02 +08:00
|
|
|
def test_func(fix):
|
|
|
|
pass
|
|
|
|
''')
|
|
|
|
result = testdir.runpytest()
|
|
|
|
result.stdout.fnmatch_lines([
|
|
|
|
'*== %s ==*' % WARNINGS_SUMMARY_HEADER,
|
|
|
|
|
2017-10-03 22:09:24 +08:00
|
|
|
'*test_py2_unicode.py:8: UserWarning: \\u6d4b\\u8bd5',
|
2017-05-26 13:12:02 +08:00
|
|
|
'*warnings.warn(u"\u6d4b\u8bd5")',
|
2017-06-05 21:43:15 +08:00
|
|
|
'*warnings.py:*: UnicodeWarning: Warning is using unicode non*',
|
2017-05-26 13:12:02 +08:00
|
|
|
'* 1 passed, 2 warnings*',
|
|
|
|
])
|
2017-05-30 05:59:34 +08:00
|
|
|
|
|
|
|
|
2017-10-03 03:20:51 +08:00
|
|
|
def test_py2_unicode_ascii(testdir):
|
|
|
|
"""Ensure that our warning about 'unicode warnings containing non-ascii messages'
|
|
|
|
does not trigger with ascii-convertible messages"""
|
|
|
|
testdir.makeini('[pytest]')
|
|
|
|
testdir.makepyfile('''
|
|
|
|
import pytest
|
|
|
|
import warnings
|
|
|
|
|
|
|
|
@pytest.mark.filterwarnings('always')
|
|
|
|
def test_func():
|
|
|
|
warnings.warn(u"hello")
|
|
|
|
''')
|
|
|
|
result = testdir.runpytest()
|
|
|
|
result.stdout.fnmatch_lines([
|
|
|
|
'*== %s ==*' % WARNINGS_SUMMARY_HEADER,
|
|
|
|
'*warnings.warn(u"hello")',
|
|
|
|
'* 1 passed, 1 warnings in*'
|
|
|
|
])
|
|
|
|
|
|
|
|
|
2017-05-30 05:59:34 +08:00
|
|
|
def test_works_with_filterwarnings(testdir):
|
|
|
|
"""Ensure our warnings capture does not mess with pre-installed filters (#2430)."""
|
|
|
|
testdir.makepyfile('''
|
|
|
|
import warnings
|
|
|
|
|
|
|
|
class MyWarning(Warning):
|
|
|
|
pass
|
2017-07-17 07:25:06 +08:00
|
|
|
|
2017-05-30 05:59:34 +08:00
|
|
|
warnings.filterwarnings("error", category=MyWarning)
|
2017-07-17 07:25:06 +08:00
|
|
|
|
2017-05-30 05:59:34 +08:00
|
|
|
class TestWarnings(object):
|
|
|
|
def test_my_warning(self):
|
|
|
|
try:
|
|
|
|
warnings.warn(MyWarning("warn!"))
|
|
|
|
assert False
|
|
|
|
except MyWarning:
|
|
|
|
assert True
|
|
|
|
''')
|
|
|
|
result = testdir.runpytest()
|
|
|
|
result.stdout.fnmatch_lines([
|
|
|
|
'*== 1 passed in *',
|
|
|
|
])
|
2017-07-21 09:02:21 +08:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('default_config', ['ini', 'cmdline'])
|
|
|
|
def test_filterwarnings_mark(testdir, default_config):
|
|
|
|
"""
|
|
|
|
Test ``filterwarnings`` mark works and takes precedence over command line and ini options.
|
|
|
|
"""
|
|
|
|
if default_config == 'ini':
|
|
|
|
testdir.makeini("""
|
|
|
|
[pytest]
|
|
|
|
filterwarnings = always
|
|
|
|
""")
|
|
|
|
testdir.makepyfile("""
|
|
|
|
import warnings
|
|
|
|
import pytest
|
|
|
|
|
|
|
|
@pytest.mark.filterwarnings('ignore::RuntimeWarning')
|
|
|
|
def test_ignore_runtime_warning():
|
|
|
|
warnings.warn(RuntimeWarning())
|
|
|
|
|
|
|
|
@pytest.mark.filterwarnings('error')
|
|
|
|
def test_warning_error():
|
|
|
|
warnings.warn(RuntimeWarning())
|
|
|
|
|
|
|
|
def test_show_warning():
|
|
|
|
warnings.warn(RuntimeWarning())
|
|
|
|
""")
|
|
|
|
result = testdir.runpytest('-W always' if default_config == 'cmdline' else '')
|
|
|
|
result.stdout.fnmatch_lines(['*= 1 failed, 2 passed, 1 warnings in *'])
|