blob: 6dbaed10be7976c36c85901621b6ff656adc81ef [file] [log] [blame]
/*
* 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.rat.configuration;
import java.lang.reflect.InvocationTargetException;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import java.util.Objects;
import org.apache.commons.lang3.StringUtils;
import org.apache.commons.text.WordUtils;
import org.apache.rat.ConfigurationException;
import org.apache.rat.Defaults;
import org.apache.rat.configuration.builders.AbstractBuilder;
/**
* A class to track the Matcher Builders as they are defined. Matchers may be defined in multiple configuration files
* this method tracks them so that they can be referenced across the configuration files.
*/
public final class MatcherBuilderTracker {
/** The instance of the BuildTracker. */
public static MatcherBuilderTracker INSTANCE;
private final Map<String, Class<? extends AbstractBuilder>> matcherBuilders;
private static synchronized MatcherBuilderTracker instance() {
if (INSTANCE == null) {
INSTANCE = new MatcherBuilderTracker();
Defaults.init();
}
return INSTANCE;
}
/**
* Adds a builder to the tracker.
* If the {@code name} is null then the builder class name simple is used with the "Builder" suffix removed.
* @param className the Class name for the builder.
* @param name the short name for the builder.
*/
public static void addBuilder(final String className, final String name) {
instance().addBuilderImpl(className, name);
}
/**
* Get the matching builder for the name.
* @param name The name of the builder.
* @return the builder for that name.
*/
public static AbstractBuilder getMatcherBuilder(final String name) {
Class<? extends AbstractBuilder> clazz = instance().matcherBuilders.get(name);
if (clazz == null) {
StringBuilder sb = new StringBuilder(System.lineSeparator()).append("Valid builders").append(System.lineSeparator());
instance().matcherBuilders.keySet().forEach(x -> sb.append(x).append(System.lineSeparator()));
sb.append("ERROR MSG").append(System.lineSeparator());
throw new ConfigurationException(sb.append("No matcher builder named ").append(name).toString());
}
try {
return clazz.getConstructor().newInstance();
} catch (NoSuchMethodException | SecurityException | InstantiationException | IllegalAccessException
| IllegalArgumentException | InvocationTargetException e) {
throw new ConfigurationException(
String.format("Can not instantiate matcher builder named %s (%s)", name, clazz.getName()), e);
}
}
private MatcherBuilderTracker() {
matcherBuilders = new HashMap<>();
}
/**
* Gets a collection of classes that are recognized as builders.
* @return the collection of builder classes
*/
public Collection<Class<? extends AbstractBuilder>> getClasses() {
return Collections.unmodifiableCollection(matcherBuilders.values());
}
private void addBuilderImpl(final String className, String name) {
Objects.requireNonNull(className, "className may not be null");
Class<?> clazz;
try {
clazz = getClass().getClassLoader().loadClass(className);
} catch (ClassNotFoundException e) {
throw new ConfigurationException(e);
}
if (AbstractBuilder.class.isAssignableFrom(clazz)) {
@SuppressWarnings("unchecked")
Class<? extends AbstractBuilder> candidate = (Class<? extends AbstractBuilder>) clazz;
if (StringUtils.isBlank(name)) {
name = candidate.getSimpleName();
if (!name.endsWith("Builder")) {
throw new ConfigurationException(
"name is required, or " + candidate.getName() + " must end with 'Builder'");
}
name = name.substring(0, name.lastIndexOf("Builder"));
if (StringUtils.isBlank(name)) {
throw new ConfigurationException("Last segment of " + candidate.getName()
+ " may not be 'Builder', but must end in 'Builder'");
}
name = WordUtils.uncapitalize(name);
}
matcherBuilders.put(name, candidate);
} else {
throw new ConfigurationException("Class " + clazz.getName() + " does not extend " + AbstractBuilder.class);
}
}
}