450 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			Python
		
	
	
	
			
		
		
	
	
			450 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			Python
		
	
	
	
| # mypy: disable-error-code="attr-defined"
 | |
| import logging
 | |
| 
 | |
| import pytest
 | |
| from _pytest.logging import caplog_records_key
 | |
| from _pytest.pytester import Pytester
 | |
| 
 | |
| logger = logging.getLogger(__name__)
 | |
| sublogger = logging.getLogger(__name__ + ".baz")
 | |
| 
 | |
| 
 | |
| @pytest.fixture
 | |
| def cleanup_disabled_logging():
 | |
|     """Simple fixture that ensures that a test doesn't disable logging.
 | |
| 
 | |
|     This is necessary because ``logging.disable()`` is global, so a test disabling logging
 | |
|     and not cleaning up after will break every test that runs after it.
 | |
| 
 | |
|     This behavior was moved to a fixture so that logging will be un-disabled even if the test fails an assertion.
 | |
|     """
 | |
|     yield
 | |
|     logging.disable(logging.NOTSET)
 | |
| 
 | |
| 
 | |
| def test_fixture_help(pytester: Pytester) -> None:
 | |
|     result = pytester.runpytest("--fixtures")
 | |
|     result.stdout.fnmatch_lines(["*caplog*"])
 | |
| 
 | |
| 
 | |
| def test_change_level(caplog):
 | |
|     caplog.set_level(logging.INFO)
 | |
|     logger.debug("handler DEBUG level")
 | |
|     logger.info("handler INFO level")
 | |
| 
 | |
|     caplog.set_level(logging.CRITICAL, logger=sublogger.name)
 | |
|     sublogger.warning("logger WARNING level")
 | |
|     sublogger.critical("logger CRITICAL level")
 | |
| 
 | |
|     assert "DEBUG" not in caplog.text
 | |
|     assert "INFO" in caplog.text
 | |
|     assert "WARNING" not in caplog.text
 | |
|     assert "CRITICAL" in caplog.text
 | |
| 
 | |
| 
 | |
| def test_change_level_logging_disabled(caplog, cleanup_disabled_logging):
 | |
|     logging.disable(logging.CRITICAL)
 | |
|     assert logging.root.manager.disable == logging.CRITICAL
 | |
|     caplog.set_level(logging.WARNING)
 | |
|     logger.info("handler INFO level")
 | |
|     logger.warning("handler WARNING level")
 | |
| 
 | |
|     caplog.set_level(logging.CRITICAL, logger=sublogger.name)
 | |
|     sublogger.warning("logger SUB_WARNING level")
 | |
|     sublogger.critical("logger SUB_CRITICAL level")
 | |
| 
 | |
|     assert "INFO" not in caplog.text
 | |
|     assert "WARNING" in caplog.text
 | |
|     assert "SUB_WARNING" not in caplog.text
 | |
|     assert "SUB_CRITICAL" in caplog.text
 | |
| 
 | |
| 
 | |
| def test_change_level_undo(pytester: Pytester) -> None:
 | |
|     """Ensure that 'set_level' is undone after the end of the test.
 | |
| 
 | |
|     Tests the logging output themselves (affected both by logger and handler levels).
 | |
|     """
 | |
|     pytester.makepyfile(
 | |
|         """
 | |
|         import logging
 | |
| 
 | |
|         def test1(caplog):
 | |
|             caplog.set_level(logging.INFO)
 | |
|             # using + operator here so fnmatch_lines doesn't match the code in the traceback
 | |
|             logging.info('log from ' + 'test1')
 | |
|             assert 0
 | |
| 
 | |
|         def test2(caplog):
 | |
|             # using + operator here so fnmatch_lines doesn't match the code in the traceback
 | |
|             logging.info('log from ' + 'test2')
 | |
|             assert 0
 | |
|     """
 | |
|     )
 | |
|     result = pytester.runpytest()
 | |
|     result.stdout.fnmatch_lines(["*log from test1*", "*2 failed in *"])
 | |
|     result.stdout.no_fnmatch_line("*log from test2*")
 | |
| 
 | |
| 
 | |
| def test_change_disabled_level_undo(
 | |
|     pytester: Pytester, cleanup_disabled_logging
 | |
| ) -> None:
 | |
