Avoid use of SimpleNamingContextBuilder as it pollutes JVM’s JNDI config
Closes gh-2397
This commit is contained in:
parent
82d1e61ab1
commit
5922cfa41f
|
@ -16,23 +16,17 @@
|
||||||
|
|
||||||
package org.springframework.boot.autoconfigure.condition;
|
package org.springframework.boot.autoconfigure.condition;
|
||||||
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.net.URL;
|
|
||||||
import java.util.Collections;
|
|
||||||
import java.util.Enumeration;
|
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.Hashtable;
|
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
import javax.naming.Context;
|
import javax.naming.Context;
|
||||||
import javax.naming.InitialContext;
|
|
||||||
import javax.naming.NamingException;
|
|
||||||
import javax.naming.spi.InitialContextFactory;
|
|
||||||
|
|
||||||
import org.hamcrest.Matcher;
|
import org.hamcrest.Matcher;
|
||||||
import org.junit.After;
|
import org.junit.After;
|
||||||
import org.junit.Before;
|
import org.junit.Before;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
import org.springframework.boot.autoconfigure.jndi.JndiPropertiesHidingClassLoader;
|
||||||
|
import org.springframework.boot.autoconfigure.jndi.TestableInitialContextFactory;
|
||||||
import org.springframework.boot.test.EnvironmentTestUtils;
|
import org.springframework.boot.test.EnvironmentTestUtils;
|
||||||
import org.springframework.context.ConfigurableApplicationContext;
|
import org.springframework.context.ConfigurableApplicationContext;
|
||||||
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
|
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
|
||||||
|
@ -206,92 +200,4 @@ public class ConditionalOnJndiTests {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static class TestableInitialContextFactory implements InitialContextFactory {
|
|
||||||
|
|
||||||
private static TestableContext context;
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Context getInitialContext(Hashtable<?, ?> environment)
|
|
||||||
throws NamingException {
|
|
||||||
return getContext();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void bind(String name, Object obj) {
|
|
||||||
try {
|
|
||||||
getContext().bind(name, obj);
|
|
||||||
}
|
|
||||||
catch (NamingException ex) {
|
|
||||||
throw new IllegalStateException(ex);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void clearAll() {
|
|
||||||
getContext().clearAll();
|
|
||||||
}
|
|
||||||
|
|
||||||
private static TestableContext getContext() {
|
|
||||||
if (context == null) {
|
|
||||||
try {
|
|
||||||
context = new TestableContext();
|
|
||||||
}
|
|
||||||
catch (NamingException ex) {
|
|
||||||
throw new IllegalStateException(ex);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return context;
|
|
||||||
}
|
|
||||||
|
|
||||||
private static class TestableContext extends InitialContext {
|
|
||||||
|
|
||||||
private final Map<String, Object> bindings = new HashMap<String, Object>();
|
|
||||||
|
|
||||||
private TestableContext() throws NamingException {
|
|
||||||
super(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void bind(String name, Object obj) throws NamingException {
|
|
||||||
this.bindings.put(name, obj);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Object lookup(String name) throws NamingException {
|
|
||||||
return this.bindings.get(name);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Hashtable<?, ?> getEnvironment() throws NamingException {
|
|
||||||
return new Hashtable<Object, Object>(); // Used to detect if JNDI is
|
|
||||||
// available
|
|
||||||
}
|
|
||||||
|
|
||||||
public void clearAll() {
|
|
||||||
this.bindings.clear();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Used as the thread context classloader to prevent jndi.properties resources found
|
|
||||||
* on the classpath from triggering configuration of an InitialContextFactory that is
|
|
||||||
* outside the control of these tests.
|
|
||||||
*/
|
|
||||||
private static class JndiPropertiesHidingClassLoader extends ClassLoader {
|
|
||||||
|
|
||||||
public JndiPropertiesHidingClassLoader(ClassLoader parent) {
|
|
||||||
super(parent);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Enumeration<URL> getResources(String name) throws IOException {
|
|
||||||
if ("jndi.properties".equals(name)) {
|
|
||||||
return Collections.enumeration(Collections.<URL> emptyList());
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
return super.getResources(name);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -18,18 +18,21 @@ package org.springframework.boot.autoconfigure.jdbc;
|
||||||
|
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
|
||||||
|
import javax.naming.Context;
|
||||||
import javax.naming.NamingException;
|
import javax.naming.NamingException;
|
||||||
import javax.sql.DataSource;
|
import javax.sql.DataSource;
|
||||||
|
|
||||||
import org.apache.commons.dbcp2.BasicDataSource;
|
import org.apache.commons.dbcp2.BasicDataSource;
|
||||||
import org.junit.After;
|
import org.junit.After;
|
||||||
|
import org.junit.Before;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
import org.springframework.beans.DirectFieldAccessor;
|
import org.springframework.beans.DirectFieldAccessor;
|
||||||
|
import org.springframework.boot.autoconfigure.jndi.JndiPropertiesHidingClassLoader;
|
||||||
|
import org.springframework.boot.autoconfigure.jndi.TestableInitialContextFactory;
|
||||||
import org.springframework.boot.test.EnvironmentTestUtils;
|
import org.springframework.boot.test.EnvironmentTestUtils;
|
||||||
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
|
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
|
||||||
import org.springframework.context.annotation.Bean;
|
import org.springframework.context.annotation.Bean;
|
||||||
import org.springframework.jmx.export.MBeanExporter;
|
import org.springframework.jmx.export.MBeanExporter;
|
||||||
import org.springframework.mock.jndi.SimpleNamingContextBuilder;
|
|
||||||
|
|
||||||
import static org.hamcrest.Matchers.contains;
|
import static org.hamcrest.Matchers.contains;
|
||||||
import static org.hamcrest.Matchers.hasSize;
|
import static org.hamcrest.Matchers.hasSize;
|
||||||
|
@ -43,25 +46,47 @@ import static org.junit.Assert.assertThat;
|
||||||
*/
|
*/
|
||||||
public class JndiDataSourceAutoConfigurationTests {
|
public class JndiDataSourceAutoConfigurationTests {
|
||||||
|
|
||||||
|
private ClassLoader threadContextClassLoader;
|
||||||
|
|
||||||
|
private String initialContextFactory;
|
||||||
|
|
||||||
private AnnotationConfigApplicationContext context;
|
private AnnotationConfigApplicationContext context;
|
||||||
|
|
||||||
private SimpleNamingContextBuilder jndi;
|
@Before
|
||||||
|
public void setupJndi() {
|
||||||
|
this.initialContextFactory = System.getProperty(Context.INITIAL_CONTEXT_FACTORY);
|
||||||
|
System.setProperty(Context.INITIAL_CONTEXT_FACTORY,
|
||||||
|
TestableInitialContextFactory.class.getName());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void setupThreadContextClassLoader() {
|
||||||
|
this.threadContextClassLoader = Thread.currentThread().getContextClassLoader();
|
||||||
|
Thread.currentThread().setContextClassLoader(
|
||||||
|
new JndiPropertiesHidingClassLoader(getClass().getClassLoader()));
|
||||||
|
}
|
||||||
|
|
||||||
@After
|
@After
|
||||||
public void cleanup() {
|
public void close() {
|
||||||
if (this.jndi != null) {
|
TestableInitialContextFactory.clearAll();
|
||||||
this.jndi.clear();
|
if (this.initialContextFactory != null) {
|
||||||
|
System.setProperty(Context.INITIAL_CONTEXT_FACTORY,
|
||||||
|
this.initialContextFactory);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
System.clearProperty(Context.INITIAL_CONTEXT_FACTORY);
|
||||||
}
|
}
|
||||||
if (this.context != null) {
|
if (this.context != null) {
|
||||||
this.context.close();
|
this.context.close();
|
||||||
}
|
}
|
||||||
|
Thread.currentThread().setContextClassLoader(this.threadContextClassLoader);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void dataSourceIsAvailableFromJndi() throws IllegalStateException,
|
public void dataSourceIsAvailableFromJndi() throws IllegalStateException,
|
||||||
NamingException {
|
NamingException {
|
||||||
DataSource dataSource = new BasicDataSource();
|
DataSource dataSource = new BasicDataSource();
|
||||||
this.jndi = configureJndi("foo", dataSource);
|
configureJndi("foo", dataSource);
|
||||||
|
|
||||||
this.context = new AnnotationConfigApplicationContext();
|
this.context = new AnnotationConfigApplicationContext();
|
||||||
EnvironmentTestUtils.addEnvironment(this.context,
|
EnvironmentTestUtils.addEnvironment(this.context,
|
||||||
|
@ -77,7 +102,7 @@ public class JndiDataSourceAutoConfigurationTests {
|
||||||
public void mbeanDataSourceIsExcludedFromExport() throws IllegalStateException,
|
public void mbeanDataSourceIsExcludedFromExport() throws IllegalStateException,
|
||||||
NamingException {
|
NamingException {
|
||||||
DataSource dataSource = new BasicDataSource();
|
DataSource dataSource = new BasicDataSource();
|
||||||
this.jndi = configureJndi("foo", dataSource);
|
configureJndi("foo", dataSource);
|
||||||
|
|
||||||
this.context = new AnnotationConfigApplicationContext();
|
this.context = new AnnotationConfigApplicationContext();
|
||||||
EnvironmentTestUtils.addEnvironment(this.context,
|
EnvironmentTestUtils.addEnvironment(this.context,
|
||||||
|
@ -98,7 +123,7 @@ public class JndiDataSourceAutoConfigurationTests {
|
||||||
public void standardDataSourceIsNotExcludedFromExport() throws IllegalStateException,
|
public void standardDataSourceIsNotExcludedFromExport() throws IllegalStateException,
|
||||||
NamingException {
|
NamingException {
|
||||||
DataSource dataSource = new org.apache.commons.dbcp.BasicDataSource();
|
DataSource dataSource = new org.apache.commons.dbcp.BasicDataSource();
|
||||||
this.jndi = configureJndi("foo", dataSource);
|
configureJndi("foo", dataSource);
|
||||||
|
|
||||||
this.context = new AnnotationConfigApplicationContext();
|
this.context = new AnnotationConfigApplicationContext();
|
||||||
EnvironmentTestUtils.addEnvironment(this.context,
|
EnvironmentTestUtils.addEnvironment(this.context,
|
||||||
|
@ -114,12 +139,9 @@ public class JndiDataSourceAutoConfigurationTests {
|
||||||
assertThat(excludedBeans, hasSize(0));
|
assertThat(excludedBeans, hasSize(0));
|
||||||
}
|
}
|
||||||
|
|
||||||
private SimpleNamingContextBuilder configureJndi(String name, DataSource dataSource)
|
private void configureJndi(String name, DataSource dataSource)
|
||||||
throws IllegalStateException, NamingException {
|
throws IllegalStateException, NamingException {
|
||||||
SimpleNamingContextBuilder builder = SimpleNamingContextBuilder
|
TestableInitialContextFactory.bind(name, dataSource);
|
||||||
.emptyActivatedContextBuilder();
|
|
||||||
builder.bind(name, dataSource);
|
|
||||||
return builder;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private static class MBeanExporterConfiguration {
|
private static class MBeanExporterConfiguration {
|
||||||
|
|
|
@ -0,0 +1,46 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2015 the original author or authors.
|
||||||
|
*
|
||||||
|
* 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 org.springframework.boot.autoconfigure.jndi;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.net.URL;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.Enumeration;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Used as the thread context classloader to prevent {@code jndi.properties} resources
|
||||||
|
* found on the classpath from triggering configuration of an InitialContextFactory.
|
||||||
|
*
|
||||||
|
* @author Andy Wilkinson
|
||||||
|
*/
|
||||||
|
public class JndiPropertiesHidingClassLoader extends ClassLoader {
|
||||||
|
|
||||||
|
public JndiPropertiesHidingClassLoader(ClassLoader parent) {
|
||||||
|
super(parent);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Enumeration<URL> getResources(String name) throws IOException {
|
||||||
|
if ("jndi.properties".equals(name)) {
|
||||||
|
return Collections.enumeration(Collections.<URL> emptyList());
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return super.getResources(name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,95 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2015 the original author or authors.
|
||||||
|
*
|
||||||
|
* 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 org.springframework.boot.autoconfigure.jndi;
|
||||||
|
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Hashtable;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import javax.naming.Context;
|
||||||
|
import javax.naming.InitialContext;
|
||||||
|
import javax.naming.NamingException;
|
||||||
|
import javax.naming.spi.InitialContextFactory;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An {@code InitialContextFactory} implementation to be used for testing JNDI.
|
||||||
|
*
|
||||||
|
* @author Stephane Nicoll
|
||||||
|
*/
|
||||||
|
public class TestableInitialContextFactory implements InitialContextFactory {
|
||||||
|
|
||||||
|
private static TestableContext context;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Context getInitialContext(Hashtable<?, ?> environment) throws NamingException {
|
||||||
|
return getContext();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void bind(String name, Object obj) {
|
||||||
|
try {
|
||||||
|
getContext().bind(name, obj);
|
||||||
|
}
|
||||||
|
catch (NamingException ex) {
|
||||||
|
throw new IllegalStateException(ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void clearAll() {
|
||||||
|
getContext().clearAll();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static TestableContext getContext() {
|
||||||
|
if (context == null) {
|
||||||
|
try {
|
||||||
|
context = new TestableContext();
|
||||||
|
}
|
||||||
|
catch (NamingException ex) {
|
||||||
|
throw new IllegalStateException(ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return context;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class TestableContext extends InitialContext {
|
||||||
|
|
||||||
|
private final Map<String, Object> bindings = new HashMap<String, Object>();
|
||||||
|
|
||||||
|
private TestableContext() throws NamingException {
|
||||||
|
super(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void bind(String name, Object obj) throws NamingException {
|
||||||
|
this.bindings.put(name, obj);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Object lookup(String name) throws NamingException {
|
||||||
|
return this.bindings.get(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Hashtable<?, ?> getEnvironment() throws NamingException {
|
||||||
|
return new Hashtable<Object, Object>(); // Used to detect if JNDI is
|
||||||
|
// available
|
||||||
|
}
|
||||||
|
|
||||||
|
public void clearAll() {
|
||||||
|
this.bindings.clear();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue