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

org.apache.camel.impl.engine.DefaultFactoryFinder 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.camel.impl.engine;

import java.io.BufferedInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.Optional;
import java.util.Properties;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.function.Function;

import org.apache.camel.RuntimeCamelException;
import org.apache.camel.spi.ClassResolver;
import org.apache.camel.spi.FactoryFinder;
import org.apache.camel.support.ObjectHelper;
import org.apache.camel.util.IOHelper;

/**
 * Default factory finder.
 */
public class DefaultFactoryFinder implements FactoryFinder {

    protected ConcurrentMap> classMap = new ConcurrentHashMap<>();
    protected ConcurrentMap classesNotFound = new ConcurrentHashMap<>();
    protected ConcurrentMap classesNotFoundExceptions = new ConcurrentHashMap<>();
    protected ClassResolver classResolver;
    private final String path;

    public DefaultFactoryFinder(ClassResolver classResolver, String resourcePath) {
        this.classResolver = classResolver;
        this.path = resourcePath;
    }

    @Override
    public String getResourcePath() {
        return path;
    }

    @Override
    public Optional newInstance(String key) {
        return Optional.ofNullable(doNewInstance(key));
    }

    @Override
    public  Optional newInstance(String key, Class type) {
        Object obj = doNewInstance(key);
        return Optional.ofNullable(type.cast(obj));
    }

    @Override
    public Optional> findClass(String key) {
        Class clazz = addToClassMap(key, () -> {
            Properties prop = doFindFactoryProperties(key);
            if (prop != null) {
                return doNewInstance(prop, true).orElse(null);
            } else {
                return null;
            }
        });
        return Optional.ofNullable(clazz);
    }

    @Override
    public Optional> findOptionalClass(String key) {
        Class clazz = addToClassMap(key, () -> {
            Properties prop = doFindFactoryProperties(key);
            if (prop != null) {
                return doNewInstance(prop, false).orElse(null);
            } else {
                return null;
            }
        });
        return Optional.ofNullable(clazz);
    }

    @Override
    public void clear() {
        if (classMap != null) {
            classMap.clear();
        }
        if (classesNotFound != null) {
            classesNotFound.clear();
        }
        if (classesNotFoundExceptions != null) {
            classesNotFoundExceptions.clear();
        }
    }

    private Object doNewInstance(String key) {
        Optional> clazz = findClass(key);
        return clazz.map(ObjectHelper::newInstance).orElse(null);
    }

    private Optional> doNewInstance(Properties properties, boolean mandatory) throws IOException {
        String className = properties.getProperty("class");
        if (className == null && mandatory) {
            throw new IOException("Expected property is missing: class");
        } else if (className == null) {
            return Optional.empty();
        }

        Class clazz = classResolver.resolveClass(className);
        return Optional.ofNullable(clazz);
    }

    private Properties doFindFactoryProperties(String key) throws IOException {
        String uri = path + key;

        InputStream in = classResolver.loadResourceAsStream(uri);
        if (in == null) {
            return null;
        }

        // lets load the file
        BufferedInputStream reader = null;
        try {
            reader = IOHelper.buffered(in);
            Properties properties = new Properties();
            properties.load(reader);
            return properties;
        } finally {
            IOHelper.close(reader, key, null);
            IOHelper.close(in, key, null);
        }
    }

    /*
     * This is a wrapper function to deal with exceptions in lambdas: the exception
     * is wrapped by a runtime exception (WrappedRuntimeException) which we catch
     * later on with the only purpose to re-throw the original exception.
     */
    protected Class addToClassMap(String key, ClassSupplier mappingFunction) {
        if (classesNotFound.containsKey(key) || classesNotFoundExceptions.containsKey(key)) {
            Exception e = classesNotFoundExceptions.get(key);
            if (e == null) {
                return null;
            } else {
                throw RuntimeCamelException.wrapRuntimeException(e);
            }
        }

        Class suppliedClass = classMap.computeIfAbsent(key, new Function<>() {
            @Override
            public Class apply(String classKey) {
                try {
                    return mappingFunction.get();
                } catch (Exception e) {
                    classesNotFoundExceptions.put(key, e);
                    throw RuntimeCamelException.wrapRuntimeException(e);
                }
            }
        });

        if (suppliedClass == null) {
            // mark the key as non-resolvable to prevent pointless searching
            classesNotFound.put(key, Boolean.TRUE);
        }

        return suppliedClass;
    }

    @FunctionalInterface
    protected interface ClassSupplier {
        Class get() throws Exception;
    }

}