I've been looking at the Moq documentation and the comments are too short for me to understand each of things it can do.
The first thing I don't get is It.IsAny<string>(). //example using string
Is there an advantage of using this over just putting some value in? I know people say to use this if you don't care about the value, but if you don't care about the value can't you just do "a" or something? This just seems like more typing.
Secondly, when would be an example be of when you would not care about the value? I thought Moq needs the value to match up stuff.
I don't get what It.Is<>
is for at all or how to use it. I don't understand the example and what it is trying to show.
Next, I don't get when to use Times
(and its AtMost
methods and similar). Why would you limit the number of times something is set up? I have some AppConfig
value that I need to use twice. Why would I want to limit it to, say, once? This would just make the test fail. Is this to stop other people from adding another one to your code or something?
I don't get how to use mock.SetupAllProperties();
What does it set up the properties with?
I don't also get why there are so many different ways to set up a property and what their differences are. The documentation has:
SetupGet(of property)
SetupGet<TProperty>
I noticed that a lot of the stuff in Moq shows ()
and <>
- what's the difference between them and what would they look like in use?
I also don't get why they have SetupGet
. Would you not use SetupSet
to set a property?
SetupSet
has five different ways to use it in the documentation. Plus another one called SetupProperty
. So I don't understand why there are so many.
On a side note, I am wondering if variables used in lambdas are independent of other lambdas. E.g.:
mock.setup(m => m.Test);
stop.setup(m => m.Test);
Would this be ok or would there be some conflict between the variable m
?
Finally, I was watching this video and I am wondering if it shows Visual Studio. His Intellisense looks different. A lightbulb pops up for him (I am happy mine does not, as it brings back painful memories of netbeans), and there are lines going from one opening brace to the closing brace and etc.
Thanks :)
If you don't care about the exact value of a property, it's far better to use .IsAny because you are being explicit about the fact that the exact value is not important. If you hardcode it as "abc", then it is not clear if your code you are testing depends on starting with "a" or ending with "c" or being 3 chars long, etc. etc.
It.IsAny / It.Is
These can be useful when you're passing a new reference type within the code under test. For instance if you had a method along the lines of:-
You might want to check the add method has been called on the repository
If you wanted to make this test more explicit you can use It.Is by supplying a predicate the person object must match
This way the test will through an exception if the person object that was used to call the add method didn't have the age property set to 12.
Times
If you had a method along the lines of:-
One of the things that you might want to test is that the pay method does not get called when a person aged over 65 is passed into the method
Similarly it's possible to imagine situations where you're iterating over a collection and calling a method for each item in the collection and you'd like to make sure that it's been called a certain amount of times, other times you simply don't care.
SetupGet / SetupSet
What you need to be aware of with these guys is that they reflect how your code is interacting with the mock rather than how you're setting up the mock
In this case the code is setting the ModifiedBy property of the IAuditable instance while it's getting the Name property of the current instance of IPrincipal
In this case we're setting up the name property on the mock of IPrincipal so it returns "test" when the getter is called on the Name property of Identity we're not setting the property itself.
SetupProperty / SetupAllProperties
Looking at the test above if it was changed to read
The test would fail. This is because the proxy created by Moq doesn't actually do anything in the set method of a property unless you tell it to. In affect the mock object looks a bit like this
To get the test to pass you have to tell Moq to setup the property to have the standard property behaviour. You can do this by calling SetupProperty and the mock will look more like
and the test above would pass as the value "test" would now get stored against the mock. When mocking complex objects you might want to do this for all properties, hence the SetupAllProperties shortcut
Finally, the lightbulb in the IDE is the resharper plugin.