forked from LBRYCommunity/lbry-sdk
cleanup setup_logging()
added more logging tests and refactored for merging into master
This commit is contained in:
parent
dd5507036f
commit
98b9972fa5
3 changed files with 90 additions and 30 deletions
|
@ -7,10 +7,10 @@ import asyncio
|
||||||
import argparse
|
import argparse
|
||||||
import logging
|
import logging
|
||||||
import logging.handlers
|
import logging.handlers
|
||||||
from docopt import docopt
|
|
||||||
|
|
||||||
import aiohttp
|
import aiohttp
|
||||||
from aiohttp.web import GracefulExit
|
from aiohttp.web import GracefulExit
|
||||||
|
from docopt import docopt
|
||||||
|
|
||||||
from lbry import __version__ as lbrynet_version
|
from lbry import __version__ as lbrynet_version
|
||||||
from lbry.extras.daemon.loggly_handler import get_loggly_handler
|
from lbry.extras.daemon.loggly_handler import get_loggly_handler
|
||||||
|
@ -142,7 +142,7 @@ class ArgumentParser(argparse.ArgumentParser):
|
||||||
|
|
||||||
def error(self, message):
|
def error(self, message):
|
||||||
self.print_help(argparse._sys.stderr)
|
self.print_help(argparse._sys.stderr)
|
||||||
self.exit(2, '\n'+message+'\n')
|
self.exit(2, f"\n{message}\n")
|
||||||
|
|
||||||
|
|
||||||
class HelpFormatter(argparse.HelpFormatter):
|
class HelpFormatter(argparse.HelpFormatter):
|
||||||
|
@ -223,11 +223,9 @@ def ensure_directory_exists(path: str):
|
||||||
LOG_MODULES = ('lbry', 'torba', 'aioupnp')
|
LOG_MODULES = ('lbry', 'torba', 'aioupnp')
|
||||||
|
|
||||||
|
|
||||||
def setup_logging(args: argparse.Namespace, conf: Config, logger: logging.Logger):
|
def setup_logging(logger: logging.Logger, args: argparse.Namespace, conf: Config):
|
||||||
default_formatter = logging.Formatter("%(asctime)s %(levelname)-8s %(name)s:%(lineno)d: %(message)s")
|
default_formatter = logging.Formatter("%(asctime)s %(levelname)-8s %(name)s:%(lineno)d: %(message)s")
|
||||||
file_handler = logging.handlers.RotatingFileHandler(
|
file_handler = logging.handlers.RotatingFileHandler(conf.log_file_path, maxBytes=2097152, backupCount=5)
|
||||||
conf.log_file_path, maxBytes=2097152, backupCount=5
|
|
||||||
)
|
|
||||||
file_handler.setFormatter(default_formatter)
|
file_handler.setFormatter(default_formatter)
|
||||||
for module_name in LOG_MODULES:
|
for module_name in LOG_MODULES:
|
||||||
logger.getChild(module_name).addHandler(file_handler)
|
logger.getChild(module_name).addHandler(file_handler)
|
||||||
|
@ -259,7 +257,7 @@ def run_daemon(args: argparse.Namespace, conf: Config):
|
||||||
loop = asyncio.get_event_loop()
|
loop = asyncio.get_event_loop()
|
||||||
if args.verbose is not None:
|
if args.verbose is not None:
|
||||||
loop.set_debug(True)
|
loop.set_debug(True)
|
||||||
setup_logging(args, conf, logging.getLogger())
|
setup_logging(logging.getLogger(), args, conf)
|
||||||
daemon = Daemon(conf)
|
daemon = Daemon(conf)
|
||||||
|
|
||||||
def __exit():
|
def __exit():
|
||||||
|
|
|
@ -67,6 +67,10 @@ class HTTPSLogglyHandler(logging.Handler):
|
||||||
def emit(self, record):
|
def emit(self, record):
|
||||||
asyncio.ensure_future(self._emit(record))
|
asyncio.ensure_future(self._emit(record))
|
||||||
|
|
||||||
|
def close(self):
|
||||||
|
super().close()
|
||||||
|
asyncio.create_task(self._session.close())
|
||||||
|
|
||||||
|
|
||||||
def get_loggly_handler():
|
def get_loggly_handler():
|
||||||
handler = HTTPSLogglyHandler(LOGGLY_TOKEN)
|
handler = HTTPSLogglyHandler(LOGGLY_TOKEN)
|
||||||
|
|
|
@ -1,7 +1,12 @@
|
||||||
|
import os
|
||||||
|
import tempfile
|
||||||
|
import shutil
|
||||||
import contextlib
|
import contextlib
|
||||||
import logging
|
import logging
|
||||||
from io import StringIO
|
from io import StringIO
|
||||||
from unittest import TestCase
|
from unittest import TestCase
|
||||||
|
from types import SimpleNamespace
|
||||||
|
from contextlib import asynccontextmanager
|
||||||
|
|
||||||
import docopt
|
import docopt
|
||||||
from torba.testcase import AsyncioTestCase
|
from torba.testcase import AsyncioTestCase
|
||||||
|
@ -9,39 +14,92 @@ from torba.testcase import AsyncioTestCase
|
||||||
from lbry.extras.cli import normalize_value, main, setup_logging
|
from lbry.extras.cli import normalize_value, main, setup_logging
|
||||||
from lbry.extras.system_info import get_platform
|
from lbry.extras.system_info import get_platform
|
||||||
from lbry.extras.daemon.Daemon import Daemon
|
from lbry.extras.daemon.Daemon import Daemon
|
||||||
|
from lbry.extras.daemon.loggly_handler import HTTPSLogglyHandler
|
||||||
from lbry.conf import Config
|
from lbry.conf import Config
|
||||||
from lbry.extras import cli
|
from lbry.extras import cli
|
||||||
|
|
||||||
|
|
||||||
class CLILoggingTest(AsyncioTestCase):
|
@asynccontextmanager
|
||||||
|
async def get_logger(argv, **conf_options):
|
||||||
|
# loggly requires loop, so we do this in async function
|
||||||
|
|
||||||
async def test_setup_logging(self):
|
logger = logging.getLogger('test-root-logger')
|
||||||
# test needs to be async to avoid warnings from loggly
|
temp_dir = tempfile.mkdtemp()
|
||||||
|
temp_config = os.path.join(temp_dir, 'settings.yml')
|
||||||
|
|
||||||
def setup(argv):
|
try:
|
||||||
|
# create a config (to be loaded on startup)
|
||||||
|
_conf = Config.create_from_arguments(SimpleNamespace(config=temp_config))
|
||||||
|
with _conf.update_config():
|
||||||
|
for opt_name, opt_value in conf_options.items():
|
||||||
|
setattr(_conf, opt_name, opt_value)
|
||||||
|
|
||||||
|
# do what happens on startup
|
||||||
|
argv.extend(['--data-dir', temp_dir])
|
||||||
|
argv.extend(['--wallet-dir', temp_dir])
|
||||||
|
argv.extend(['--config', temp_config])
|
||||||
parser = cli.get_argument_parser()
|
parser = cli.get_argument_parser()
|
||||||
args, command_args = parser.parse_known_args(argv)
|
args, command_args = parser.parse_known_args(argv)
|
||||||
conf = Config.create_from_arguments(args)
|
conf: Config = Config.create_from_arguments(args)
|
||||||
conf.data_dir = '/tmp'
|
setup_logging(logger, args, conf)
|
||||||
setup_logging(args, conf, logger)
|
yield logger
|
||||||
|
|
||||||
logger = logging.getLogger('test_logger')
|
finally:
|
||||||
|
shutil.rmtree(temp_dir, ignore_errors=True)
|
||||||
|
for mod in cli.LOG_MODULES:
|
||||||
|
log = logger.getChild(mod)
|
||||||
|
log.setLevel(logging.NOTSET)
|
||||||
|
while log.handlers:
|
||||||
|
h = log.handlers[0]
|
||||||
|
log.removeHandler(log.handlers[0])
|
||||||
|
h.close()
|
||||||
|
|
||||||
setup(["start"])
|
|
||||||
self.assertTrue(logger.getChild("lbry").isEnabledFor(logging.INFO))
|
|
||||||
self.assertFalse(logger.getChild("lbry").isEnabledFor(logging.DEBUG))
|
|
||||||
|
|
||||||
setup(["start", "--verbose"])
|
class CLILoggingTest(AsyncioTestCase):
|
||||||
self.assertTrue(logger.getChild("lbry").isEnabledFor(logging.DEBUG))
|
|
||||||
self.assertTrue(logger.getChild("lbry").isEnabledFor(logging.INFO))
|
|
||||||
self.assertFalse(logger.getChild("torba").isEnabledFor(logging.DEBUG))
|
|
||||||
|
|
||||||
setup(["start", "--verbose", "lbry.extras", "lbry.wallet", "torba.client"])
|
async def test_verbose_logging(self):
|
||||||
self.assertTrue(logger.getChild("lbry.extras").isEnabledFor(logging.DEBUG))
|
async with get_logger(["start", "--quiet"], share_usage_data=False) as log:
|
||||||
self.assertTrue(logger.getChild("lbry.wallet").isEnabledFor(logging.DEBUG))
|
log = log.getChild("lbry")
|
||||||
self.assertTrue(logger.getChild("torba.client").isEnabledFor(logging.DEBUG))
|
self.assertTrue(log.isEnabledFor(logging.INFO))
|
||||||
self.assertFalse(logger.getChild("lbry").isEnabledFor(logging.DEBUG))
|
self.assertFalse(log.isEnabledFor(logging.DEBUG))
|
||||||
self.assertFalse(logger.getChild("torba").isEnabledFor(logging.DEBUG))
|
self.assertFalse(log.isEnabledFor(logging.DEBUG))
|
||||||
|
self.assertEqual(len(log.handlers), 1)
|
||||||
|
self.assertIsInstance(log.handlers[0], logging.handlers.RotatingFileHandler)
|
||||||
|
|
||||||
|
async with get_logger(["start", "--verbose"]) as log:
|
||||||
|
self.assertTrue(log.getChild("lbry").isEnabledFor(logging.DEBUG))
|
||||||
|
self.assertTrue(log.getChild("lbry").isEnabledFor(logging.INFO))
|
||||||
|
self.assertFalse(log.getChild("torba").isEnabledFor(logging.DEBUG))
|
||||||
|
|
||||||
|
async with get_logger(["start", "--verbose", "lbry.extras", "lbry.wallet", "torba.client"]) as log:
|
||||||
|
self.assertTrue(log.getChild("lbry.extras").isEnabledFor(logging.DEBUG))
|
||||||
|
self.assertTrue(log.getChild("lbry.wallet").isEnabledFor(logging.DEBUG))
|
||||||
|
self.assertTrue(log.getChild("torba.client").isEnabledFor(logging.DEBUG))
|
||||||
|
self.assertFalse(log.getChild("lbry").isEnabledFor(logging.DEBUG))
|
||||||
|
self.assertFalse(log.getChild("torba").isEnabledFor(logging.DEBUG))
|
||||||
|
|
||||||
|
async def test_loggly(self):
|
||||||
|
async with get_logger(["start"], share_usage_data=False) as log:
|
||||||
|
self.assertEqual(len(log.getChild("lbry").handlers), 2) # file and console
|
||||||
|
async with get_logger(["start"]) as log: # default share_usage_data=True
|
||||||
|
log = log.getChild("lbry")
|
||||||
|
self.assertEqual(len(log.handlers), 3)
|
||||||
|
self.assertIsInstance(log.handlers[2], HTTPSLogglyHandler)
|
||||||
|
async with get_logger(["start"], share_usage_data=True) as log: # explicit share_usage_data=True
|
||||||
|
log = log.getChild("lbry")
|
||||||
|
self.assertEqual(len(log.handlers), 3)
|
||||||
|
self.assertIsInstance(log.handlers[2], HTTPSLogglyHandler)
|
||||||
|
|
||||||
|
async def test_quiet(self):
|
||||||
|
async with get_logger(["start"]) as log: # default is loud
|
||||||
|
log = log.getChild("lbry")
|
||||||
|
self.assertEqual(len(log.handlers), 3)
|
||||||
|
self.assertIs(type(log.handlers[1]), logging.StreamHandler)
|
||||||
|
async with get_logger(["start", "--quiet"]) as log:
|
||||||
|
log = log.getChild("lbry")
|
||||||
|
self.assertEqual(len(log.handlers), 2)
|
||||||
|
self.assertIsNot(type(log.handlers[0]), logging.StreamHandler)
|
||||||
|
self.assertIsNot(type(log.handlers[1]), logging.StreamHandler)
|
||||||
|
|
||||||
|
|
||||||
class CLITest(AsyncioTestCase):
|
class CLITest(AsyncioTestCase):
|
||||||
|
|
Loading…
Add table
Reference in a new issue