The simplest way is as an expression in a debugger. You can change debug expressions without recompiling, and you can wait to decide what to write until you have seen the running objects. You can also write test expressions as statements which print to the standard output stream. Both styles of tests are limited because they require human judgment to analyze their results. Also, they don't compose nicely- you can only execute one debug expression at a time and a program with too many print statements causes the dreaded "Scroll Blindness".
NUnit tests do not require human judgment to interpret, and it is easy to run many of them at the same time. When you need to test something, here is what you do:
public void TestSimpleAdd() { Money m12CHF= new Money(12, "CHF"); Money m14CHF= new Money(14, "CHF"); Money expected= new Money(26, "CHF"); Money result= m12CHF.Add(m14CHF); Assert(expected.Equals(result)); }If you want to write a test similar to one you have already written, write a Fixture instead. When you want to run more than one test, create a Suite.
Tests need to run against the background of a known set of objects. This set of objects is called a test fixture. When you are writing tests you will often find that you spend more time writing the code to set up the fixture than you do in actually testing values.
To some extent, you can make writing the fixture code easier by paying careful attention to the constructors you write. However, a much bigger saving comes from sharing fixture code. Often, you will be able to use the same fixture for several different tests. Each case will send slightly different messages or parameters to the fixture and will check for different results.
When you have a common fixture, here is what you do:
public class MoneyTest: TestCase { private Money f12CHF; private Money f14CHF; private Money f28USD; protected override void SetUp() { f12CHF= new Money(12, "CHF"); f14CHF= new Money(14, "CHF"); f28USD= new Money(28, "USD"); } }Once you have the Fixture in place, you can write as many Test Cases as you'd like.
Writing a test case without a fixture is simple- override RunTest in a subclass of TestCase. You write test cases for a Fixture the same way, by making a subclass of TestCase for your set up code and then making subclasses for the individual test cases. However, after a few such tests you would notice that a large percentage of your lines of code are sacrificed to syntax.
NUnit provides a more concise way to write a test against a Fixture. Here is what you do:
public void TestMoneyMoneyBag() { // [12 CHF] + [14 CHF] + [28 USD] == {[26 CHF][28 USD]} Money bag[]= { new Money(26, "CHF"), new Money(28, "USD") }; MoneyBag expected= new MoneyBag(bag); AssertEquals(expected, f12CHF.Add(f28USD.Add(f14CHF))); }Create an instance of MoneyTest that will run this test case like this:
new MoneyTest("TestMoneyMoneyBag")When the test is run, the name of the test is used to look up the method to run.
Once you have several tests, organize them into a Suite.
As soon as you have two tests, you'll want to run them together. You could run the tests one at a time yourself, but you would quickly grow tired of that. Instead, NUnit provides an object, TestSuite which runs any number of test cases together.
For example, to run a single test case, you execute:
TestResult result= (new MoneyTest("TestMoneyMoneyBag")).Run();To create a suite of two test cases and run them together, execute:
TestSuite suite= new TestSuite(); suite.AddTest(new MoneyTest("TestMoneyEquals")); suite.AddTest(new MoneyTest("TestSimpleAdd")); TestResult result= suite.Run();Another way is to let NUnit extract a suite from a TestCase. To do so you pass the class of your TestCase to the
TestSuite suite= new TestSuite(typeof(MoneyTest));
TestResult result= suite.Run();
Use the manual way when you want a suite to only contain a subset of the test cases. Otherwise the automatic suite extraction is the preferred way. It avoids you having to update the suite creation code when you add a new test case.
TestSuites don't only have to contain TestCases. They contain any object that implements the ITest interface. For example, you can create a TestSuite in your code and I can create one in mine, and we can run them together by creating a TestSuite that contains both:
TestSuite suite= new TestSuite(); suite.AddTest(Kent.Suite()); suite.AddTest(Erich.Suite()); TestResult result= suite.Run();
Once you have a test suite, you'll want to run it. NUnit provides tools
to define the suite to be run and to display its results. You make your
suite accessible to a TestRunner tool with a static property Suite
that returns a test suite.
For example, to make a MoneyTest suite available to a TestRunner, add
the following code to MoneyTest:
public static ITest Suite { get { TestSuite suite= new TestSuite(); suite.addTest(new MoneyTest("TestMoneyEquals")); suite.addTest(new MoneyTest("TestSimpleAdd")); return suite; } }If a TestCase class doesn't define a suite method a TestRunner will extract a suite and fill it with all the methods starting with "Test".
NUnit provides both a graphical and a textual version of a TestRunner tool. Start it by typing NUnit. The graphical user interface presents a window with:
In a dynamic programming environment which reloads compiled classes you can leave the NUnit window up all the time. In other environments you have to restart the graphical version for each run. This is tedious and time consuming and will be fixed in a future version.
There is a batch interface to NUnit, also. To use it type NUnitConsole followed by the Assembly-qualified name of the class with a Suite property at an operating system prompt.
For example, to start the batch TestRunner for MoneyTest, you might write:
C:\NUnitConsole NUnit.Samples.Money.MoneyTest,NUnitSamples.dll
The batch interface shows the result as text output. An alternative way to invoke the batch interface is to define a main method in your TestCase class.
For example, to start the batch TestRunner for MoneyTest, write:
public static void Main(String[] args) { NUnit.TextUI.TestRunner.Run(Suite); }With this definition of Main you can run your tests by simply typing MoneyTest at an operating system prompt.