An Introduction to Unit Testing - ISISComputingGroup/ibex_developers_manual GitHub Wiki
To create unit tests for an Eclipse plug-in a Fragment Project is used. When creating a Fragment Project we assign the plug-in we wish to test as a Host Plug-in. Eclipse automatically gives the Fragment access to the classes in the original plug-in. In the Fragment Project we create classes to test the classes in the original plug-in.
Open the wizard for creating a standard plug-in in Eclipse (File->New->Plug-in Project) and complete the following steps (if values not specified then use the defaults):
- Set "Project name" to org.myexample.plugin
- Click "Next"
- Uncheck "Generate an activator, a Java class that controls the plug-in's life cycle"
- Uncheck "This plug-in will make contributions to the UI"
- Put "No" for "Would you like to create a rich client application?"
- Click "Finish"
The will create the plug-in. Inside the src folder create a package called org.myexample.plugin.classes and add a class called StringManipulator. Add the following code to the class:
package org.myexample.plugin.classes;
public class StringManipulator {
public String addStrings(String one, String two) {
return one;
}
}
Now to create the Fragment Project. Open the Fragment Project wizard under File->New->Other->Plug-in Development->Fragment Project and complete the following steps:
- Set "Project name" to org.myexample.plugin.tests (i.e the original plug-in name plus ".tests" - this is our naming convention)
- Click "Next"
- Under "Host Plug-in" click the "Browse" button and select the original plug-in
- Click "Finish"
Eclipse will now create the Fragment Project. We need to manually add the JUnit plug-in as a dependency for the Fragment Project, to do this:
- Open the MANIFEST.MF file
- Select the "Dependencies" tab
- Under "Required Plug-ins" click the "Add" button
- In the dialog, type org.junit and select the plug-in listed (it should be version 4+)
- Click "OK"
- Save the changes
In the src directory of the Fragment Project create a package called "org.myexample.plugin.tests". Add a class called "StringManipulatorTest" - the name MUST end in Test for the build system to recognise it.
Okay let's create a test, add the following code to the StringManipulatorTest class:
package org.myexample.plugin.tests;
import static org.junit.Assert.*;
import org.junit.Test;
import org.myexample.plugin.classes.StringManipulator;
public class StringManipulatorTest {
@Test
public void add_hello_to_world() {
// Arrange
StringManipulator strMan = new StringManipulator();
// Act
String ans = strMan.addStrings("Hello", "World");
// Assert
assertEquals("HelloWorld", ans);
}
}
To run the test right-click on the Fragment Project and select Run As->JUnit Test. The test should run and fail like so:
Clearly there is something wrong with the original code in addStrings, so let's fix that by changing:
public String addStrings(String one, String two) {
return one;
}
to:
public String addStrings(String one, String two) {
return one + two;
}
Now the test should pass if it is run again:
JUnit has many useful features, here are a select few.
-
other asserts such as assertTrue, assertArrayEquals, assertNotEqual and assertNotNull
-
assert that an error is thrown:
@Test(expected=IndexOutOfBoundsException.class)
public void raises_IndexOutOfBoundsException() {
ArrayList emptyList = new ArrayList();
Object o = emptyList.get(0);
}
- set-up and teardown methods - these are code snippets that are run before and after each test:
private List<String> names;
@Before
public void setUp() {
// Called before each test
names = new ArrayList<String>();
names.add("Tom");
names.add("Dick");
names.add("Harry");
}
@After
public void tearDown() {
// Called after each test
names.clear();
}
@Test
public void concatenate_names() {
// Arrange
StringManipulator strMan = new StringManipulator();
// Act
String ans = strMan.concatenateNames(names);
// Assert
assertEquals("Tom, Dick and Harry", ans);
}
Note: Each test should be independent of the other tests as there is no guarantee of the order they are run in.
- BeforeClass and AfterClass - these are run once before the first test and after the last test in a class respectively:
@BeforeClass
public static void oneTimeSetUp() {
// Perhaps create a dummy file or something shared by more than one test
}
@AfterClass
public static void oneTimeTearDown() {
// Clean up
}
See test naming.
Mockito is a framework for creating mock objects that can be substituted for real objects to make testing easier and more specific. For example: writing tests that don't rely on a database, file or network connection being present.
Like JUnit is can be used inside a Fragment Project after the dependency is added (org.mockito).
An example of using Mockito would be to mock a database wrapper so that a real database is not required:
@Test
public void get_row_data() {
// Arrange
// Create a mock database wrapper as we are not testing that
DatabaseWrapper wrapper = mock(DatabaseWrapper.class);
// Create a mock "response"
List<String> data = new ArrayList<String>();
data.add("John");
data.add("Smith");
data.add("01/01/1955");
when(wrapper.getRowData(0)).thenReturn(data); // This is the key line
// This is the object we are really testing
DataHolder dataHolder = new DataHolder(wrapper);
// Act
List<String> ans = dataHolder.getFirstRow();
// Assert
assertEquals(data, ans);
}
For more detail on Mockito see the using_mockito_for_testing_in_ibex document.
It is useful to see what parts of a plug-in's code are used or not used by the unit tests. If a piece of code is not used by the unit tests then that may mean that an extra test is required
Unit test code coverage can be examined inside Eclipse using EclEmma which can be installed via the Eclipse Marketplace (under the "Help" menu). Once EclEmma is installed the coverage of the unit tests can be examined. Right-click on the test project and select Coverage As->JUnit Test. This will run the tests and calculate the coverage, the results should look something like this:
From the results it can be seen that 63.2% of the StringManipulator code is used by the unit tests. The code that isn't used is highlighted in red - for this example we can see that we need to write a test that tests the reverseString method.
Running the tests in Eclipse might crash with an error like:
Class not found org.myexample.plugin.tests.StringManipulatorTest
java.lang.ClassNotFoundException: org.myexample.plugin.tests.StringManipulatorTest
at java.net.URLClassLoader.findClass(Unknown Source)
at java.lang.ClassLoader.loadClass(Unknown Source)
at sun.misc.Launcher$AppClassLoader.loadClass(Unknown Source)
at java.lang.ClassLoader.loadClass(Unknown Source)
...
This is a known bug and there is a workaround:
- Right-click on the test plug-in, and under "Run As" select "Run Configurations"
- In the new dialog click on the "Classpath" tab
- Select "User Entries" and click the "Advanced" button
- Select "Add Folders" and click "OK"
- In the new dialog, expand the test plug-in and select the "bin" folder and click "OK"
- On the original dialog, click "Apply" and then "Run"
- Hopefully, the tests will now work and you should be able to re-run them in the normal way
If Eclipse is not picking up changes when you add tests you may need to change the default output folder for tests for Maven to pick it up.
- Right-click on the tests plug-in, go to properties, Java build path
- Change the output folder to target/test-classes (you may need to create this folder first by clicking browse, selecting target and adding the test-classes folder)
- If this does not work try deleting the target/test-classes folder first, if it existed already, and do a clean rebuild of the workspace
If the tests are failing because of an IncompatibleClassChangeError error then the solution is to delete the bin and target folders for both the main plug-in and the corresponding test plug-in