|     """Ensure that '_force_enable_logging' in 'set_level' is undone after the end of the test.
 | |
| 
 | |
|     Tests the logging output themselves (affected by disabled logging level).
 | |
|     """
 | |
|     pytester.makepyfile(
 | |
|         """
 | |
|         import logging
 | |
| 
 | |
|         def test1(caplog):
 | |
|             logging.disable(logging.CRITICAL)
 | |
|             caplog.set_level(logging.INFO)
 | |
|             # using + operator here so fnmatch_lines doesn't match the code in the traceback
 | |
|             logging.info('log from ' + 'test1')
 | |
|             assert 0
 | |
| 
 | |
|         def test2(caplog):
 | |
|             # using + operator here so fnmatch_lines doesn't match the code in the traceback
 | |
|             # use logging.warning because we need a level that will show up if logging.disabled
 | |
|             # isn't reset to ``CRITICAL`` after test1.
 | |
|             logging.warning('log from ' + 'test2')
 | |
|             assert 0
 | |
|     """
 | |
|     )
 | |
|     result = pytester.runpytest()
 | |
|     result.stdout.fnmatch_lines(["*log from test1*", "*2 failed in *"])
 | |
|     result.stdout.no_fnmatch_line("*log from test2*")
 | |
| 
 | |
| 
 | |
| def test_change_level_undos_handler_level(pytester: Pytester) -> None:
 | |
|     """Ensure that 'set_level' is undone after the end of the test (handler).
 | |
| 
 | |
|     Issue #7569. Tests the handler level specifically.
 | |
|     """
 | |
|     pytester.makepyfile(
 | |
|         """
 | |
|         import logging
 | |
| 
 | |
|         def test1(caplog):
 | |
|             assert caplog.handler.level == 0
 | |
|             caplog.set_level(9999)
 | |
|             caplog.set_level(41)
 | |
|             assert caplog.handler.level == 41
 | |
| 
 | |
|         def test2(caplog):
 | |
|             assert caplog.handler.level == 0
 | |
| 
 | |
|         def test3(caplog):
 | |
|             assert caplog.handler.level == 0
 | |
|             caplog.set_level(43)
 | |
|             assert caplog.handler.level == 43
 | |
|     """
 | |
|     )
 | |
|     result = pytester.runpytest()
 | |
|     result.assert_outcomes(passed=3)
 | |
| 
 | |
| 
 | |
| def test_with_statement(caplog):
 | |
|     with caplog.at_level(logging.INFO):
 | |
|         logger.debug("handler DEBUG level")
 | |
|         logger.info("handler INFO level")
 | |
| 
 | |
|         with caplog.at_level(logging.CRITICAL, logger=sublogger.name):
 | |
|             sublogger.warning("logger WARNING level")
 | |
|             sublogger.critical("logger CRITICAL level")
 | |
| 
 | |
|     assert "DEBUG" not in caplog.text
 | |
|     assert "INFO" in caplog.text
 | |
|     assert "WARNING" not in caplog.text
 | |
|     assert "CRITICAL" in caplog.text
 | |
| 
 | |
| 
 | |
| def test_with_statement_logging_disabled(caplog, cleanup_disabled_logging):
 | |
|     logging.disable(logging.CRITICAL)
 | |
|     assert logging.root.manager.disable == logging.CRITICAL
 | |
|     with caplog.at_level(logging.WARNING):
 | |
|         logger.debug("handler DEBUG level")
 | |
|         logger.info("handler INFO level")
 | |
|         logger.warning("handler WARNING level")
 | |
|         logger.error("handler ERROR level")
 | |
|         logger.critical("handler CRITICAL level")
 | |
| 
 | |
|         assert logging.root.manager.disable == logging.INFO
 | |
| 
 | |
|         with caplog.at_level(logging.CRITICAL, logger=sublogger.name):
 | |
|             sublogger.warning("logger SUB_WARNING level")
 | |
|             sublogger.critical("logger SUB_CRITICAL level")
 | |
| 
 | |
|     assert "DEBUG" not in caplog.text
 | |
|     assert "INFO" not in caplog.text
 | |
|     assert "WARNING" in caplog.text
 | |
|     assert "ERROR" in caplog.text
 | |
