org.junit.jupiter.engine.extension.MutableExtensionRegistry Maven / Gradle / Ivy
Show all versions of junit-jupiter-engine Show documentation
/*
* Copyright 2015-2024 the original author or authors.
*
* All rights reserved. This program and the accompanying materials are
* made available under the terms of the Eclipse Public License v2.0 which
* accompanies this distribution and is available at
*
* https://www.eclipse.org/legal/epl-v20.html
*/
package org.junit.jupiter.engine.extension;
import static java.util.stream.Stream.concat;
import static org.apiguardian.api.API.Status.INTERNAL;
import java.lang.reflect.Member;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.ServiceLoader;
import java.util.Set;
import java.util.stream.Stream;
import org.apiguardian.api.API;
import org.junit.jupiter.api.extension.Extension;
import org.junit.jupiter.engine.config.JupiterConfiguration;
import org.junit.platform.commons.logging.Logger;
import org.junit.platform.commons.logging.LoggerFactory;
import org.junit.platform.commons.util.ClassLoaderUtils;
import org.junit.platform.commons.util.Preconditions;
import org.junit.platform.commons.util.ReflectionUtils;
/**
* Default, mutable implementation of {@link ExtensionRegistry}.
*
* A registry has a reference to its parent registry, and all lookups are
* performed first in the current registry itself and then recursively in its
* ancestors.
*
* @since 5.5
*/
@API(status = INTERNAL, since = "5.5")
public class MutableExtensionRegistry implements ExtensionRegistry, ExtensionRegistrar {
private static final Logger logger = LoggerFactory.getLogger(MutableExtensionRegistry.class);
private static final List DEFAULT_STATELESS_EXTENSIONS = Collections.unmodifiableList(Arrays.asList(//
new DisabledCondition(), //
new TimeoutExtension(), //
new RepeatedTestExtension(), //
new TestInfoParameterResolver(), //
new TestReporterParameterResolver()));
/**
* Factory for creating and populating a new root registry with the default
* extensions.
*
* If the {@link org.junit.jupiter.engine.Constants#EXTENSIONS_AUTODETECTION_ENABLED_PROPERTY_NAME}
* configuration parameter has been set to {@code true}, extensions will be
* auto-detected using Java's {@link ServiceLoader} mechanism and automatically
* registered after the default extensions.
*
* @param configuration configuration parameters used to retrieve the extension
* auto-detection flag; never {@code null}
* @return a new {@code ExtensionRegistry}; never {@code null}
*/
public static MutableExtensionRegistry createRegistryWithDefaultExtensions(JupiterConfiguration configuration) {
MutableExtensionRegistry extensionRegistry = new MutableExtensionRegistry(null);
DEFAULT_STATELESS_EXTENSIONS.forEach(extensionRegistry::registerDefaultExtension);
extensionRegistry.registerDefaultExtension(new TempDirectory(configuration));
if (configuration.isExtensionAutoDetectionEnabled()) {
registerAutoDetectedExtensions(extensionRegistry);
}
return extensionRegistry;
}
private static void registerAutoDetectedExtensions(MutableExtensionRegistry extensionRegistry) {
ServiceLoader.load(Extension.class, ClassLoaderUtils.getDefaultClassLoader())//
.forEach(extensionRegistry::registerAutoDetectedExtension);
}
/**
* Factory for creating and populating a new registry from a list of
* extension types and a parent registry.
*
* @param parentRegistry the parent registry
* @param extensionTypes the types of extensions to be registered in
* the new registry
* @return a new {@code ExtensionRegistry}; never {@code null}
*/
public static MutableExtensionRegistry createRegistryFrom(MutableExtensionRegistry parentRegistry,
Stream> extensionTypes) {
Preconditions.notNull(parentRegistry, "parentRegistry must not be null");
MutableExtensionRegistry registry = new MutableExtensionRegistry(parentRegistry);
extensionTypes.forEach(registry::registerExtension);
return registry;
}
private final MutableExtensionRegistry parent;
private final Set> registeredExtensionTypes = new LinkedHashSet<>();
private final List registeredExtensions = new ArrayList<>();
private MutableExtensionRegistry(MutableExtensionRegistry parent) {
this.parent = parent;
}
@Override
public Stream stream(Class extensionType) {
if (this.parent == null) {
return streamLocal(extensionType);
}
return concat(this.parent.stream(extensionType), streamLocal(extensionType));
}
/**
* Stream all {@code Extensions} of the specified type that are present
* in this registry.
*
* Extensions in ancestors are ignored.
*
* @param extensionType the type of {@link Extension} to stream
*/
private Stream streamLocal(Class extensionType) {
// @formatter:off
return this.registeredExtensions.stream()
.filter(extensionType::isInstance)
.map(extensionType::cast);
// @formatter:on
}
@Override
public void registerExtension(Class extends Extension> extensionType) {
if (!isAlreadyRegistered(extensionType)) {
registerLocalExtension(ReflectionUtils.newInstance(extensionType));
}
}
/**
* Determine if the supplied type is already registered in this registry or in a
* parent registry.
*/
private boolean isAlreadyRegistered(Class extends Extension> extensionType) {
return (this.registeredExtensionTypes.contains(extensionType)
|| (this.parent != null && this.parent.isAlreadyRegistered(extensionType)));
}
@Override
public void registerExtension(Extension extension, Object source) {
Preconditions.notNull(source, "source must not be null");
registerExtension("local", extension, source);
}
@Override
public void registerSyntheticExtension(Extension extension, Object source) {
registerExtension("synthetic", extension, source);
}
private void registerDefaultExtension(Extension extension) {
registerExtension("default", extension);
}
private void registerAutoDetectedExtension(Extension extension) {
registerExtension("auto-detected", extension);
}
private void registerLocalExtension(Extension extension) {
registerExtension("local", extension);
}
private void registerExtension(String category, Extension extension) {
registerExtension(category, extension, null);
}
private void registerExtension(String category, Extension extension, Object source) {
Preconditions.notBlank(category, "category must not be null or blank");
Preconditions.notNull(extension, "Extension must not be null");
logger.trace(
() -> String.format("Registering %s extension [%s]%s", category, extension, buildSourceInfo(source)));
this.registeredExtensions.add(extension);
this.registeredExtensionTypes.add(extension.getClass());
}
private String buildSourceInfo(Object source) {
if (source == null) {
return "";
}
if (source instanceof Member) {
Member member = (Member) source;
Object type = (member instanceof Method ? "method" : "field");
source = String.format("%s %s.%s", type, member.getDeclaringClass().getName(), member.getName());
}
return " from source [" + source + "]";
}
}