Blame tests/test_logging.py

Packit 6f3914
# -*- coding: utf-8 -*-
Packit 6f3914
Packit 6f3914
# Copyright (C) 2013-2018  Red Hat, Inc.
Packit 6f3914
#
Packit 6f3914
# This copyrighted material is made available to anyone wishing to use,
Packit 6f3914
# modify, copy, or redistribute it subject to the terms and conditions of
Packit 6f3914
# the GNU General Public License v.2, or (at your option) any later version.
Packit 6f3914
# This program is distributed in the hope that it will be useful, but WITHOUT
Packit 6f3914
# ANY WARRANTY expressed or implied, including the implied warranties of
Packit 6f3914
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General
Packit 6f3914
# Public License for more details.  You should have received a copy of the
Packit 6f3914
# GNU General Public License along with this program; if not, write to the
Packit 6f3914
# Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
Packit 6f3914
# 02110-1301, USA.  Any Red Hat trademarks that are incorporated in the
Packit 6f3914
# source code or documentation are not subject to the GNU General Public
Packit 6f3914
# License and may only be used or replicated with the express permission of
Packit 6f3914
# Red Hat, Inc.
Packit 6f3914
#
Packit 6f3914
Packit 6f3914
from __future__ import absolute_import
Packit 6f3914
from __future__ import unicode_literals
Packit 6f3914
Packit 6f3914
import logging
Packit 6f3914
import collections
Packit 6f3914
import operator
Packit 6f3914
import os
Packit 6f3914
import tempfile
Packit 6f3914
Packit 6f3914
import dnf.const
Packit 6f3914
import dnf.logging
Packit 6f3914
Packit 6f3914
import tests.support
Packit 6f3914
from tests.support import mock
Packit 6f3914
Packit 6f3914
Packit 6f3914
LogfileEntry = collections.namedtuple('LogfileEntry', ('date', 'time', 'message'))
Packit 6f3914
Packit 6f3914
Packit 6f3914
def _split_logfile_entry(entry):
Packit 6f3914
    record = entry.split(' ')
Packit 6f3914
    datetime = record[0].split('T')
Packit 6f3914
    # strip the trailing '\n' from the message:
Packit 6f3914
    message = ' '.join(record[2:])[:-1]
Packit 6f3914
    return LogfileEntry(date=datetime[0],
Packit 6f3914
                        time=datetime[1], message=message)
Packit 6f3914
Packit 6f3914
Packit 6f3914
def drop_all_handlers():
Packit 6f3914
    for logger_name in ('dnf', 'dnf.rpm'):
Packit 6f3914
        logger = logging.getLogger(logger_name)
Packit 6f3914
        for handler in logger.handlers[:]:
Packit 6f3914
            logger.removeHandler(handler)
Packit 6f3914
Packit 6f3914
Packit 6f3914
class TestLogging(tests.support.TestCase):
Packit 6f3914
    """Tests the logging mechanisms in DNF.
Packit 6f3914
Packit 6f3914
    If it causes a problem in the future that loggers are singletons that don't
Packit 6f3914
    get torn down between tests, look at logging.Manager internals.
Packit 6f3914
Packit 6f3914
    """
Packit 6f3914
Packit 6f3914
    def setUp(self):
Packit 6f3914
        self.logdir = tempfile.mkdtemp(prefix="dnf-logtest-")
Packit 6f3914
        self.log_size = 1024 * 1024
Packit 6f3914
        self.log_rotate = 4
Packit 6f3914
        self.logging = dnf.logging.Logging()
Packit 6f3914
Packit 6f3914
    def tearDown(self):
Packit 6f3914
        drop_all_handlers()
Packit 6f3914
        dnf.util.rm_rf(self.logdir)
Packit 6f3914
Packit 6f3914
    @staticmethod
Packit 6f3914
    def _bench(logger):
Packit 6f3914
        logger.debug(u"d")
Packit 6f3914
        logger.info(u"i")
Packit 6f3914
        logger.warning(u"w")
Packit 6f3914
        logger.error(u"e")
Packit 6f3914
Packit 6f3914
    def test_level_conversion(self):
Packit 6f3914
        self.assertRaises(AssertionError, dnf.logging._cfg_verbose_val2level, 11)
Packit 6f3914
        self.assertEqual(dnf.logging._cfg_verbose_val2level(0),
Packit 6f3914
                         dnf.logging.SUPERCRITICAL)
Packit 6f3914
        self.assertEqual(dnf.logging._cfg_verbose_val2level(7),
Packit 6f3914
                         dnf.logging.DDEBUG)
Packit 6f3914
Packit 6f3914
    def test_setup(self):
Packit 6f3914
        logger = logging.getLogger("dnf")
Packit 6f3914
        with tests.support.patch_std_streams() as (stdout, stderr):
Packit 6f3914
            self.logging._setup(
Packit Service a564d0
                logging.INFO, logging.ERROR, dnf.logging.TRACE, self.logdir, self.log_size, self.log_rotate)
Packit 6f3914
            self._bench(logger)
Packit 6f3914
        self.assertEqual("i\n", stdout.getvalue())
Packit 6f3914
        self.assertEqual("e\n", stderr.getvalue())
Packit 6f3914
Packit 6f3914
    def test_setup_verbose(self):
Packit 6f3914
        logger = logging.getLogger("dnf")
Packit 6f3914
        with tests.support.patch_std_streams() as (stdout, stderr):
Packit 6f3914
            self.logging._setup(
Packit Service a564d0
                logging.DEBUG, logging.WARNING, dnf.logging.TRACE, self.logdir, self.log_size, self.log_rotate)