|     assert " CRITICAL" in caplog.text
 | |
|     assert "SUB_WARNING" not in caplog.text
 | |
|     assert "SUB_CRITICAL" in caplog.text
 | |
|     assert logging.root.manager.disable == logging.CRITICAL
 | |
| 
 | |
| 
 | |
| @pytest.mark.parametrize(
 | |
|     "level_str,expected_disable_level",
 | |
|     [
 | |
|         ("CRITICAL", logging.ERROR),
 | |
|         ("ERROR", logging.WARNING),
 | |
|         ("WARNING", logging.INFO),
 | |
|         ("INFO", logging.DEBUG),
 | |
|         ("DEBUG", logging.NOTSET),
 | |
|         ("NOTSET", logging.NOTSET),
 | |
|         ("NOTVALIDLEVEL", logging.NOTSET),
 | |
|     ],
 | |
| )
 | |
| def test_force_enable_logging_level_string(
 | |
|     caplog, cleanup_disabled_logging, level_str, expected_disable_level
 | |
| ):
 | |
|     """Test _force_enable_logging using a level string.
 | |
| 
 | |
|     ``expected_disable_level`` is one level below ``level_str`` because the disabled log level
 | |
|     always needs to be *at least* one level lower than the level that caplog is trying to capture.
 | |
|     """
 | |
|     test_logger = logging.getLogger("test_str_level_force_enable")
 | |
|     # Emulate a testing environment where all logging is disabled.
 | |
|     logging.disable(logging.CRITICAL)
 | |
|     # Make sure all logging is disabled.
 | |
|     assert not test_logger.isEnabledFor(logging.CRITICAL)
 | |
|     # Un-disable logging for `level_str`.
 | |
|     caplog._force_enable_logging(level_str, test_logger)
 | |
|     # Make sure that the disabled level is now one below the requested logging level.
 | |
|     # We don't use `isEnabledFor` here because that also checks the level set by
 | |
|     # `logging.setLevel()` which is irrelevant to `logging.disable()`.
 | |
|     assert test_logger.manager.disable == expected_disable_level
 | |
| 
 | |
| 
 | |
| def test_log_access(caplog):
 | |
|     caplog.set_level(logging.INFO)
 | |
|     logger.info("boo %s", "arg")
 | |
|     assert caplog.records[0].levelname == "INFO"
 | |
|     assert caplog.records[0].msg == "boo %s"
 | |
|     assert "boo arg" in caplog.text
 | |
| 
 | |
| 
 | |
| def test_messages(caplog):
 | |
|     caplog.set_level(logging.INFO)
 | |
|     logger.info("boo %s", "arg")
 | |
|     logger.info("bar %s\nbaz %s", "arg1", "arg2")
 | |
|     assert "boo arg" == caplog.messages[0]
 | |
|     assert "bar arg1\nbaz arg2" == caplog.messages[1]
 | |
|     assert caplog.text.count("\n") > len(caplog.messages)
 | |
|     assert len(caplog.text.splitlines()) > len(caplog.messages)
 | |
| 
 | |
|     try:
 | |
|         raise Exception("test")
 | |
|     except Exception:
 | |
|         logger.exception("oops")
 | |
| 
 | |
|     assert "oops" in caplog.text
 | |
|     assert "oops" in caplog.messages[-1]
 | |
|     # Tracebacks are stored in the record and not added until the formatter or handler.
 | |
|     assert "Exception" in caplog.text
 | |
|     assert "Exception" not in caplog.messages[-1]
 | |
| 
 | |
| 
 | |
| def test_record_tuples(caplog):
 | |
|     caplog.set_level(logging.INFO)
 | |
|     logger.info("boo %s", "arg")
 | |
| 
 | |
|     assert caplog.record_tuples == [(__name__, logging.INFO, "boo arg")]
 | |
| 
 | |
| 
 | |
| def test_unicode(caplog):
 | |
|     caplog.set_level(logging.INFO)
 | |
|     logger.info("bū")
 | |
|     assert caplog.records[0].levelname == "INFO"
 | |
|     assert caplog.records[0].msg == "bū"
 | |
|     assert "bū" in caplog.text
 | |
| 
 | |
| 
 | |
| def test_clear(caplog):
 | |
|     caplog.set_level(logging.INFO)
 | |
