package de.platon42.intellij.jupiter; import com.intellij.openapi.module.Module; import com.intellij.openapi.projectRoots.Sdk; import com.intellij.openapi.projectRoots.impl.JavaAwareProjectJdkTableImpl; import com.intellij.openapi.roots.ContentEntry; import com.intellij.openapi.roots.ModifiableRootModel; import com.intellij.openapi.util.Disposer; import com.intellij.openapi.vfs.LocalFileSystem; import com.intellij.openapi.vfs.VirtualFile; import com.intellij.testFramework.*; import com.intellij.testFramework.fixtures.IdeaTestExecutionPolicy; import com.intellij.testFramework.fixtures.JavaCodeInsightTestFixture; import com.intellij.testFramework.fixtures.LightJavaCodeInsightFixtureTestCase; import org.jetbrains.annotations.NotNull; import org.junit.jupiter.api.extension.*; import org.junit.jupiter.api.extension.ExtensionContext.Namespace; import org.junit.jupiter.api.extension.ExtensionContext.Store; import java.lang.annotation.Annotation; import java.lang.reflect.Method; import java.lang.reflect.Parameter; import java.net.URISyntaxException; import java.nio.file.Path; import java.nio.file.Paths; import java.util.logging.Logger; import java.util.stream.Stream; public class LightCodeInsightExtension implements ParameterResolver, AfterTestExecutionCallback, InvocationInterceptor { private static final Logger LOG = Logger.getLogger(LightCodeInsightExtension.class.getName()); @Override public boolean supportsParameter(ParameterContext parameterContext, ExtensionContext extensionContext) throws ParameterResolutionException { Parameter parameter = parameterContext.getParameter(); return parameter.isAnnotationPresent(MyFixture.class) || parameter.isAnnotationPresent(MyTestCase.class); } @Override public Object resolveParameter(ParameterContext parameterContext, ExtensionContext extensionContext) throws ParameterResolutionException { LightCodeInsightFixtureTestCaseWrapper testCase = getWrapper(extensionContext); Parameter parameter = parameterContext.getParameter(); if (parameter.isAnnotationPresent(MyFixture.class)) { return testCase.getMyFixture(); } else if (parameter.isAnnotationPresent(MyTestCase.class)) { return testCase; } return null; } private LightCodeInsightFixtureTestCaseWrapper getWrapper(ExtensionContext extensionContext) { Store store = getStore(extensionContext); return (LightCodeInsightFixtureTestCaseWrapper) store.getOrComputeIfAbsent("testCase", key -> { LightCodeInsightFixtureTestCaseWrapper wrapper = new LightCodeInsightFixtureTestCaseWrapper(extensionContext); try { wrapper.setUp(); } catch (Exception e) { LOG.severe("Exception during setUp(): " + e); throw new IllegalStateException("Exception during setUp()", e); } return wrapper; }); } @Override public void afterTestExecution(ExtensionContext context) throws Exception { Store store = getStore(context); LightCodeInsightFixtureTestCaseWrapper testCase = (LightCodeInsightFixtureTestCaseWrapper) store.get("testCase"); if (testCase != null) { testCase.tearDown(); } } private static Store getStore(ExtensionContext context) { return context.getStore(Namespace.create(LightCodeInsightExtension.class, context.getRequiredTestMethod())); } @Override public void interceptTestMethod(Invocation invocation, ReflectiveInvocationContext invocationContext, ExtensionContext extensionContext) throws Throwable { Throwable[] throwables = new Throwable[1]; Runnable runnable = () -> { try { TestLoggerFactory.onTestStarted(); invocation.proceed(); TestLoggerFactory.onTestFinished(true); } catch (Throwable e) { TestLoggerFactory.onTestFinished(false); throwables[0] = e; } }; invokeTestRunnable(runnable); if (throwables[0] != null) { throw throwables[0]; } } private static void invokeTestRunnable(@NotNull Runnable runnable) { IdeaTestExecutionPolicy policy = IdeaTestExecutionPolicy.current(); if (policy != null && !policy.runInDispatchThread()) { runnable.run(); } else { EdtTestUtilKt.runInEdtAndWait(() -> { runnable.run(); return null; }); } } private static class LightCodeInsightFixtureTestCaseWrapper extends LightJavaCodeInsightFixtureTestCase { private final ExtensionContext extensionContext; private LightCodeInsightFixtureTestCaseWrapper(ExtensionContext extensionContext) { this.extensionContext = extensionContext; } @Override public void setUp() throws Exception { super.setUp(); } @Override public void tearDown() throws Exception { super.tearDown(); UsefulTestCase.clearFields(this); if (myFixture != null && getProject() != null && !getProject().isDisposed()) { Disposer.dispose(getProject()); } } @NotNull @Override protected LightProjectDescriptor getProjectDescriptor() { TestJdk testJdk = getMethodOrClassAnnotation(TestJdk.class); if (testJdk == null) { return super.getProjectDescriptor(); } return new ProjectDescriptor(testJdk.value(), testJdk.annotations()) { @Override public Sdk getSdk() { return testJdk.useInternal() ? JavaAwareProjectJdkTableImpl.getInstanceEx().getInternalJdk() : super.getSdk(); } @Override public void configureModule(@NotNull Module module, @NotNull ModifiableRootModel model, @NotNull ContentEntry contentEntry) { super.configureModule(module, model, contentEntry); AddLocalJarToModule methodOrClassAnnotation = getMethodOrClassAnnotation(AddLocalJarToModule.class); if (methodOrClassAnnotation != null) { Stream.of(methodOrClassAnnotation.value()).forEach(it -> addJarContaining(model, it)); } } }; } void addJarContaining(ModifiableRootModel model, Class clazz) { try { Path jarPath = Paths.get(clazz.getProtectionDomain().getCodeSource().getLocation().toURI()); VirtualFile jarFile = LocalFileSystem.getInstance().findFileByIoFile(jarPath.toFile()); myFixture.allowTreeAccessForFile(jarFile); PsiTestUtil.addLibrary( model, jarPath.getFileName().toString().replace(".jar", ""), jarPath.getParent().toString(), jarPath.getFileName().toString() ); } catch (URISyntaxException e) { throw new IllegalArgumentException("Class URL malformed", e); } } @Override protected String getTestDataPath() { TestDataPath testDataPath = getMethodOrClassAnnotation(TestDataPath.class); if (testDataPath == null) { return super.getTestDataPath(); } TestDataSubPath testDataSubPath = getMethodOrClassAnnotation(TestDataSubPath.class); if (testDataSubPath == null) { return testDataPath.value(); } return Paths.get(testDataPath.value(), testDataSubPath.value()).toString(); } public JavaCodeInsightTestFixture getMyFixture() { return myFixture; } private T getMethodOrClassAnnotation(Class clazz) { T annotation = extensionContext.getRequiredTestMethod().getAnnotation(clazz); if (annotation == null) { annotation = extensionContext.getRequiredTestClass().getAnnotation(clazz); } return annotation; } } }