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

org.apache.paimon.testutils.junit.parameterized.ParameterizedTestExtension Maven / Gradle / Ivy

The newest version!
/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you 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.apache.paimon.testutils.junit.parameterized;

import org.junit.jupiter.api.TestTemplate;
import org.junit.jupiter.api.extension.BeforeEachCallback;
import org.junit.jupiter.api.extension.Extension;
import org.junit.jupiter.api.extension.ExtensionContext;
import org.junit.jupiter.api.extension.ParameterContext;
import org.junit.jupiter.api.extension.ParameterResolutionException;
import org.junit.jupiter.api.extension.ParameterResolver;
import org.junit.jupiter.api.extension.TestTemplateInvocationContext;
import org.junit.jupiter.api.extension.TestTemplateInvocationContextProvider;
import org.junit.platform.commons.support.AnnotationSupport;
import org.junit.platform.commons.support.HierarchyTraversalMode;

import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.text.MessageFormat;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.function.Function;
import java.util.stream.Stream;

/**
 * This extension is used to implement parameterized tests for Junit 5 to replace Parameterized in
 * Junit4.
 *
 * 

When use this extension, all tests must be annotated by {@link TestTemplate}. */ public class ParameterizedTestExtension implements TestTemplateInvocationContextProvider { private static final ExtensionContext.Namespace NAMESPACE = ExtensionContext.Namespace.create("parameterized"); private static final String PARAMETERS_STORE_KEY = "parameters"; private static final String PARAMETER_FIELD_STORE_KEY_PREFIX = "parameterField_"; private static final String INDEX_TEMPLATE = "{index}"; @Override public boolean supportsTestTemplate(ExtensionContext context) { return true; } @Override public Stream provideTestTemplateInvocationContexts( ExtensionContext context) { // Search method annotated with @Parameters final List parameterProviders = AnnotationSupport.findAnnotatedMethods( context.getRequiredTestClass(), Parameters.class, HierarchyTraversalMode.TOP_DOWN); if (parameterProviders.isEmpty()) { throw new IllegalStateException("Cannot find any parameter provider"); } if (parameterProviders.size() > 1) { throw new IllegalStateException("Multiple parameter providers are found"); } Method parameterProvider = parameterProviders.get(0); // Get potential test name String testNameTemplate = parameterProvider.getAnnotation(Parameters.class).name(); // Get parameter values final Object parameterValues; try { parameterValues = parameterProvider.invoke(null); context.getStore(NAMESPACE).put(PARAMETERS_STORE_KEY, parameterValues); } catch (Exception e) { throw new IllegalStateException("Failed to invoke parameter provider", e); } assert parameterValues != null; // Parameter values could be Object[][] if (parameterValues instanceof Object[][]) { Object[][] typedParameterValues = (Object[][]) parameterValues; return createContextForParameters( Arrays.stream(typedParameterValues), testNameTemplate, context); } // or a Collection if (parameterValues instanceof Collection) { final Collection typedParameterValues = (Collection) parameterValues; final Stream parameterValueStream = typedParameterValues.stream() .map( (Function) parameterValue -> { if (parameterValue instanceof Object[]) { return (Object[]) parameterValue; } else { return new Object[] {parameterValue}; } }); return createContextForParameters(parameterValueStream, testNameTemplate, context); } throw new IllegalStateException( String.format( "Return type of @Parameters annotated method \"%s\" should be either Object[][] or Collection", parameterProvider)); } private static class FieldInjectingInvocationContext implements TestTemplateInvocationContext { private final String testNameTemplate; private final Object[] parameterValues; public FieldInjectingInvocationContext(String testNameTemplate, Object[] parameterValues) { this.testNameTemplate = testNameTemplate; this.parameterValues = parameterValues; } @Override public String getDisplayName(int invocationIndex) { if (INDEX_TEMPLATE.equals(testNameTemplate)) { return TestTemplateInvocationContext.super.getDisplayName(invocationIndex); } else { return MessageFormat.format(testNameTemplate, parameterValues); } } @Override public List getAdditionalExtensions() { return Collections.singletonList(new FieldInjectingHook(parameterValues)); } private static class FieldInjectingHook implements BeforeEachCallback { private final Object[] parameterValues; public FieldInjectingHook(Object[] parameterValues) { this.parameterValues = parameterValues; } @Override public void beforeEach(ExtensionContext context) throws Exception { for (int i = 0; i < parameterValues.length; i++) { getParameterField(i, context) .set(context.getRequiredTestInstance(), parameterValues[i]); } } } } private static class ConstructorParameterResolverInvocationContext implements TestTemplateInvocationContext { private final String testNameTemplate; private final Object[] parameterValues; public ConstructorParameterResolverInvocationContext( String testNameTemplate, Object[] parameterValues) { this.testNameTemplate = testNameTemplate; this.parameterValues = parameterValues; } @Override public String getDisplayName(int invocationIndex) { if (INDEX_TEMPLATE.equals(testNameTemplate)) { return TestTemplateInvocationContext.super.getDisplayName(invocationIndex); } else { return MessageFormat.format(testNameTemplate, parameterValues); } } @Override public List getAdditionalExtensions() { return Collections.singletonList(new ConstructorParameterResolver(parameterValues)); } private static class ConstructorParameterResolver implements ParameterResolver { private final Object[] parameterValues; public ConstructorParameterResolver(Object[] parameterValues) { this.parameterValues = parameterValues; } @Override public boolean supportsParameter( ParameterContext parameterContext, ExtensionContext extensionContext) throws ParameterResolutionException { return true; } @Override public Object resolveParameter( ParameterContext parameterContext, ExtensionContext extensionContext) throws ParameterResolutionException { return parameterValues[parameterContext.getIndex()]; } } } // -------------------------------- Helper functions ------------------------------------------- private Stream createContextForParameters( Stream parameterValueStream, String testNameTemplate, ExtensionContext context) { // Search fields annotated by @Parameter final List parameterFields = AnnotationSupport.findAnnotatedFields( context.getRequiredTestClass(), Parameter.class); // Use constructor parameter style if (parameterFields.isEmpty()) { return parameterValueStream.map( parameterValue -> new ConstructorParameterResolverInvocationContext( testNameTemplate, parameterValue)); } // Use field injection style for (Field parameterField : parameterFields) { final int index = parameterField.getAnnotation(Parameter.class).value(); context.getStore(NAMESPACE).put(getParameterFieldStoreKey(index), parameterField); } return parameterValueStream.map( parameterValue -> new FieldInjectingInvocationContext(testNameTemplate, parameterValue)); } private static String getParameterFieldStoreKey(int parameterIndex) { return PARAMETER_FIELD_STORE_KEY_PREFIX + parameterIndex; } private static Field getParameterField(int parameterIndex, ExtensionContext context) { return (Field) context.getStore(NAMESPACE).get(getParameterFieldStoreKey(parameterIndex)); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy