Getting Started with Unit Testing and Moq
We had a new team lead start recently, he seems to have had a fair amount of experience in areas I’m only vaguely familiar with, mostly through reading. One of the first things being pushed for is a concentration on unit testing. While I did begin implementing some tests into our codebase a few months ago (around 250 so far), I feel that there’s still a long way to go. Luckily Chris is here to help impart knowledge to us, hooray!
Part 1 you are here
Part 2
Part 3
Part 4
So getting started with unit testing - first it should be defined what a unit test is.
From https://en.wikipedia.org/wiki/Unit_testing:
In computer programming, unit testing is a software testing method by which individual units of source code, sets of one or more computer program modules together with associated control data, usage procedures, and operating procedures, are tested to determine whether they are fit for use.
Given the following classes/methods:
1 |
|
Note in the above, I am using interfaces to allow for the injection of dependencies (an important part of unit testing with mocks, and in general). The basic idea is you provide sample (unimportant) implementations to the dependent pieces of the whole, the pieces that are not currently being tested, therefore are unimportant to the test - at least when testing the “Put it all together” methods.
I see the following things that need to be tested - there could very well be more, but here it is at a glance:
INumberFunctions.AddNumbers
IDbGetSomeNumbers.GetSomeNumbers
GetNumbersAndAddThem.Execute
There are a few other stragglers in there that will become apparent (if they aren’t already) like null testing parameters in the constructor, testing empty array for add numbers, etc.
For
INumbersFunctions.AddNumbers
, we need to of course, check the numbers are being added properly. I have accomplished that with the following tests:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89using System;
using System.Diagnostics.CodeAnalysis;
using System.Linq;
using Microsoft.VisualStudio.TestTools.UnitTesting;
namespace RussUnitTestSample.Business.Tests
{
[TestClass]
[ExcludeFromCodeCoverage]
public class NumberFunctionTests
{
#region Public methods
/// <summary>
/// Test exception thrown when numbers provided is null
/// </summary>
[ExpectedException(typeof(ArgumentNullException))]
[TestMethod]
public void AddNumbers_NullParameterNumbers()
{
// Arrange / Act / Assert
NumberFunctions nf = new NumberFunctions();
var result = nf.AddNumbers(null);
}
/// <summary>
/// Test exception thrown when 0 numbers provided in array
/// </summary>
[ExpectedException(typeof(ArgumentException))]
[TestMethod]
public void AddNumbers_EmptyArrayNumbers()
{
// Arrange / Act / Assert
NumberFunctions nf = new NumberFunctions();
var result = nf.AddNumbers(new double[] { });
}
/// <summary>
/// Add two positive numbers
/// </summary>
[TestMethod]
public void AddNumbers_TwoNumbers()
{
// Arrange
double[] numbers = { 1, 2 };
NumberFunctions nf = new NumberFunctions();
// Act
var result = nf.AddNumbers(numbers);
// Assert
Assert.AreEqual(numbers.Sum(), result);
}
/// <summary>
/// Add 10 numbers mixed positive and negative
/// </summary>
[TestMethod]
public void AddNumbers_TenNumbersWithPositiveAndNegatives()
{
// Arrange
double[] numbers = { 1, -2, 3, 4, -5, 6, 7, -8, 9, 10 };
NumberFunctions nf = new NumberFunctions();
// Act
var result = nf.AddNumbers(numbers);
// Assert
Assert.AreEqual(numbers.Sum(), result);
}
[TestMethod]
public void AddNumbers_ProvideOneNumber()
{
// Arrange
double[] numbers = { 1 };
NumberFunctions nf = new NumberFunctions();
// Act
var result = nf.AddNumbers(numbers);
// Assert
Assert.AreEqual(numbers.Sum(), result);
}
#endregion Public methods
}
}There is most definitely some overlap in some of the tests, but with something so simple there’s still quite a few! I don’t feel the actual implementation of NumberFunctions is important, as I’m concentrating on the tests.
That takes care of the class that has completed non Moq-ed tests. In the next post (which I will hopefully do soon) I’ll go into how I accomplished my first unit tests with Moq. If I can get a well enough cadence going on, I hope to cover Moq-ing WCF service calls - as that’s what we use at work for our communication to our DB, so being able to moq that as well would be beneficial.
Full code including the Moq unit tests can be found at: https://github.com/Kritner/RussUnitTestSample
Getting Started with Unit Testing and Moq
https://blog.kritner.com/2015/11/30/Getting-Started-with-Unit-Testing-and-Moq/