JTest: Roll Out Your Own JUnit-Like Testing Framework
- January 17, 2023
- 1946 Unique Views
- 5 min read
In this interactive tutorial and educational project, you'll be helped in your understanding of the following.
- A basic insight into JUnit internals.
- The usage of Java annotations and reflection.
You are given minimum starter code (~20 loc), and will have to develop a simple yet self-contained framework that can be used to write real-world unit tests.
Are you ready? Let's go! 🚀
Task 0: The Big Picture
Before getting your hands dirty, it is a good idea to get a big picture of what you will code up.
I suppose most Java programmers have some familiarity with JUnit. But this tutorial also targets beginners.
In case you have never used JUnit before, please go over its official "getting started guide" to learn its core APIs, which can be summarized below:
@Test
annotation to mark a function as a test case.- a bunch of
assert...
functions (e.g.assertEquals
) used in each test case to verify the actual result is same as the expected result.
And... one more almost invisible component:
- A runner.
As a downstream developer, you don't code against runners directly. But remember that java programs start execution with public static void main
? If you followed the JUnit guide closely, you probably have noticed that you didn't write any main
method, yet your tests run just fine. How is that possible?
The answer is the runner. In the "Run the test" section of the guide, you do not start your test class directly, but run a org.junit.runner.JUnitCore
class instead. That is the runner. Your test class is passed as an argument to the runner.
So three main components in total, in nine steps below.
Task 1: @JTest
- Marker for Test Cases
Annotation is a new language construct introduced in Java 1.5. It is somewhat similar to comment in that it does not contain executable code directly, and is mainly used to provide some additional information to the program.
But unlike comment, which is completely discarded after compilation, annotations can be preserved so that programs can check for these extra information and behave accordingly.
@Test
of JUnit is an annotation to mark test methods. To provide the same functionality, we have prepared a skeleton class JTest
for you. But unlike JUnit's @Test
, which can (almost) only be used against methods, @JTest
in the skeleton code can be annotated on types as well. Your task is to fix this so that the only valid target of @JTest
is methods.
In case you haven't written any annotation before, here are some tips:
@Retention(...)
is a fixed boilerplate that you can just copy/paste and ignore its meaning for now.@Target(...)
controls where can an annotation be used.
After completing this task, run T1.java
to check your implementation.
Task 2: Finding Marked Methods
To run the tests, we need to find all the methods annotated by the @JTest
and execute them one by one. Let's create a dedicated class for this.
This class will be named JTestRunner
, because eventually it will drive the tests just like runners from JUnit.
In this newly created JTestRunner
class, please implement the static method below.
/** * Find all methods annotated by {@link JTest} in a class. * You can assume all its methods are public. * * @param clazz The class to be analyzed * @return all methods annotated by {@link JTest} */ public static List<Method> getTestMethods(Class<?> clazz)
After completing this task, run T2.java
to check your implementation.
Task 3: Running A Method
Now if anyone develops tests based on your JTest framework, you can successfully find the test case methods annotated by @JTest
. The next step, of course, is to run them.
To simplify the case, you can assume @JTest
is always annotated on static methods. This is different from how JUnit works, and is discussed in next steps section below.
Please continue editing JTestRunner
and implement the static method below.
/** * Run a method via reflection. * You can assume the method is always static. * * @param method the method to run */ public static void runMethod(Method method)
After completing this task, run T3.java
to check your implementation.
Task 4: Assertions
During the development of runMethod
in task 3, you may have noticed that invoke
can throw exceptions. These exceptions can be categorized into two types:
- A bug in your main code is found, for example by a (not-yet-implemented)
assertEquals
statement, then an exception is thrown to indicate this. We will term this case as the test FAILED. - The test code itself has a bug, for example zero division or NPE, then causes an exception. We will term this case as the test ERRORED out.
To distinguish between there two cases, you must first implement the assertions. Please modify JAssertions.java
such that:
public static void assertTrue(boolean condition)
checks if the argument istrue
. When it is not, throw a (JDK built-in)AssertionError
exception.public static void assertEquals(Object expected, Object actual)
check if the two arguments are equal (in the sense of.equals(...)
). When they are not, throw anAssertionError
exception.
After completing this task, run T4.java
to check your implementation.
Task 5: Test Result Class
Our goal is to find the test methods with getTestMethods
, and then execute them with runMethod
. Each test method may produce several possible results:
- test passed: method run normally and all assertions passed.
- test failed: method run normally until an assertion failed.
- test errored out: method terminated abnormally.
To save the result of a test method (primarily for better output), please write a new class named JTestResult
with the following two public methods:
public String getTestStatus()
returns the result of this test method as a string, which will be one ofPASS
,FAIL
andERROR
respectively.public String getTestMethodName()
returns the name of the test method (annotated by@JTest
) as a string.
Fow now only the method signature matters, so you can return whatever placeholder value.
After completing this task, run T5.java
to check your implementation.
Task 6: Returning Test Results
Update runMethod
from task 3 so that after invoking the method passed in, it returns a JTestResult
object containing proper information about the test result.
You may want to use getCause
to recover the original exception thrown in invoke
.
After completing this task, run T6.java
to check your implementation.
Task 7: Running A Test Class
runMethod
executes only a single method. To run a test class containing multiple test methods, please add the following function JTestRunner.
/** * Run a test class by executing all its methods annotated by {@link JTest}. * You can assume all its methods are static. * * @param testClassName the full name of the test class * @return results of all test cases in the class */ public static List<JTestResult> runTestClass(String testClassName)
After completing this task, run T7.java
to check your implementation.
Task 8: Entrance
Add public static void main
to JTestRunner
that starts the test. The main
method should behave like what JUnit runner does, namely taking the first argument from commandline as the test class, running it, and printing the results.
The output should contain three lines, corresponding to the passed, failed and errored cases respectively.
Concretely, a sample test class written against JTest is provided in src/test/resources/ToyTest.java
. (Feel free to read it, or compile and run it if you know how to handle classpath issues.) The output for it should be as follows.
1/3 have passed Failed tests are: [assertTrueFail] Errored tests are: [shouldError]
After completing this task, run T8.java
to check your implementation.
Task 9: JAR
JTest is already a working framework now, but why not go even further to package it as a standalone JAR? This way, downstream developers can use it almost identically with JUnit.
If you are familiar with maven, mvn package
should generate a jar package in target/
. But in this task, you need to put your JTest framework jar at src/test/resources/jtest.jar
. Also make sure JDK11+ in on your PATH
. Then you should be able to:
cd
intosrc/test/resources
directory- Compile the test class
ToyTest.java
withjavac -cp .:jtest.jar ToyTest.java
- Run the test class with
java -cp .:jtest.jar com.wlnirvana.JTestRunner ToyTest
After completing this task, run T9.java
to check your implementation.
Next Steps
Congratulations! You have successfully grown a 20-line skeleton project to a self-contained testing framework. But in case you want to dive deeper, here are some topics you may be interested in:
- Add
assertThrows
to check if an exception is thrown during a function call. Possible usage ofassertThrows
is given in comments inToyTest.ava
. - In task 3, we did a simplification that
@JTest
only be annotated onstatic
methods. JUnit takes a different approach, for a good reason. Can you modify your framework to support this as well? - Write more powerful runners. E.g. one that can run all test classes in a package, instead of a single test class.
Credits
This project is inspired by the following efforts.
- Alonso Del Arte's great tutorial Making A Java Unit Testing Framework From Scratch
- JUnit
Don’t Forget to Share This Post!
Comments (0)
No comments yet. Be the first.