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

org.apache.log4j.builders.BuilderManager Maven / Gradle / Ivy

/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements. See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You under the Apache license, Version 2.0
 * (the "License"); you may not use this file except in compliance with
 * the License. You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the license for the specific language governing permissions and
 * limitations under the license.
 */
package org.apache.log4j.builders;

import java.util.Locale;
import java.util.Map;
import java.util.Objects;
import java.util.Properties;
import java.util.function.Function;

import org.apache.log4j.Appender;
import org.apache.log4j.Layout;
import org.apache.log4j.bridge.AppenderWrapper;
import org.apache.log4j.bridge.FilterWrapper;
import org.apache.log4j.bridge.LayoutWrapper;
import org.apache.log4j.bridge.RewritePolicyWrapper;
import org.apache.log4j.builders.appender.AppenderBuilder;
import org.apache.log4j.builders.filter.FilterBuilder;
import org.apache.log4j.builders.layout.LayoutBuilder;
import org.apache.log4j.builders.rewrite.RewritePolicyBuilder;
import org.apache.log4j.builders.rolling.TriggeringPolicyBuilder;
import org.apache.log4j.config.PropertiesConfiguration;
import org.apache.log4j.rewrite.RewritePolicy;
import org.apache.log4j.spi.Filter;
import org.apache.log4j.xml.XmlConfiguration;
import org.apache.logging.log4j.Logger;
import org.apache.logging.log4j.core.appender.rolling.TriggeringPolicy;
import org.apache.logging.log4j.core.config.plugins.util.PluginManager;
import org.apache.logging.log4j.core.config.plugins.util.PluginType;
import org.apache.logging.log4j.status.StatusLogger;
import org.apache.logging.log4j.util.LoaderUtil;
import org.w3c.dom.Element;

/**
 *
 */
public class BuilderManager {

    /** Plugin category. */
    public static final String CATEGORY = "Log4j Builder";
    public static final Appender INVALID_APPENDER = new AppenderWrapper(null);
    public static final Filter INVALID_FILTER = new FilterWrapper(null);
    public static final Layout INVALID_LAYOUT = new LayoutWrapper(null);
    public static final RewritePolicy INVALID_REWRITE_POLICY = new RewritePolicyWrapper(null);

    private static final Logger LOGGER = StatusLogger.getLogger();
    private static Class[] CONSTRUCTOR_PARAMS = new Class[] {String.class, Properties.class};
    private final Map> plugins;

    /**
     * Constructs a new instance.
     */
    public BuilderManager() {
        final PluginManager manager = new PluginManager(CATEGORY);
        manager.collectPlugins();
        plugins = manager.getPlugins();
    }

    private , U> T createBuilder(final PluginType plugin, final String prefix, final Properties props) {
        if (plugin == null) {
            return null;
        }
        try {
            final Class clazz = plugin.getPluginClass();
            if (AbstractBuilder.class.isAssignableFrom(clazz)) {
                return clazz.getConstructor(CONSTRUCTOR_PARAMS).newInstance(prefix, props);
            }
            final T builder = LoaderUtil.newInstanceOf(clazz);
            // Reasonable message instead of `ClassCastException`
            if (!Builder.class.isAssignableFrom(clazz)) {
                LOGGER.warn("Unable to load plugin: builder {} does not implement {}", clazz, Builder.class);
                return null;
            }
            return builder;
        } catch (final ReflectiveOperationException ex) {
            LOGGER.warn("Unable to load plugin: {} due to: {}", plugin.getKey(), ex.getMessage());
            return null;
        }
    }

    @SuppressWarnings("unchecked")
    private  PluginType getPlugin(final String className) {
        Objects.requireNonNull(plugins, "plugins");
        Objects.requireNonNull(className, "className");
        final String key = className.toLowerCase(Locale.ROOT).trim();
        final PluginType pluginType = plugins.get(key);
        if (pluginType == null) {
            LOGGER.warn("Unable to load plugin class name {} with key {}", className, key);
        }
        return (PluginType) pluginType;
    }

    private , U> U newInstance(final PluginType plugin, final Function consumer,
            final U invalidValue) {
        if (plugin != null) {
            try {
                final T builder = LoaderUtil.newInstanceOf(plugin.getPluginClass());
                if (builder != null) {
                    U result = consumer.apply(builder);
                    // returning an empty wrapper is short for "we support this legacy class, but it has validation errors"
                    return result != null ? result : invalidValue;
                }
            } catch (final ReflectiveOperationException ex) {
                LOGGER.warn("Unable to load plugin: {} due to: {}", plugin.getKey(), ex.getMessage());
            }
        }
        return null;
    }

    public 

, T> T parse(final String className, final String prefix, final Properties props, final PropertiesConfiguration config, T invalidValue) { final P parser = createBuilder(getPlugin(className), prefix, props); if (parser != null) { final T value = parser.parse(config); return value != null ? value : invalidValue; } return null; } public Appender parseAppender(final String className, final Element appenderElement, final XmlConfiguration config) { return newInstance(this.>getPlugin(className), b -> b.parseAppender(appenderElement, config), INVALID_APPENDER); } public Appender parseAppender(final String name, final String className, final String prefix, final String layoutPrefix, final String filterPrefix, final Properties props, final PropertiesConfiguration config) { final AppenderBuilder builder = createBuilder(getPlugin(className), prefix, props); if (builder != null) { final Appender appender = builder.parseAppender(name, prefix, layoutPrefix, filterPrefix, props, config); return appender != null ? appender : INVALID_APPENDER; } return null; } public Filter parseFilter(final String className, final Element filterElement, final XmlConfiguration config) { return newInstance(this.getPlugin(className), b -> b.parse(filterElement, config), INVALID_FILTER); } public Layout parseLayout(final String className, final Element layoutElement, final XmlConfiguration config) { return newInstance(this.getPlugin(className), b -> b.parse(layoutElement, config), INVALID_LAYOUT); } public RewritePolicy parseRewritePolicy(final String className, final Element rewriteElement, final XmlConfiguration config) { return newInstance(this.getPlugin(className), b -> b.parse(rewriteElement, config), INVALID_REWRITE_POLICY); } public TriggeringPolicy parseTriggeringPolicy(final String className, final Element policyElement, final XmlConfiguration config) { return newInstance(this.getPlugin(className), b -> b.parse(policyElement, config), (TriggeringPolicy) null); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy