All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.lenskit.cli.util.Logging Maven / Gradle / Ivy

/*
 * LensKit, an open source recommender systems toolkit.
 * Copyright 2010-2014 LensKit Contributors.  See CONTRIBUTORS.md.
 * Work on LensKit has been funded by the National Science Foundation under
 * grants IIS 05-34939, 08-08692, 08-12148, and 10-17697.
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as
 * published by the Free Software Foundation; either version 2.1 of the
 * License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
 * FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
 * details.
 *
 * You should have received a copy of the GNU General Public License along with
 * this program; if not, write to the Free Software Foundation, Inc., 51
 * Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
 */
package org.lenskit.cli.util;

import ch.qos.logback.classic.Level;
import ch.qos.logback.classic.Logger;
import ch.qos.logback.classic.LoggerContext;
import ch.qos.logback.classic.encoder.PatternLayoutEncoder;
import ch.qos.logback.classic.filter.ThresholdFilter;
import ch.qos.logback.classic.spi.ILoggingEvent;
import ch.qos.logback.core.ConsoleAppender;
import ch.qos.logback.core.FileAppender;
import net.sourceforge.argparse4j.impl.Arguments;
import net.sourceforge.argparse4j.inf.ArgumentGroup;
import net.sourceforge.argparse4j.inf.ArgumentParser;
import net.sourceforge.argparse4j.inf.Namespace;
import org.fusesource.jansi.AnsiConsole;
import org.slf4j.LoggerFactory;

import java.io.File;

/**
 * CLI support for configuring the logging infrastructure.
 *
 * @see org.lenskit.cli.Main
 */
public final class Logging {
    private Logging() {}

    private static final String CONSOLE_PATTERN =
            "%highlight(%-5level) %white(%date{HH:mm:ss.SSS}) [%yellow(%thread)] " +
            "%cyan(%logger{24}) %msg%n";
    private static final String FILE_PATTERN =
            "%date{HH:mm:ss.SSS} %level [%thread] %logger: %msg%n";

    public static void addLoggingGroup(ArgumentParser parser) {
        ArgumentGroup logging = parser.addArgumentGroup("logging")
                                      .description("Control the logging output.");
        logging.addArgument("--log-file")
               .type(File.class)
               .metavar("FILE")
               .help("write logging output to FILE");
        logging.addArgument("--log-level")
               .type(String.class)
               .metavar("LEVEL")
               .help("include logging messages at LEVEL in log output");
        logging.addArgument("--log-file-level")
               .type(String.class)
               .metavar("LEVEL")
               .help("include logging messages at LEVEL in log file (defaults to --log-level value)");
        logging.addArgument("--debug-grapht")
               .action(Arguments.storeTrue())
               .help("include debug output from Grapht");
    }

    public static void configureLogging(Namespace options) {
        // if the user has explicitly configured a Logback config, honor that.
        if (System.getProperty("logback.configurationFile") != null) {
            return;
        }
        boolean debugGrapht = options.getBoolean("debug_grapht");
        File logFile = options.get("log_file");

        String lstr = options.getString("log_level");
        Level logLevel = Level.toLevel(lstr, Level.INFO);

        LoggerContext context = (LoggerContext) LoggerFactory.getILoggerFactory();
        Logger root = context.getLogger(Logger.ROOT_LOGGER_NAME);
        root.detachAndStopAllAppenders();

        // manually run JAnsi setup to use its terminal detection logic
        AnsiConsole.systemInstall();
        ConsoleAppender console = new ConsoleAppender<>();
        console.setContext(context);
        console.setTarget("System.err");
        PatternLayoutEncoder consolePat = new PatternLayoutEncoder();
        consolePat.setContext(context);
        consolePat.setPattern(CONSOLE_PATTERN);
        consolePat.start();
        console.setEncoder(consolePat);
        root.addAppender(console);

        Level rootLevel = logLevel;

        if (logFile != null) {
            // sort out the log level situation
            String lfstr = options.getString("log_file_level");
            Level logFileLevel = Level.toLevel(lfstr, logLevel);

            if (!logFileLevel.equals(logLevel)) {
                // filter the console log
                ThresholdFilter filter = new ThresholdFilter();
                filter.setContext(context);
                filter.setLevel(logLevel.toString());
                filter.start();
                console.addFilter(filter);

                // root level needs to be decreased
                if (logLevel.isGreaterOrEqual(logFileLevel)) {
                    rootLevel = logFileLevel;
                }
            }

            FileAppender fileOutput = new FileAppender<>();
            fileOutput.setAppend(false);
            fileOutput.setContext(context);
            fileOutput.setFile(logFile.getAbsolutePath());
            PatternLayoutEncoder filePat = new PatternLayoutEncoder();
            filePat.setContext(context);
            filePat.setPattern(FILE_PATTERN);
            filePat.start();
            fileOutput.setEncoder(filePat);

            // filter the file output
            ThresholdFilter filter = new ThresholdFilter();
            filter.setContext(context);
            filter.setLevel(logFileLevel.toString());
            filter.start();
            fileOutput.addFilter(filter);

            fileOutput.start();

            root.addAppender(fileOutput);
        }

        // set root level to min needed to pass a filter
        root.setLevel(rootLevel);

        console.start();

        // tone down Grapht logging
        if (!debugGrapht) {
            context.getLogger("org.grouplens.grapht").setLevel(Level.WARN);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy