mc68000-asm-plugin/src/test/java/de/platon42/intellij/jupiter/ParsingTestExtension.java

233 lines
8.6 KiB
Java

package de.platon42.intellij.jupiter;
import com.intellij.lang.ParserDefinition;
import com.intellij.openapi.Disposable;
import com.intellij.openapi.util.Disposer;
import com.intellij.psi.PsiFile;
import com.intellij.testFramework.EdtTestUtilKt;
import com.intellij.testFramework.ParsingTestCase;
import com.intellij.testFramework.TestLoggerFactory;
import com.intellij.testFramework.fixtures.IdeaTestExecutionPolicy;
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.io.IOException;
import java.lang.annotation.Annotation;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Parameter;
import java.nio.file.Paths;
import java.util.logging.Logger;
public class ParsingTestExtension implements ParameterResolver, AfterTestExecutionCallback, InvocationInterceptor {
private static final Logger LOG = Logger.getLogger(ParsingTestExtension.class.getName());
@Override
public boolean supportsParameter(ParameterContext parameterContext, ExtensionContext extensionContext) throws ParameterResolutionException {
Parameter parameter = parameterContext.getParameter();
return parameter.isAnnotationPresent(MyTestCase.class);
}
@Override
public Object resolveParameter(ParameterContext parameterContext, ExtensionContext extensionContext) throws ParameterResolutionException {
ParsingTestCaseWrapper testCase = getWrapper(extensionContext);
Parameter parameter = parameterContext.getParameter();
if (parameter.isAnnotationPresent(MyTestCase.class)) {
return testCase;
}
return null;
}
private ParsingTestCaseWrapper getWrapper(ExtensionContext extensionContext) {
Store store = getStore(extensionContext);
return (ParsingTestCaseWrapper) store.getOrComputeIfAbsent("testCase",
key -> {
MyParser myParser = extensionContext.getRequiredTestClass().getAnnotation(MyParser.class);
ParserDefinition parserDefinition;
try {
parserDefinition = myParser.value().getDeclaredConstructor().newInstance();
} catch (InstantiationException | IllegalAccessException | InvocationTargetException | NoSuchMethodException ex) {
throw new RuntimeException("Could not instantiate ParserDefinition!", ex);
}
ParsingTestCaseWrapper wrapper = new ParsingTestCaseWrapper(extensionContext, myParser.extension(), parserDefinition);
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);
ParsingTestCaseWrapper testCase = (ParsingTestCaseWrapper) store.get("testCase");
if (testCase != null) {
testCase.tearDown();
}
}
private static Store getStore(ExtensionContext context) {
return context.getStore(Namespace.create(ParsingTestExtension.class, context.getRequiredTestMethod()));
}
@Override
public void interceptTestMethod(Invocation<Void> invocation, ReflectiveInvocationContext<Method> 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;
});
}
}
public interface IParsingTestCase {
void ensureNoErrorElements();
void doTest(boolean checkResult);
void doTest(boolean checkResult, boolean ensureNoErrorElements);
PsiFile parseFile(String name, String text);
void doCodeTest(@NotNull String code) throws IOException;
PsiFile getFile();
void ensureCorrectReparse();
}
private static class ParsingTestCaseWrapper extends ParsingTestCase implements IParsingTestCase {
private ExtensionContext extensionContext;
private static ExtensionContext extensionContextHack;
private ParsingTestCaseWrapper(ExtensionContext extensionContext, String extension, ParserDefinition parserDefinition) {
super(passthroughInitHack(extensionContext), extension, parserDefinition);
this.extensionContext = extensionContext;
}
private static String passthroughInitHack(ExtensionContext extensionContext) {
extensionContextHack = extensionContext;
return "";
}
@Override
public void setUp() throws Exception {
super.setUp();
Store store = getStore(extensionContext);
store.put("disposable", Disposer.newDisposable("ParsingTestCaseWrapper"));
}
@Override
public void tearDown() throws Exception {
Store store = getStore(extensionContext);
Disposable disposable = (Disposable) store.get("disposable");
if (disposable != null) {
Disposer.dispose(disposable);
store.remove("disposable");
}
super.tearDown(); // Clears fields!
}
@Override
public String getName() {
String testname = extensionContext.getDisplayName().replace(" ", "_");
ParserResultFile parserResultFile = extensionContext.getRequiredTestMethod().getAnnotation(ParserResultFile.class);
if (parserResultFile != null) {
testname = parserResultFile.value();
}
return testname;
}
@Override
protected String getTestDataPath() {
if (extensionContext == null) // Method is called by super-constructor, damnit!
{
extensionContext = extensionContextHack;
}
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();
}
private <T extends Annotation> T getMethodOrClassAnnotation(Class<T> clazz) {
T annotation = extensionContext.getRequiredTestMethod().getAnnotation(clazz);
if (annotation == null) {
annotation = extensionContext.getRequiredTestClass().getAnnotation(clazz);
}
return annotation;
}
@Override
public void ensureNoErrorElements() {
super.ensureNoErrorElements();
}
@Override
public void doTest(boolean checkResult) {
super.doTest(checkResult);
}
@Override
public void doTest(boolean checkResult, boolean ensureNoErrorElements) {
super.doTest(checkResult, ensureNoErrorElements);
}
@Override
public PsiFile parseFile(String name, String text) {
return super.parseFile(name, text);
}
@Override
public void doCodeTest(@NotNull String code) throws IOException {
super.doCodeTest(code);
}
@Override
public PsiFile getFile() {
return myFile;
}
@Override
public void ensureCorrectReparse() {
ensureCorrectReparse(myFile);
}
}
}