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

org.junit.jupiter.engine.extension.MutableExtensionRegistry Maven / Gradle / Ivy

There is a newer version: 5.11.3
Show newest version
/*
 * Copyright 2015-2022 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 * @see #getReversedExtensions(Class) */ private Stream streamLocal(Class extensionType) { // @formatter:off return this.registeredExtensions.stream() .filter(extensionType::isInstance) .map(extensionType::cast); // @formatter:on } @Override public void registerExtension(Class 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 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 + "]"; } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy