|
28 | 28 | import java.util.Optional;
|
29 | 29 | import java.util.Set;
|
30 | 30 | import java.util.function.Function;
|
| 31 | +import java.util.stream.Collectors; |
31 | 32 |
|
32 | 33 | import org.apiguardian.api.API;
|
33 | 34 | import org.junit.jupiter.api.TestInstance.Lifecycle;
|
34 | 35 | import org.junit.jupiter.api.extension.AfterAllCallback;
|
35 | 36 | import org.junit.jupiter.api.extension.BeforeAllCallback;
|
36 | 37 | import org.junit.jupiter.api.extension.Extension;
|
37 | 38 | import org.junit.jupiter.api.extension.ExtensionContext;
|
| 39 | +import org.junit.jupiter.api.extension.ExtensionContextException; |
| 40 | +import org.junit.jupiter.api.extension.TestInstanceFactory; |
| 41 | +import org.junit.jupiter.api.extension.TestInstanceFactoryContext; |
38 | 42 | import org.junit.jupiter.api.extension.TestInstancePostProcessor;
|
39 | 43 | import org.junit.jupiter.engine.execution.AfterEachMethodAdapter;
|
40 | 44 | import org.junit.jupiter.engine.execution.BeforeEachMethodAdapter;
|
| 45 | +import org.junit.jupiter.engine.execution.DefaultTestInstanceFactoryContext; |
41 | 46 | import org.junit.jupiter.engine.execution.ExecutableInvoker;
|
42 | 47 | import org.junit.jupiter.engine.execution.JupiterEngineExecutionContext;
|
43 | 48 | import org.junit.jupiter.engine.execution.TestInstanceProvider;
|
@@ -235,11 +240,60 @@ private Object instantiateAndPostProcessTestInstance(JupiterEngineExecutionConte
|
235 | 240 | return instance;
|
236 | 241 | }
|
237 | 242 |
|
| 243 | + protected Object invokeTestInstanceConstructor(Optional<Object> outerInstance, ExtensionRegistry registry, |
| 244 | + ExtensionContext extensionContext) { |
| 245 | + Constructor<?> constructor = ReflectionUtils.getDeclaredConstructor(this.testClass); |
| 246 | + if (outerInstance.isPresent()) { |
| 247 | + return executableInvoker.invoke(constructor, outerInstance.get(), extensionContext, registry); |
| 248 | + } |
| 249 | + else { |
| 250 | + return executableInvoker.invoke(constructor, extensionContext, registry); |
| 251 | + } |
| 252 | + } |
| 253 | + |
| 254 | + protected TestInstanceFactory resolveTestInstanceFactory(ExtensionRegistry registry, |
| 255 | + ExtensionRegistry parentRegistry) { |
| 256 | + |
| 257 | + List<TestInstanceFactory> factories = registry.getExtensions(TestInstanceFactory.class); |
| 258 | + if (factories.isEmpty()) { |
| 259 | + return null; |
| 260 | + } |
| 261 | + |
| 262 | + if (parentRegistry != null) { |
| 263 | + List<TestInstanceFactory> parentFactories = parentRegistry.getExtensions(TestInstanceFactory.class); |
| 264 | + factories.removeAll(parentFactories); |
| 265 | + } |
| 266 | + |
| 267 | + if (factories.size() > 1) { |
| 268 | + String factoryNames = factories.stream().map(factory -> factory.getClass().getSimpleName()).collect( |
| 269 | + Collectors.joining(",")); |
| 270 | + |
| 271 | + String errorMessage = String.format( |
| 272 | + "Too many TestInstanceFactory extensions [%s] registered on test class: %s (allowed is at most 1)", |
| 273 | + factoryNames, testClass.getSimpleName()); |
| 274 | + |
| 275 | + throw new ExtensionContextException(errorMessage); |
| 276 | + } |
| 277 | + |
| 278 | + return factories.get(0); |
| 279 | + } |
| 280 | + |
| 281 | + protected Object invokeTestInstanceFactory(Optional<Object> outerInstance, ExtensionRegistry registry, |
| 282 | + ExtensionRegistry parentRegistry, ExtensionContext extensionContext) { |
| 283 | + |
| 284 | + TestInstanceFactory factory = resolveTestInstanceFactory(registry, parentRegistry); |
| 285 | + if (factory == null) { |
| 286 | + return invokeTestInstanceConstructor(outerInstance, registry, extensionContext); |
| 287 | + } |
| 288 | + |
| 289 | + TestInstanceFactoryContext factoryContext = new DefaultTestInstanceFactoryContext(this.testClass, |
| 290 | + outerInstance); |
| 291 | + return factory.instantiateTestClass(factoryContext, extensionContext); |
| 292 | + } |
| 293 | + |
238 | 294 | protected Object instantiateTestClass(JupiterEngineExecutionContext parentExecutionContext,
|
239 | 295 | ExtensionRegistry registry, ExtensionContext extensionContext) {
|
240 |
| - |
241 |
| - Constructor<?> constructor = ReflectionUtils.getDeclaredConstructor(this.testClass); |
242 |
| - return executableInvoker.invoke(constructor, extensionContext, registry); |
| 296 | + return invokeTestInstanceFactory(Optional.empty(), registry, null, extensionContext); |
243 | 297 | }
|
244 | 298 |
|
245 | 299 | private void invokeTestInstancePostProcessors(Object instance, ExtensionRegistry registry,
|
|
0 commit comments