|     logger.info("bū")
 | |
|     assert len(caplog.records)
 | |
|     assert caplog.text
 | |
|     caplog.clear()
 | |
|     assert not len(caplog.records)
 | |
|     assert not caplog.text
 | |
| 
 | |
| 
 | |
| @pytest.fixture
 | |
| def logging_during_setup_and_teardown(caplog):
 | |
|     caplog.set_level("INFO")
 | |
|     logger.info("a_setup_log")
 | |
|     yield
 | |
|     logger.info("a_teardown_log")
 | |
|     assert [x.message for x in caplog.get_records("teardown")] == ["a_teardown_log"]
 | |
| 
 | |
| 
 | |
| def test_caplog_captures_for_all_stages(caplog, logging_during_setup_and_teardown):
 | |
|     assert not caplog.records
 | |
|     assert not caplog.get_records("call")
 | |
|     logger.info("a_call_log")
 | |
|     assert [x.message for x in caplog.get_records("call")] == ["a_call_log"]
 | |
| 
 | |
|     assert [x.message for x in caplog.get_records("setup")] == ["a_setup_log"]
 | |
| 
 | |
|     # This reaches into private API, don't use this type of thing in real tests!
 | |
|     assert set(caplog._item.stash[caplog_records_key]) == {"setup", "call"}
 | |
| 
 | |
| 
 | |
| def test_clear_for_call_stage(caplog, logging_during_setup_and_teardown):
 | |
|     logger.info("a_call_log")
 | |
|     assert [x.message for x in caplog.get_records("call")] == ["a_call_log"]
 | |
|     assert [x.message for x in caplog.get_records("setup")] == ["a_setup_log"]
 | |
|     assert set(caplog._item.stash[caplog_records_key]) == {"setup", "call"}
 | |
| 
 | |
|     caplog.clear()
 | |
| 
 | |
|     assert caplog.get_records("call") == []
 | |
|     assert [x.message for x in caplog.get_records("setup")] == ["a_setup_log"]
 | |
|     assert set(caplog._item.stash[caplog_records_key]) == {"setup", "call"}
 | |
| 
 | |
|     logging.info("a_call_log_after_clear")
 | |
|     assert [x.message for x in caplog.get_records("call")] == ["a_call_log_after_clear"]
 | |
|     assert [x.message for x in caplog.get_records("setup")] == ["a_setup_log"]
 | |
|     assert set(caplog._item.stash[caplog_records_key]) == {"setup", "call"}
 | |
| 
 | |
| 
 | |
| def test_ini_controls_global_log_level(pytester: Pytester) -> None:
 | |
|     pytester.makepyfile(
 | |
|         """
 | |
|         import pytest
 | |
|         import logging
 | |
|         def test_log_level_override(request, caplog):
 | |
|             plugin = request.config.pluginmanager.getplugin('logging-plugin')
 | |
|             assert plugin.log_level == logging.ERROR
 | |
|             logger = logging.getLogger('catchlog')
 | |
|             logger.warning("WARNING message won't be shown")
 | |
|             logger.error("ERROR message will be shown")
 | |
|             assert 'WARNING' not in caplog.text
 | |
|             assert 'ERROR' in caplog.text
 | |
|     """
 | |
|     )
 | |
|     pytester.makeini(
 | |
|         """
 | |
|         [pytest]
 | |
|         log_level=ERROR
 | |
|     """
 | |
|     )
 | |
| 
 | |
|     result = pytester.runpytest()
 | |
|     # make sure that that we get a '0' exit code for the testsuite
 | |
|     assert result.ret == 0
 | |
| 
 | |
| 
 | |
| def test_caplog_can_override_global_log_level(pytester: Pytester) -> None:
 | |
