C#: Mocking and testing protected (or private) methods in sealed classes -- approaches
I have a sealed class with protected methods whose behaviour I want to test. This makes it h开发者_如何转开发ard to test directly, and hard to mock.
It's in a codebase that wasn't developed in a TDD manner, and I'm now adding unit tests for specific functionality.
What are the general approaches possible in this case? At the moment I have:
- Have the class unsealed. Then create a proxy or adapter derived from the class in our test code to tunnel access to the protected method.
- Factor out the behaviour in the protected method to a delegate/functor, and re-inject it. Then test the factored out behaviour independently.
- Test by calling the closest public method in the inheritance hierarchy that uses the protected method. Potentially leads to lots of mocking, and exposure to risk when code other than that under test changes -- creating fragile tests.
- Use reflection to get access to the protected method. Then call it directly.
Are there any more?
A protected
method in a sealed class is effectively the same as private
(I guess if you seal a derived class where the base class has a protected
member these can come up naturally.)
And there's no point testing private
methods. Because they have no public
behavior except that which is accessible through the public
methods of the defining class, their behavior should be tested by testing the public
methods.
Microsoft Moles helps mocking unsealed classes with non-virtual methods. It can't mock private methods, but this is redundant, because you can mock higher level public methods, that used outside particular class and you can emulate all necessary behaviour mocking one public method.
And why you should test private/protected methods? You can use internal methods and InternalVisibleToAttribute to achieve this. But in general you should test only public behaviour (i.e. only public interface).
I would generally go for the second option assuming that you have the ability to make the changes reasonably easy. Given that if you cannot test what a protected /private method is doing through it's public interface it probably isn't conforming to Single responsibility principle anyway and the code could probably do with being broken into two classes and use composition instead.
You can use some framework for mocking - for example JustMock (by Telerik) supports mock of sealed private methods ... Unfortunately, this part of JustMock is paid, I think, but at least you can try the trial version.
You could use the JustMock framework. For example:
double value = 0;
var fakeFilterSetHelper = Mock.Create<FilterSetHelper>(Behavior.CallOriginal);
Mock.NonPublic.Arrange<double>(fakeFilterSetHelper, memberName: "GetPriceRangeFromSession").Returns(value);
精彩评论