162 lines
5.4 KiB
Java
Raw Normal View History

2015-09-07 11:44:36 -06:00
// com/mindviewinc/atunit/AtUnit.java
2015-11-14 16:18:05 -08:00
// <20>2016 MindView LLC: see Copyright.txt
2015-06-15 17:47:35 -07:00
// An annotation-based unit-test framework.
package com.mindviewinc.atunit;
import java.lang.reflect.*;
import java.io.*;
import java.util.*;
import onjava.*;
2015-06-15 17:47:35 -07:00
public class AtUnit implements ProcessFiles.Strategy {
static Class<?> testClass;
static List<String> failedTests= new ArrayList<>();
static long testsRun = 0;
static long failures = 0;
public static void main(String[] args) throws Exception {
ClassLoader.getSystemClassLoader()
.setDefaultAssertionStatus(true); // Enable asserts
new ProcessFiles(new AtUnit(), "class").start(args);
if(failures == 0)
2015-11-03 12:00:44 -08:00
System.out.println("OK (" + testsRun + " tests)");
2015-06-15 17:47:35 -07:00
else {
2015-11-03 12:00:44 -08:00
System.out.println("(" + testsRun + " tests)");
System.out.println("\n>>> " + failures + " FAILURE" +
2015-06-15 17:47:35 -07:00
(failures > 1 ? "S" : "") + " <<<");
for(String failed : failedTests)
2015-11-03 12:00:44 -08:00
System.out.println(" " + failed);
2015-06-15 17:47:35 -07:00
}
}
@Override
public void process(File cFile) {
try {
String cName = ClassNameFinder.thisClass(
BinaryFile.read(cFile));
if(!cName.contains("."))
return; // Ignore unpackaged classes
testClass = Class.forName(cName);
} catch(IOException | ClassNotFoundException e) {
throw new RuntimeException(e);
}
TestMethods testMethods = new TestMethods();
Method creator = null;
Method cleanup = null;
for(Method m : testClass.getDeclaredMethods()) {
testMethods.addIfTestMethod(m);
if(creator == null)
creator = checkForCreatorMethod(m);
if(cleanup == null)
cleanup = checkForCleanupMethod(m);
}
if(testMethods.size() > 0) {
if(creator == null)
try {
if(!Modifier.isPublic(testClass
.getDeclaredConstructor().getModifiers())) {
2015-11-03 12:00:44 -08:00
System.out.println("Error: " + testClass +
" no-arg constructor must be public");
2015-06-15 17:47:35 -07:00
System.exit(1);
}
} catch(NoSuchMethodException e) {
2015-11-03 12:00:44 -08:00
// Synthesized no-arg constructor; OK
2015-06-15 17:47:35 -07:00
}
2015-11-03 12:00:44 -08:00
System.out.println(testClass.getName());
2015-06-15 17:47:35 -07:00
}
for(Method m : testMethods) {
2015-11-03 12:00:44 -08:00
System.out.print(" . " + m.getName() + " ");
2015-06-15 17:47:35 -07:00
try {
Object testObject = createTestObject(creator);
boolean success = false;
try {
if(m.getReturnType().equals(boolean.class))
success = (Boolean)m.invoke(testObject);
else {
m.invoke(testObject);
success = true; // If no assert fails
}
} catch(InvocationTargetException e) {
// Actual exception is inside e:
2015-11-03 12:00:44 -08:00
System.out.println(e.getCause());
2015-06-15 17:47:35 -07:00
}
2015-11-03 12:00:44 -08:00
System.out.println(success ? "" : "(failed)");
2015-06-15 17:47:35 -07:00
testsRun++;
if(!success) {
failures++;
failedTests.add(testClass.getName() +
": " + m.getName());
}
if(cleanup != null)
cleanup.invoke(testObject, testObject);
} catch(IllegalAccessException |
IllegalArgumentException |
InvocationTargetException e) {
throw new RuntimeException(e);
}
}
}
static class TestMethods extends ArrayList<Method> {
void addIfTestMethod(Method m) {
if(m.getAnnotation(Test.class) == null)
return;
if(!(m.getReturnType().equals(boolean.class) ||
m.getReturnType().equals(void.class)))
throw new RuntimeException("@Test method" +
" must return boolean or void");
m.setAccessible(true); // In case it's private, etc.
add(m);
}
}
private static Method checkForCreatorMethod(Method m) {
if(m.getAnnotation(TestObjectCreate.class) == null)
return null;
if(!m.getReturnType().equals(testClass))
throw new RuntimeException("@TestObjectCreate " +
"must return instance of Class to be tested");
if((m.getModifiers() &
java.lang.reflect.Modifier.STATIC) < 1)
throw new RuntimeException("@TestObjectCreate " +
"must be static.");
m.setAccessible(true);
return m;
}
private static Method checkForCleanupMethod(Method m) {
if(m.getAnnotation(TestObjectCleanup.class) == null)
return null;
if(!m.getReturnType().equals(void.class))
throw new RuntimeException("@TestObjectCleanup " +
"must return void");
if((m.getModifiers() &
java.lang.reflect.Modifier.STATIC) < 1)
throw new RuntimeException("@TestObjectCleanup " +
"must be static.");
if(m.getParameterTypes().length == 0 ||
m.getParameterTypes()[0] != testClass)
throw new RuntimeException("@TestObjectCleanup " +
"must take an argument of the tested type.");
m.setAccessible(true);
return m;
}
private static Object createTestObject(Method creator) {
if(creator != null) {
try {
return creator.invoke(testClass);
} catch(IllegalAccessException |
IllegalArgumentException |
InvocationTargetException e) {
throw new RuntimeException("Couldn't run " +
"@TestObject (creator) method.");
}
2015-11-03 12:00:44 -08:00
} else { // Use the no-arg constructor:
2015-06-15 17:47:35 -07:00
try {
return testClass.newInstance();
} catch(InstantiationException |
IllegalAccessException e) {
throw new RuntimeException("Couldn't create a " +
"test object. Try using a @TestObject method.");
}
}
}
2015-09-07 11:44:36 -06:00
}
/* Output:
2015-06-15 17:47:35 -07:00
OK (0 tests)
2015-09-07 11:44:36 -06:00
*/