|     pytester.makepyfile(
 | |
|         """
 | |
|         import pytest
 | |
|         import logging
 | |
|         def test_log_level_override(request, caplog):
 | |
|             logger = logging.getLogger('catchlog')
 | |
|             plugin = request.config.pluginmanager.getplugin('logging-plugin')
 | |
|             assert plugin.log_level == logging.WARNING
 | |
| 
 | |
|             logger.info("INFO message won't be shown")
 | |
| 
 | |
|             caplog.set_level(logging.INFO, logger.name)
 | |
| 
 | |
|             with caplog.at_level(logging.DEBUG, logger.name):
 | |
|                 logger.debug("DEBUG message will be shown")
 | |
| 
 | |
|             logger.debug("DEBUG message won't be shown")
 | |
| 
 | |
|             with caplog.at_level(logging.CRITICAL, logger.name):
 | |
|                 logger.warning("WARNING message won't be shown")
 | |
| 
 | |
|             logger.debug("DEBUG message won't be shown")
 | |
|             logger.info("INFO message will be shown")
 | |
| 
 | |
|             assert "message won't be shown" not in caplog.text
 | |
|     """
 | |
|     )
 | |
|     pytester.makeini(
 | |
|         """
 | |
|         [pytest]
 | |
|         log_level=WARNING
 | |
|     """
 | |
|     )
 | |
| 
 | |
|     result = pytester.runpytest()
 | |
|     assert result.ret == 0
 | |
| 
 | |
| 
 | |
| def test_caplog_captures_despite_exception(pytester: Pytester) -> None:
 | |
|     pytester.makepyfile(
 | |
|         """
 | |
|         import pytest
 | |
|         import logging
 | |
|         def test_log_level_override(request, caplog):
 | |
|             logger = logging.getLogger('catchlog')
 | |
|             plugin = request.config.pluginmanager.getplugin('logging-plugin')
 | |
|             assert plugin.log_level == logging.WARNING
 | |
| 
 | |
|             logger.error("ERROR message " + "will be shown")
 | |
| 
 | |
|             with caplog.at_level(logging.DEBUG, logger.name):
 | |
|                 logger.debug("DEBUG message " + "won't be shown")
 | |
|                 raise Exception()
 | |
|     """
 | |
|     )
 | |
|     pytester.makeini(
 | |
|         """
 | |
|         [pytest]
 | |
|         log_level=WARNING
 | |
|     """
 | |
|     )
 | |
| 
 | |
|     result = pytester.runpytest()
 | |
|     result.stdout.fnmatch_lines(["*ERROR message will be shown*"])
 | |
|     result.stdout.no_fnmatch_line("*DEBUG message won't be shown*")
 | |
|     assert result.ret == 1
 | |
| 
 | |
| 
 | |
| def test_log_report_captures_according_to_config_option_upon_failure(
 | |
|     pytester: Pytester,
 | |
| ) -> None:
 | |
|     """Test that upon failure:
 | |
|     (1) `caplog` succeeded to capture the DEBUG message and assert on it => No `Exception` is raised.
 | |
|     (2) The `DEBUG` message does NOT appear in the `Captured log call` report.
 | |
|     (3) The stdout, `INFO`, and `WARNING` messages DO appear in the test reports due to `--log-level=INFO`.
 | |
|     """
 | |
|     pytester.makepyfile(
 | |
|         """
 | |
|         import pytest
 | |
|         import logging
 | |
| 
 | |
|         def function_that_logs():
 | |
|             logging.debug('DEBUG log ' + 'message')
 | |
|             logging.info('INFO log ' + 'message')
 | |
|             logging.warning('WARNING log ' + 'message')
 | |
|             print('Print ' + 'message')
 | |
| 
 | |
|         def test_that_fails(request, caplog):
 | |
|             plugin = request.config.pluginmanager.getplugin('logging-plugin')
 | |
|             assert plugin.log_level == logging.INFO
 | |
| 
 | |
|             with caplog.at_level(logging.DEBUG):
 | |
|                 function_that_logs()
 | |
| 
 | |
|             if 'DEBUG log ' + 'message' not in caplog.text:
 | |
|                 raise Exception('caplog failed to ' + 'capture DEBUG')
 | |
| 
 | |
|             assert False
 | |
|     """
 | |
|     )
 | |
| 
 | |
|     result = pytester.runpytest("--log-level=INFO")
 | |
|     result.stdout.no_fnmatch_line("*Exception: caplog failed to capture DEBUG*")
 | |
|     result.stdout.no_fnmatch_line("*DEBUG log message*")
 | |
|     result.stdout.fnmatch_lines(
 | |
|         ["*Print message*", "*INFO log message*", "*WARNING log message*"]
 | |
|     )
 | |
|     assert result.ret == 1
 |