Packit 6f3914
            self._bench(logger)
Packit 6f3914
        self.assertEqual("d\ni\n", stdout.getvalue())
Packit 6f3914
        self.assertEqual("w\ne\n", stderr.getvalue())
Packit 6f3914
Packit 6f3914
    @mock.patch('dnf.logging.Logging._setup')
Packit 6f3914
    def test_setup_from_dnf_conf(self, setup_m):
Packit 6f3914
        conf = mock.Mock(
Packit Service a564d0
            debuglevel=2, errorlevel=3, logfilelevel=2, logdir=self.logdir,
Packit 6f3914
            log_size=self.log_size, log_rotate=self.log_rotate)
Packit 6f3914
        self.logging._setup_from_dnf_conf(conf)
Packit 6f3914
        self.assertEqual(setup_m.call_args, mock.call(dnf.logging.INFO,
Packit 6f3914
                                                      dnf.logging.WARNING,
Packit Service a564d0
                                                      dnf.logging.INFO,
Packit 6f3914
                                                      self.logdir,
Packit 6f3914
                                                      self.log_size,
Packit 6f3914
                                                      self.log_rotate))
Packit 6f3914
        conf = mock.Mock(
Packit Service a564d0
            debuglevel=6, errorlevel=6, logfilelevel=6, logdir=self.logdir,
Packit 6f3914
            log_size=self.log_size, log_rotate=self.log_rotate)
Packit 6f3914
        self.logging._setup_from_dnf_conf(conf)
Packit 6f3914
        self.assertEqual(setup_m.call_args, mock.call(dnf.logging.DEBUG,
Packit 6f3914
                                                      dnf.logging.WARNING,
Packit Service a564d0
                                                      dnf.logging.DEBUG,
Packit 6f3914
                                                      self.logdir,
Packit 6f3914
                                                      self.log_size,
Packit 6f3914
                                                      self.log_rotate))
Packit 6f3914
Packit 6f3914
    def test_file_logging(self):
Packit 6f3914
        # log nothing to the console:
Packit 6f3914
        self.logging._setup(
Packit Service a564d0
            dnf.logging.SUPERCRITICAL, dnf.logging.SUPERCRITICAL, dnf.logging.TRACE,
Packit 6f3914
            self.logdir, self.log_size, self.log_rotate)
Packit 6f3914
        logger = logging.getLogger("dnf")
Packit 6f3914
        with tests.support.patch_std_streams() as (stdout, stderr):
Packit 6f3914
            logger.info("i")
Packit 6f3914
            logger.critical("c")
Packit 6f3914
        self.assertEqual(stdout.getvalue(), '')
Packit 6f3914
        self.assertEqual(stderr.getvalue(), '')
Packit 6f3914
        # yet the file should contain both the entries:
Packit 6f3914
        logfile = os.path.join(self.logdir, "dnf.log")
Packit 6f3914
        self.assertFile(logfile)
Packit 6f3914
        with open(logfile) as f:
Packit 6f3914
            msgs = map(operator.attrgetter("message"),
Packit 6f3914
                       map(_split_logfile_entry, f.readlines()))
Packit 6f3914
        self.assertSequenceEqual(list(msgs), [dnf.const.LOG_MARKER, 'i', 'c'])
Packit 6f3914
Packit 6f3914
    def test_rpm_logging(self):
Packit 6f3914
        # log everything to the console:
Packit 6f3914
        self.logging._setup(
Packit Service a564d0
            dnf.logging.SUBDEBUG, dnf.logging.SUBDEBUG, dnf.logging.TRACE,
Packit 6f3914
            self.logdir, self.log_size, self.log_rotate)
Packit 6f3914
        logger = logging.getLogger("dnf.rpm")
Packit 6f3914
        with tests.support.patch_std_streams() as (stdout, stderr):
Packit 6f3914
            logger.info('rpm transaction happens.')
Packit 6f3914
        # rpm logger never outputs to the console:
Packit 6f3914
        self.assertEqual(stdout.getvalue(), "")
Packit 6f3914
        self.assertEqual(stderr.getvalue(), "")
Packit 6f3914
        logfile = os.path.join(self.logdir, "dnf.rpm.log")
Packit 6f3914
        self.assertFile(logfile)
Packit 6f3914
        with open(logfile) as f:
Packit 6f3914
            msgs = map(operator.attrgetter("message"),
Packit 6f3914
                       map(_split_logfile_entry, f.readlines()))
Packit 6f3914
        self.assertSequenceEqual(
Packit 6f3914
            list(msgs),
Packit 6f3914
            [dnf.const.LOG_MARKER, 'rpm transaction happens.']
Packit 6f3914
        )
Packit 6f3914
Packit 6f3914
    def test_setup_only_once(self):
Packit 6f3914
        logger = logging.getLogger("dnf")
Packit 6f3914
        self.assertLength(logger.handlers, 0)
Packit 6f3914
        self.logging._setup(
Packit Service a564d0
            dnf.logging.SUBDEBUG, dnf.logging.SUBDEBUG, dnf.logging.TRACE,
Packit 6f3914
            self.logdir, self.log_size, self.log_rotate)
Packit 6f3914
        cnt = len(logger.handlers)
Packit 6f3914
        self.assertGreater(cnt, 0)
Packit 6f3914
        self.logging._setup(
Packit 6f3914
            dnf.logging.SUBDEBUG, dnf.logging.SUBDEBUG,
Packit 6f3914
            self.logdir, self.log_size, self.log_rotate)
Packit 6f3914
        # no new handlers
Packit 6f3914
        self.assertEqual(cnt, len(logger.handlers))