In this tutorial we will show how to install and use SimpleStubs.
To install SimpleStubs, simply install the Etg.SimpleStubs
NuGet package to your unit test project (also see Tips and Tricks section for installation tips).
Once installed, SimpleStubs will create stubs for all interfaces (public
interfaces and optionally internal
interfaces) in all referenced projects.
The generated stubs will be added to the Properties\SimpleStubs.generated.cs
file and compiled as part of the build process. Because the stubs are generated, modifying the stubs manually has no effect (if you really want to modify a stub, copy it to a different file).
Important note for UWP: Because of a limitation in NuGet support for UWP (see discussion here), the Properties\SimpleStubs.generated.cs
file will not be automatically added to UWP projects and must be manually added (simply add the file to the Properties
folder of your test project).
Let's consider the following interface and look at how we can use SimpleStubs to create stubs for it.
public interface IPhoneBook
{
long GetContactPhoneNumber(string firstName, string lastName);
long MyNumber { get; set; }
event EventHandler<long> PhoneNumberChanged;
}
// for any number of calls
var stub = new StubIPhoneBook().GetContactPhoneNumber((firstName, lastName) => 6041234567);
or
// For one call; an exception will be thrown if more calls occur
var stub = new StubIPhoneBook().GetContactPhoneNumber((firstName, lastName) => 6041234567, Times.Once);
or
// For a specific number of calls
var stub = new StubIPhoneBook().GetContactPhoneNumber((firstName, lastName) => 6041234567, count:5);
You can also copy and verify the parameters values:
string firstName = null;
string lastName = null;
var stub = new StubIPhoneBook().GetContactPhoneNumber((fn, ln) =>
{
firstName = fn;
lastName = ln;
return number;
});
ClassUnderTest obj = new ClassUnderTest(stub);
Assert.AreEqual(expectedValue, obj.Foo());
// parameters verification
Assert.AreEqual("John", firstName);
Assert.AreEqual("Smith", lastName);
object someObj = new Foo();
var stub = new StubIContainer()
.GetElement((int index, out object value) =>
{
value = someObj;
return true;
});
var stub = new StubIRefUtils()
.Swap<int>((ref int v1, ref int v2) =>
{
int temp = v1;
v1 = v2;
v2 = temp;
});
int value = -1;
var stub = new StubIContainer()
.GetElement<int>(index => value)
.SetElement<int>((i, v) => { value = v; });
long myNumber = 6041234567;
var stub = new StubIPhoneBook()
.MyNumber_Get(() => myNumber)
.MyNumber_Set(value => newNumber = value);
var stub = new StubIGenericContainer<int>();
// stubbing indexer getter
stub.Item_Get(index =>
{
// we're expecting the code under test to get index 5
if (index != 5) throw new IndexOutOfRangeException();
return 99;
});
// stubbing indexer setter
int res = -1;
stub.Item_Set((index, value) =>
{
// we're expecting the code under test to only set index 7
if (index != 7) throw new IndexOutOfRangeException();
res = value;
});
var stub = new StubIPhoneBook();
// Pass the stub to the code under test
var obj = new ClassUnderTest(stub);
// Raise the event
stub.PhoneNumberChanged_Raise(stub, 55);
// Verify the state of obj to ensure that it has reacted to the event
In some cases, it might be useful to have a stub behave differently when it's called several times. SimpleStubs offers supports for stubbing a sequence of calls.
var stub = new StubIPhoneBook()
.GetContactPhoneNumber((p1, p2) => 12345678, Times.Once) // first call
.GetContactPhoneNumber((p1, p2) => 11122233, Times.Twice) // next two calls
.GetContactPhoneNumber((p1, p2) => 22233556, Times.Forever); // rest of the calls
It's possible to overwrite a stubbed method or property as follows:
var stub = new StubIPhoneBook().GetContactPhoneNumber((p1, p2) => 12345678);
// test code
// overwrite the stub
stub.GetContactPhoneNumber((p1, p2) => 11122233, overwrite:true);
// other test code
SimpleStubs also supports an optional configuration file that can be added to the root of your test project. The configuration file (named SimpleStubs.json
) has the following structure:
{
"IgnoredProjects": [
"IgnoredProject1",
"IgnoredProject2"
],
"IgnoredInterfaces": [
"MyNamespace.IFooInterface",
"MyNamespace.IBarInterface"
],
"StubInternalInterfaces": false,
"StubCurrentProject": false
}
The configuration file allows you to instruct SimpleStubs to omit creating stubs for a given project or interface. Note that this very useful to exclude interfaces that are causing SimpleStubs to generate stubs that don't compile (this can happen in some edge cases). If you encounter such a case, exclude the interface in question and report the problem so we can fix it.
It's also possible to instruct SimpleStubs to create stubs for internal interfaces (by default only public interfaces are stubbed) as shown in the configuration sample above.
It's also possible to generate stubs for interfaces from current project (no tonly referenced projects) as shown in the configuration sample above. It's useful if you use shared project as reference.
If you have a solution composed of multiple projects, instead of installing SimpleStubs to each of your test projects,
- Create a new project that will contain the stubs, call it something like GeneratedStubs
- Install SimpleStubs to the GeneratedStubs project.
- Reference all the projects in the solution from the GeneratedStubs project (or at least the projects that contain interfaces to be stubbed).
- Reference the GeneratedStubs project in your test projects to access the stubs.
With this approach, each stub is generated only once and only when there are code changes.
By default, SimpleStubs ignores external interfaces because there can be too many of them to generate stubs for. One simple way of generating a stub for an external interface is to create an internal interface that inherits from it (you don't need to use the internal fake interface anywhere in the code). Because SimpleStubs handles inheritance, the generated stub can be used as a stub of the original external interface.
- Only interfaces are stubbed.
- No stubs will be generated for external libraries (dlls). See Tips and Tricks section for a workaround.
Exclude the interface that is causing the problem (using the SimpleStubs.json
configuration file) and report the problem by opening an issue.