
io.smallrye.config.SmallRyeConfigProviderResolver Maven / Gradle / Ivy
/*
* Copyright 2017 Red Hat, Inc.
*
* Licensed 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 io.smallrye.config;
import static io.smallrye.config.SecuritySupport.getContextClassLoader;
import java.security.AccessController;
import java.security.PrivilegedAction;
import java.util.Iterator;
import java.util.Map;
import java.util.ServiceLoader;
import java.util.concurrent.ConcurrentHashMap;
import org.eclipse.microprofile.config.Config;
import org.eclipse.microprofile.config.spi.ConfigProviderResolver;
/**
* @author Jeff Mesnil (c) 2017 Red Hat inc.
* @author David M. Lloyd
*/
public class SmallRyeConfigProviderResolver extends ConfigProviderResolver {
/**
* @deprecated The instance stored in this field will generally not match the one instantiated by
* {@link ConfigProviderResolver}; thus this instance should not normally be used and may be removed
* from a future release. Instead use {@link ConfigProviderResolver#instance()} to get the instance.
*/
@Deprecated
public static final SmallRyeConfigProviderResolver INSTANCE = new SmallRyeConfigProviderResolver();
private final Map configsForClassLoader = new ConcurrentHashMap<>();
static final ClassLoader SYSTEM_CL;
static {
final SecurityManager sm = System.getSecurityManager();
if (sm != null) {
SYSTEM_CL = AccessController
.doPrivileged((PrivilegedAction) SmallRyeConfigProviderResolver::calculateSystemClassLoader);
} else {
SYSTEM_CL = calculateSystemClassLoader();
}
}
public SmallRyeConfigProviderResolver() {
}
private static ClassLoader calculateSystemClassLoader() {
ClassLoader cl = ClassLoader.getSystemClassLoader();
if (cl == null) {
// non-null ref that delegates to the system
cl = new ClassLoader(null) {
};
}
return cl;
}
@Override
public Config getConfig() {
return getConfig(getContextClassLoader());
}
@Override
public Config getConfig(ClassLoader classLoader) {
final ClassLoader realClassLoader = getRealClassLoader(classLoader);
final Map configsForClassLoader = this.configsForClassLoader;
Config config = configsForClassLoader.get(realClassLoader);
if (config == null) {
synchronized (configsForClassLoader) {
config = configsForClassLoader.get(realClassLoader);
if (config == null) {
config = getFactoryFor(realClassLoader, false).getConfigFor(this, classLoader);
// don't cache null, as that would leak class loaders
if (config == null) {
throw ConfigMessages.msg.noConfigForClassloader();
}
configsForClassLoader.put(realClassLoader, config);
}
}
}
return config;
}
SmallRyeConfigFactory getFactoryFor(final ClassLoader classLoader, final boolean privileged) {
final SecurityManager sm = System.getSecurityManager();
if (sm != null && !privileged) {
// run privileged so that the only things on the access control stack are us and the provider
return AccessController.doPrivileged(new PrivilegedAction() {
public SmallRyeConfigFactory run() {
return getFactoryFor(classLoader, true);
}
});
}
final ServiceLoader serviceLoader = ServiceLoader.load(SmallRyeConfigFactory.class, classLoader);
final Iterator iterator = serviceLoader.iterator();
return iterator.hasNext() ? iterator.next() : SmallRyeConfigFactory.Default.INSTANCE;
}
@Override
public SmallRyeConfigBuilder getBuilder() {
return new SmallRyeConfigBuilder().addDefaultInterceptors();
}
@Override
public void registerConfig(Config config, ClassLoader classLoader) {
if (config == null) {
throw ConfigMessages.msg.configIsNull();
}
final ClassLoader realClassLoader = getRealClassLoader(classLoader);
final Map configsForClassLoader = this.configsForClassLoader;
synchronized (configsForClassLoader) {
final Config existing = configsForClassLoader.putIfAbsent(realClassLoader, config);
if (existing != null) {
throw ConfigMessages.msg.configAlreadyRegistered();
}
}
}
@Override
public void releaseConfig(Config config) {
// todo: see https://github.com/eclipse/microprofile-config/issues/136#issuecomment-535962313
// todo: see https://github.com/eclipse/microprofile-config/issues/471
final Map configsForClassLoader = this.configsForClassLoader;
synchronized (configsForClassLoader) {
configsForClassLoader.values().removeIf(v -> v == config);
}
}
static ClassLoader getRealClassLoader(ClassLoader classLoader) {
if (classLoader == null) {
classLoader = getContextClassLoader();
}
if (classLoader == null) {
classLoader = SYSTEM_CL;
}
return classLoader;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy