Merge branch '6.0.x'
This commit is contained in:
commit
2ed10f13e9
|
|
@ -16,8 +16,6 @@
|
||||||
|
|
||||||
package org.springframework.test.context.observation;
|
package org.springframework.test.context.observation;
|
||||||
|
|
||||||
import java.lang.reflect.Method;
|
|
||||||
|
|
||||||
import io.micrometer.observation.ObservationRegistry;
|
import io.micrometer.observation.ObservationRegistry;
|
||||||
import io.micrometer.observation.contextpropagation.ObservationThreadLocalAccessor;
|
import io.micrometer.observation.contextpropagation.ObservationThreadLocalAccessor;
|
||||||
import org.apache.commons.logging.Log;
|
import org.apache.commons.logging.Log;
|
||||||
|
|
@ -27,7 +25,6 @@ import org.springframework.context.ApplicationContext;
|
||||||
import org.springframework.core.Conventions;
|
import org.springframework.core.Conventions;
|
||||||
import org.springframework.test.context.TestContext;
|
import org.springframework.test.context.TestContext;
|
||||||
import org.springframework.test.context.support.AbstractTestExecutionListener;
|
import org.springframework.test.context.support.AbstractTestExecutionListener;
|
||||||
import org.springframework.util.ReflectionUtils;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* {@code TestExecutionListener} which provides support for Micrometer's
|
* {@code TestExecutionListener} which provides support for Micrometer's
|
||||||
|
|
@ -45,11 +42,6 @@ class MicrometerObservationRegistryTestExecutionListener extends AbstractTestExe
|
||||||
|
|
||||||
private static final Log logger = LogFactory.getLog(MicrometerObservationRegistryTestExecutionListener.class);
|
private static final Log logger = LogFactory.getLog(MicrometerObservationRegistryTestExecutionListener.class);
|
||||||
|
|
||||||
private static final String THREAD_LOCAL_ACCESSOR_CLASS_NAME = "io.micrometer.context.ThreadLocalAccessor";
|
|
||||||
|
|
||||||
private static final String OBSERVATION_THREAD_LOCAL_ACCESSOR_CLASS_NAME =
|
|
||||||
"io.micrometer.observation.contextpropagation.ObservationThreadLocalAccessor";
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Attribute name for a {@link TestContext} attribute which contains the
|
* Attribute name for a {@link TestContext} attribute which contains the
|
||||||
* {@link ObservationRegistry} that was previously stored in the
|
* {@link ObservationRegistry} that was previously stored in the
|
||||||
|
|
@ -62,41 +54,46 @@ class MicrometerObservationRegistryTestExecutionListener extends AbstractTestExe
|
||||||
private static final String PREVIOUS_OBSERVATION_REGISTRY = Conventions.getQualifiedAttributeName(
|
private static final String PREVIOUS_OBSERVATION_REGISTRY = Conventions.getQualifiedAttributeName(
|
||||||
MicrometerObservationRegistryTestExecutionListener.class, "previousObservationRegistry");
|
MicrometerObservationRegistryTestExecutionListener.class, "previousObservationRegistry");
|
||||||
|
|
||||||
|
static final String DEPENDENCIES_ERROR_MESSAGE = """
|
||||||
|
MicrometerObservationRegistryTestExecutionListener requires \
|
||||||
|
io.micrometer:micrometer-observation:1.10.8 or higher and \
|
||||||
|
io.micrometer:context-propagation:1.0.3 or higher.""";
|
||||||
|
|
||||||
public MicrometerObservationRegistryTestExecutionListener() {
|
static final String THREAD_LOCAL_ACCESSOR_CLASS_NAME = "io.micrometer.context.ThreadLocalAccessor";
|
||||||
|
|
||||||
|
static final String OBSERVATION_THREAD_LOCAL_ACCESSOR_CLASS_NAME =
|
||||||
|
"io.micrometer.observation.contextpropagation.ObservationThreadLocalAccessor";
|
||||||
|
|
||||||
|
private static final String ERROR_MESSAGE;
|
||||||
|
|
||||||
|
static {
|
||||||
// Trigger eager resolution of Micrometer Observation types to ensure that
|
// Trigger eager resolution of Micrometer Observation types to ensure that
|
||||||
// this listener can be properly skipped when SpringFactoriesLoader attempts
|
// this listener can be properly skipped when SpringFactoriesLoader attempts
|
||||||
// to load it -- for example, if context-propagation and micrometer-observation
|
// to load it -- for example, if context-propagation and micrometer-observation
|
||||||
// are not in the classpath or if the version of ObservationThreadLocalAccessor
|
// are not in the classpath or if the version of ObservationThreadLocalAccessor
|
||||||
// present does not include the getObservationRegistry() method.
|
// present does not include the getObservationRegistry() method.
|
||||||
ClassLoader classLoader = getClass().getClassLoader();
|
|
||||||
String errorMessage = """
|
|
||||||
MicrometerObservationRegistryTestExecutionListener requires \
|
|
||||||
io.micrometer:micrometer-observation:1.10.8 or higher and \
|
|
||||||
io.micrometer:context-propagation:1.0.3 or higher.""";
|
|
||||||
|
|
||||||
|
String errorMessage = null;
|
||||||
|
ClassLoader classLoader = MicrometerObservationRegistryTestExecutionListener.class.getClassLoader();
|
||||||
|
String classToCheck = THREAD_LOCAL_ACCESSOR_CLASS_NAME;
|
||||||
try {
|
try {
|
||||||
Class.forName(THREAD_LOCAL_ACCESSOR_CLASS_NAME, false, classLoader);
|
Class.forName(classToCheck, false, classLoader);
|
||||||
Class<?> clazz = Class.forName(OBSERVATION_THREAD_LOCAL_ACCESSOR_CLASS_NAME, false, classLoader);
|
classToCheck = OBSERVATION_THREAD_LOCAL_ACCESSOR_CLASS_NAME;
|
||||||
Method method = ReflectionUtils.findMethod(clazz, "getObservationRegistry");
|
Class<?> clazz = Class.forName(classToCheck, false, classLoader);
|
||||||
if (method == null) {
|
clazz.getMethod("getObservationRegistry");
|
||||||
// We simulate "class not found" even though it's "method not found", since
|
|
||||||
// the ClassNotFoundException will be processed in the subsequent catch-block.
|
|
||||||
throw new ClassNotFoundException(OBSERVATION_THREAD_LOCAL_ACCESSOR_CLASS_NAME);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
catch (Throwable ex) {
|
catch (Throwable ex) {
|
||||||
// Ensure that we throw a NoClassDefFoundError so that the exception will be properly
|
errorMessage = classToCheck + ". " + DEPENDENCIES_ERROR_MESSAGE;
|
||||||
// handled in TestContextFailureHandler.
|
|
||||||
// If the original exception was a ClassNotFoundException or NoClassDefFoundError,
|
|
||||||
// we throw a NoClassDefFoundError with an augmented error message and omit the cause.
|
|
||||||
if (ex instanceof ClassNotFoundException || ex instanceof NoClassDefFoundError) {
|
|
||||||
throw new NoClassDefFoundError(ex.getMessage() + ". " + errorMessage);
|
|
||||||
}
|
}
|
||||||
// Otherwise, we throw a NoClassDefFoundError with the cause initialized.
|
ERROR_MESSAGE = errorMessage;
|
||||||
Error error = new NoClassDefFoundError(errorMessage);
|
}
|
||||||
error.initCause(ex);
|
|
||||||
throw error;
|
|
||||||
|
public MicrometerObservationRegistryTestExecutionListener() {
|
||||||
|
// If required dependencies are missing, throw a NoClassDefFoundError so
|
||||||
|
// that this listener will be properly skipped in TestContextFailureHandler.
|
||||||
|
if (ERROR_MESSAGE != null) {
|
||||||
|
throw new NoClassDefFoundError(ERROR_MESSAGE);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,112 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2002-2023 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
|
||||||
|
*
|
||||||
|
* https://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.test.context.observation;
|
||||||
|
|
||||||
|
import java.lang.reflect.Constructor;
|
||||||
|
import java.lang.reflect.InvocationTargetException;
|
||||||
|
import java.util.function.Predicate;
|
||||||
|
import java.util.stream.IntStream;
|
||||||
|
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
|
import org.springframework.core.OverridingClassLoader;
|
||||||
|
|
||||||
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
|
import static org.assertj.core.api.Assertions.assertThatExceptionOfType;
|
||||||
|
import static org.assertj.core.api.Assertions.assertThatNoException;
|
||||||
|
import static org.springframework.test.context.observation.MicrometerObservationRegistryTestExecutionListener.DEPENDENCIES_ERROR_MESSAGE;
|
||||||
|
import static org.springframework.test.context.observation.MicrometerObservationRegistryTestExecutionListener.OBSERVATION_THREAD_LOCAL_ACCESSOR_CLASS_NAME;
|
||||||
|
import static org.springframework.test.context.observation.MicrometerObservationRegistryTestExecutionListener.THREAD_LOCAL_ACCESSOR_CLASS_NAME;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Unit tests for {@link MicrometerObservationRegistryTestExecutionListener}
|
||||||
|
* behavior regarding required dependencies.
|
||||||
|
*
|
||||||
|
* @author Sam Brannen
|
||||||
|
* @since 6.0.11
|
||||||
|
*/
|
||||||
|
class MicrometerObservationRegistryTestExecutionListenerDependencyTests {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void allDependenciesArePresent() throws Exception {
|
||||||
|
FilteringClassLoader classLoader = new FilteringClassLoader(getClass().getClassLoader(), name -> false);
|
||||||
|
Class<?> listenerClass = classLoader.loadClass(MicrometerObservationRegistryTestExecutionListener.class.getName());
|
||||||
|
// Invoke multiple times to ensure consistency.
|
||||||
|
IntStream.rangeClosed(1, 5).forEach(n -> assertThatNoException().isThrownBy(() -> instantiateListener(listenerClass)));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void threadLocalAccessorIsNotPresent() throws Exception {
|
||||||
|
assertNoClassDefFoundErrorIsThrown(THREAD_LOCAL_ACCESSOR_CLASS_NAME);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void observationThreadLocalAccessorIsNotPresent() throws Exception {
|
||||||
|
assertNoClassDefFoundErrorIsThrown(OBSERVATION_THREAD_LOCAL_ACCESSOR_CLASS_NAME);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void assertNoClassDefFoundErrorIsThrown(String missingClassName) throws Exception {
|
||||||
|
FilteringClassLoader classLoader = new FilteringClassLoader(getClass().getClassLoader(), missingClassName::equals);
|
||||||
|
Class<?> listenerClass = classLoader.loadClass(MicrometerObservationRegistryTestExecutionListener.class.getName());
|
||||||
|
// Invoke multiple times to ensure the same error message is generated every time.
|
||||||
|
IntStream.rangeClosed(1, 5).forEach(n -> assertExceptionThrown(missingClassName, listenerClass));
|
||||||
|
}
|
||||||
|
|
||||||
|
private void assertExceptionThrown(String missingClassName, Class<?> listenerClass) {
|
||||||
|
assertThatExceptionOfType(InvocationTargetException.class)
|
||||||
|
.isThrownBy(() -> instantiateListener(listenerClass))
|
||||||
|
.havingCause()
|
||||||
|
.isInstanceOf(NoClassDefFoundError.class)
|
||||||
|
.withMessage(missingClassName + ". " + DEPENDENCIES_ERROR_MESSAGE);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void instantiateListener(Class<?> listenerClass) throws Exception {
|
||||||
|
assertThat(listenerClass).isNotNull();
|
||||||
|
Constructor<?> constructor = listenerClass.getDeclaredConstructor();
|
||||||
|
constructor.setAccessible(true);
|
||||||
|
constructor.newInstance();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static class FilteringClassLoader extends OverridingClassLoader {
|
||||||
|
|
||||||
|
private static final Predicate<? super String> isListenerClass =
|
||||||
|
MicrometerObservationRegistryTestExecutionListener.class.getName()::equals;
|
||||||
|
|
||||||
|
private final Predicate<String> classNameFilter;
|
||||||
|
|
||||||
|
|
||||||
|
FilteringClassLoader(ClassLoader parent, Predicate<String> classNameFilter) {
|
||||||
|
super(parent);
|
||||||
|
this.classNameFilter = classNameFilter;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected boolean isEligibleForOverriding(String className) {
|
||||||
|
return this.classNameFilter.or(isListenerClass).test(className);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {
|
||||||
|
if (this.classNameFilter.test(name)) {
|
||||||
|
throw new ClassNotFoundException(name);
|
||||||
|
}
|
||||||
|
return super.loadClass(name, resolve);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue