Skip to content

Commit

Permalink
Add a module logger (#600)
Browse files Browse the repository at this point in the history
* update base.py

* update formats.base

* update compara.base

* update graphics.base

* revert logging changes

* use the correct Console
  • Loading branch information
tanghaibao authored Sep 28, 2023
1 parent c27dc37 commit 7e4c5e5
Show file tree
Hide file tree
Showing 4 changed files with 85 additions and 114 deletions.
108 changes: 48 additions & 60 deletions jcvi/apps/base.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,8 +10,6 @@
import sys
import logging
import fnmatch
import functools


from collections.abc import Iterable
from http.client import HTTPSConnection
Expand All @@ -29,7 +27,8 @@
from typing import Any, Collection, List, Optional, Union

from natsort import natsorted
from rich.logging import Console, RichHandler
from rich.console import Console
from rich.logging import RichHandler

from jcvi import __copyright__, __version__

Expand All @@ -40,28 +39,33 @@
JCVIHELP = "JCVI utility libraries {} [{}]\n".format(__version__, __copyright__)


def patch_debug(func):
@functools.wraps(func)
def wraps(*args, **kwargs):
import inspect
def debug(level=logging.DEBUG):
"""
Turn on the debugging
"""
logging.basicConfig(
level=level,
format="%(message)s",
datefmt="[%X]",
handlers=[RichHandler(console=Console(stderr=True))],
)

callerframerecord = inspect.stack()[1]
frame = callerframerecord[0]
info = inspect.getframeinfo(frame)
# get caller function name
caller = frame.f_code.co_name
patch_message = f"{info.filename}:{info.lineno}:{caller}"
old_debug = logging.debug

def my_debug(message: str, *args, **kwargs):
old_debug(f"{patch_message} {message}", *args, **kwargs)
debug()

logging.debug = my_debug
ret = func(*args, **kwargs)
logging.debug = old_debug
return ret

return wraps
def get_logger(name: str):
"""Return a logger with a default ColoredFormatter."""
logger = logging.getLogger(name)
if logger.hasHandlers():
logger.handlers.clear()
logger.addHandler(RichHandler())
logger.propagate = False
logger.setLevel(logging.INFO)
return logger


logger = get_logger("jcvi")


class ActionDispatcher(object):
Expand Down Expand Up @@ -603,12 +607,12 @@ def set_image_options(
iopts = ImageOptions(opts)

if opts.notex:
logging.info("--notex={}. latex use is disabled.".format(opts.notex))
logger.info("--notex=%s. latex use is disabled.", opts.notex)
elif not is_tex_available():
if not bool(which("latex")):
logging.info("`latex` not found. latex use is disabled.")
logger.info("`latex` not found. latex use is disabled.")
if not bool(which("lp")):
logging.info("`lp` not found. latex use is disabled.")
logger.info("`lp` not found. latex use is disabled.")

setup_theme(style=opts.style, font=opts.font, usetex=iopts.usetex)

Expand Down Expand Up @@ -1048,9 +1052,9 @@ def ConfigSectionMap(Config, section):
try:
cfg[option] = Config.get(section, option)
if cfg[option] == -1:
logging.debug("skip: {0}".format(option))
logger.debug("skip: %s", option)
except:
logging.debug("exception on {0}!".format(option))
logger.debug("exception on %s!", option)
cfg[option] = None
return cfg

Expand Down Expand Up @@ -1131,7 +1135,7 @@ def dmain(mainfile, type="action"):
def backup(filename):
bakname = filename + ".bak"
if op.exists(filename):
logging.debug("Backup `{0}` to `{1}`".format(filename, bakname))
logger.debug("Backup `%s` to `%s`", filename, bakname)
sh("mv {0} {1}".format(filename, bakname))
return bakname

Expand Down Expand Up @@ -1205,7 +1209,7 @@ def sh(
cmd += " &"

if log:
logging.debug(cmd)
logger.debug(cmd)

call_func = check_output if check else call
return call_func(cmd, shell=True, executable=shell, stderr=redirect_error)
Expand All @@ -1218,7 +1222,7 @@ def Popen(cmd, stdin=None, stdout=PIPE, debug=False, shell="/bin/bash"):
from subprocess import Popen as P

if debug:
logging.debug(cmd)
logger.debug(cmd)
# See: <https://blog.nelhage.com/2010/02/a-very-subtle-bug/>
proc = P(cmd, bufsize=1, stdin=stdin, stdout=stdout, shell=True, executable=shell)
return proc
Expand Down Expand Up @@ -1307,15 +1311,15 @@ def mkdir(dirname, overwrite=False):
if overwrite:
cleanup(dirname)
os.mkdir(dirname)
logging.debug("Overwrite folder `{0}`.".format(dirname))
logger.debug("Overwrite folder `%s`", dirname)
else:
return False # Nothing is changed
else:
try:
os.mkdir(dirname)
except:
os.makedirs(dirname)
logging.debug("`{0}` not found. Creating new.".format(dirname))
logger.debug("`%s` not found. Creating new.", dirname)

return True

Expand Down Expand Up @@ -1377,7 +1381,7 @@ def need_update(a: str, b: str, warn: bool = False) -> bool:
or any(is_newer_file(x, y) for x in a for y in b)
)
if (not should_update) and warn:
logging.debug("File `{}` found. Computation skipped.".format(", ".join(b)))
logger.debug("File `%s` found. Computation skipped.", ", ".join(b))
return should_update


Expand Down Expand Up @@ -1476,8 +1480,7 @@ def download(
final_filename = filename or target
if op.exists(final_filename):
if debug:
msg = "File `{}` exists. Download skipped.".format(final_filename)
logging.info(msg)
logger.info("File `%s` exists. Download skipped.", final_filename)
success = True
else:
from jcvi.utils.ez_setup import get_best_downloader
Expand Down Expand Up @@ -1527,26 +1530,11 @@ def getfilesize(filename, ratio=None):
while size < heuristicsize:
size += 2**32
if size > 2**32:
logging.warning("Gzip file estimated uncompressed size: {0}.".format(size))
logger.warning("Gzip file estimated uncompressed size: %d", size)

return size


def debug(level=logging.DEBUG):
"""
Turn on the debugging
"""
logging.basicConfig(
level=level,
format="%(message)s",
datefmt="[%X]",
handlers=[RichHandler(console=Console(stderr=True))],
)


debug()


def main():
actions = (
("less", "enhance the unix `less` command"),
Expand Down Expand Up @@ -1603,7 +1591,7 @@ def expand(args):
for a in args:
oa = a.replace("/", "_")
if oa in seen:
logging.debug("Name collision `{0}`, ignored.".format(oa))
logger.debug("Name collision `%s`, ignored", oa)
continue

cmd = "cp -s" if opts.symlink else "mv"
Expand Down Expand Up @@ -1995,7 +1983,7 @@ def notify(args):
opts, args = p.parse_args(args)

if len(args) == 0:
logging.error("Please provide a brief message to be sent")
logger.error("Please provide a brief message to be sent")
sys.exit(not p.print_help())

subject = opts.subject
Expand All @@ -2005,7 +1993,7 @@ def notify(args):
toaddr = opts.email.split(",") # TO address should be in a list
for addr in toaddr:
if not is_valid_email(addr):
logging.debug("Email address `{0}` is not valid!".format(addr))
logger.debug("Email address `%s` is not valid!", addr)
sys.exit()
send_email(fromaddr, toaddr, subject, message)
else:
Expand Down Expand Up @@ -2165,7 +2153,7 @@ def waitpid(args):
msg = check_output(shlex.split(get_origcmd)).strip()
_waitpid(pid, interval=opts.interval)
else:
logging.debug("Process with PID {0} does not exist".format(pid))
logger.debug("Process with PID %d does not exist", pid)
sys.exit()

if opts.notify:
Expand All @@ -2188,9 +2176,9 @@ def get_config(path):
config.read(path)
except ParsingError:
e = sys.exc_info()[1]
logging.error(
"There was a problem reading or parsing "
"your credentials file: %s" % (e.args[0],),
logger.error(
"There was a problem reading or parsing your credentials file: %s",
e.args[0],
)
return config

Expand Down Expand Up @@ -2241,16 +2229,16 @@ def getpath(
else:
err_msg = f"Cannot execute binary `{path}`. Please verify and rerun."
if warn == "exit":
logging.fatal(err_msg)
logger.fatal(err_msg)
else:
logging.warning(err_msg)
logger.warning(err_msg)
return None

if changed:
configfile = open(cfg, "w")
config.write(configfile)
configfile.close()
logging.debug("Configuration written to `{0}`.".format(cfg))
logger.debug("Configuration written to `%s`", cfg)

return path

Expand Down
10 changes: 5 additions & 5 deletions jcvi/compara/base.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import logging
from collections import defaultdict

from ..apps.base import logger
from ..formats.base import BaseFile, read_block, must_open
from ..utils.range import Range

Expand Down Expand Up @@ -69,9 +69,9 @@ def print_to_file(self, filename="stdout", accepted=None):
print("\t".join((a, b, score)), file=fw)
fw.close()

logging.debug("Removed %d existing anchors.", nremoved)
logging.debug("Corrected scores for %d anchors.", ncorrected)
logging.debug("Anchors written to `%s`.", filename)
logger.debug("Removed %d existing anchors", nremoved)
logger.debug("Corrected scores for %d anchors", ncorrected)
logger.debug("Anchors written to `%s`", filename)

def blast(self, blastfile=None, outfile=None):
"""
Expand Down Expand Up @@ -102,7 +102,7 @@ def blast(self, blastfile=None, outfile=None):
nlines += 1
fw.close()

logging.debug("A total of %d BLAST lines written to `%s`.", nlines, outfile)
logger.debug("A total of %d BLAST lines written to `%s`", nlines, outfile)

return outfile

Expand Down
Loading

0 comments on commit 7e4c5e5

Please sign in to comment.