-
Notifications
You must be signed in to change notification settings - Fork 266
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Ability to mock protected methods with and without return value
- Loading branch information
1 parent
ab73157
commit 1941c3d
Showing
5 changed files
with
252 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,59 @@ | ||
using System.Reflection; | ||
using NSubstitute.Core; | ||
using NSubstitute.Core.Arguments; | ||
|
||
// Disable nullability for client API, so it does not affect clients. | ||
#nullable disable annotations | ||
|
||
namespace NSubstitute.Extensions; | ||
|
||
public static class ProtectedExtensions | ||
{ | ||
/// <summary> | ||
/// Configure behavior for a protected method with return value | ||
/// </summary> | ||
/// <typeparam name="T"></typeparam> | ||
/// <param name="obj">The object.</param> | ||
/// <param name="methodName">Name of the method.</param> | ||
/// <param name="args">The method arguments.</param> | ||
/// <returns>Result object from the method invocation.</returns> | ||
/// <exception cref="System.ArgumentNullException">Substitute - Cannot mock null object</exception> | ||
/// <exception cref="System.ArgumentException">Must provide valid protected method name to mock - methodName</exception> | ||
public static object Protected<T>(this T obj, string methodName, params object[] args) where T : class | ||
{ | ||
if (obj == null) { throw new ArgumentNullException(nameof(obj), "Cannot mock null object"); } | ||
if (string.IsNullOrWhiteSpace(methodName)) { throw new ArgumentException("Must provide valid protected method name to mock", nameof(methodName)); } | ||
|
||
IList<IArgumentSpecification> argTypes = SubstitutionContext.Current.ThreadContext.PeekAllArgumentSpecifications(); | ||
MethodInfo mthdInfo = obj.GetType().GetMethod(methodName, BindingFlags.NonPublic | BindingFlags.Instance, Type.DefaultBinder, argTypes.Select(x => x.ForType).ToArray(), null); | ||
|
||
if (mthdInfo == null) { throw new Exception($"Method {methodName} not found"); } | ||
if (!mthdInfo.IsVirtual) { throw new Exception($"Method {methodName} is not virtual"); } | ||
|
||
return mthdInfo.Invoke(obj, args); | ||
} | ||
|
||
/// <summary> | ||
/// Configure behavior for a protected method with no return vlaue | ||
/// </summary> | ||
/// <typeparam name="T"></typeparam> | ||
/// <param name="obj">The object.</param> | ||
/// <param name="methodName">Name of the method.</param> | ||
/// <param name="args">The method arguments.</param> | ||
/// <returns>WhenCalled<T>.</returns> | ||
/// <exception cref="System.ArgumentNullException">Substitute - Cannot mock null object</exception> | ||
/// <exception cref="System.ArgumentException">Must provide valid protected method name to mock - methodName</exception> | ||
public static WhenCalled<T> When<T>(this T obj, string methodName, params object[] args) where T : class | ||
{ | ||
if (obj == null) { throw new ArgumentNullException(nameof(obj), "Cannot mock null object"); } | ||
if (string.IsNullOrWhiteSpace(methodName)) { throw new ArgumentException("Must provide valid protected method name to mock", nameof(methodName)); } | ||
|
||
IList<IArgumentSpecification> argTypes = SubstitutionContext.Current.ThreadContext.PeekAllArgumentSpecifications(); | ||
MethodInfo mthdInfo = obj.GetType().GetMethod(methodName, BindingFlags.NonPublic | BindingFlags.Instance, Type.DefaultBinder, argTypes.Select(y => y.ForType).ToArray(), null); | ||
|
||
if (mthdInfo == null) { throw new Exception($"Method {methodName} not found"); } | ||
if (!mthdInfo.IsVirtual) { throw new Exception($"Method {methodName} is not virtual"); } | ||
|
||
return new WhenCalled<T>(SubstitutionContext.Current, obj, x => mthdInfo.Invoke(x, args), MatchArgs.AsSpecifiedInCall); | ||
} | ||
} |
46 changes: 46 additions & 0 deletions
46
tests/NSubstitute.Acceptance.Specs/Infrastructure/AnotherClass.cs
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,46 @@ | ||
namespace NSubstitute.Acceptance.Specs.Infrastructure; | ||
|
||
public abstract class AnotherClass | ||
{ | ||
protected abstract string ProtectedMethod(); | ||
|
||
protected abstract string ProtectedMethod(int i); | ||
|
||
protected abstract string ProtectedMethod(string msg, int i, char j); | ||
|
||
protected abstract void ProtectedMethodWithNoReturn(); | ||
|
||
protected abstract void ProtectedMethodWithNoReturn(int i); | ||
|
||
protected abstract void ProtectedMethodWithNoReturn(string msg, int i, char j); | ||
|
||
public string DoWork() | ||
{ | ||
return ProtectedMethod(); | ||
} | ||
|
||
public string DoWork(int i) | ||
{ | ||
return ProtectedMethod(i); | ||
} | ||
|
||
public string DoWork(string msg, int i, char j) | ||
{ | ||
return ProtectedMethod(msg, i, j); | ||
} | ||
|
||
public void DoVoidWork() | ||
{ | ||
ProtectedMethodWithNoReturn(); | ||
} | ||
|
||
public void DoVoidWork(int i) | ||
{ | ||
ProtectedMethodWithNoReturn(i); | ||
} | ||
|
||
public void DoVoidWork(string msg, int i, char j) | ||
{ | ||
ProtectedMethodWithNoReturn(msg, i, j); | ||
} | ||
} |
123 changes: 123 additions & 0 deletions
123
tests/NSubstitute.Acceptance.Specs/ProtectedExtensionsTests.cs
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,123 @@ | ||
using NSubstitute.Acceptance.Specs.Infrastructure; | ||
using NSubstitute.Extensions; | ||
using NUnit.Framework; | ||
|
||
namespace NSubstitute.Acceptance.Specs; | ||
|
||
public class ProtectedExtensionsTests | ||
{ | ||
[Test] | ||
public void Should_mock_and_verify_protected_method_with_no_args() | ||
{ | ||
var expectedMsg = "unit test message"; | ||
var sub = Substitute.For<AnotherClass>(); | ||
var worker = new Worker(); | ||
|
||
sub.Protected("ProtectedMethod").Returns(expectedMsg); | ||
|
||
Assert.That(worker.DoWork(sub), Is.EqualTo(expectedMsg)); | ||
sub.Received(1).Protected("ProtectedMethod"); | ||
} | ||
|
||
[Test] | ||
public void Should_mock_and_verify_protected_method_with_arg() | ||
{ | ||
var expectedMsg = "unit test message"; | ||
var sub = Substitute.For<AnotherClass>(); | ||
var worker = new Worker(); | ||
|
||
sub.Protected("ProtectedMethod", Arg.Any<int>()).Returns(expectedMsg); | ||
|
||
Assert.That(worker.DoMoreWork(sub, 5), Is.EqualTo(expectedMsg)); | ||
var a = sub.Received(1); | ||
a.Protected("ProtectedMethod", Arg.Any<int>()); | ||
} | ||
|
||
[Test] | ||
public void Should_mock_and_verify_protected_method_with_multiple_args() | ||
{ | ||
var expectedMsg = "unit test message"; | ||
var sub = Substitute.For<AnotherClass>(); | ||
var worker = new Worker(); | ||
|
||
sub.Protected("ProtectedMethod", Arg.Any<string>(), Arg.Any<int>(), Arg.Any<char>()).Returns(expectedMsg); | ||
|
||
Assert.That(worker.DoEvenMoreWork(sub, 3, 'x'), Is.EqualTo(expectedMsg)); | ||
sub.Received(1).Protected("ProtectedMethod", Arg.Any<string>(), Arg.Any<int>(), Arg.Any<char>()); | ||
} | ||
|
||
[Test] | ||
public void Should_mock_and_verify_method_with_no_return_and_no_args() | ||
{ | ||
var count = 0; | ||
var sub = Substitute.For<AnotherClass>(); | ||
var worker = new Worker(); | ||
|
||
sub.When("ProtectedMethodWithNoReturn").Do(x => count++); | ||
|
||
worker.DoVoidWork(sub); | ||
Assert.That(count, Is.EqualTo(1)); | ||
sub.Received(1).Protected("ProtectedMethodWithNoReturn"); | ||
} | ||
|
||
[Test] | ||
public void Should_mock_and_verify_method_with_no_return_with_arg() | ||
{ | ||
var count = 0; | ||
var sub = Substitute.For<AnotherClass>(); | ||
var worker = new Worker(); | ||
|
||
sub.When("ProtectedMethodWithNoReturn", Arg.Any<int>()).Do(x => count++); | ||
|
||
worker.DoVoidWork(sub, 5); | ||
Assert.That(count, Is.EqualTo(1)); | ||
sub.Received(1).Protected("ProtectedMethodWithNoReturn", Arg.Any<int>()); | ||
} | ||
|
||
[Test] | ||
public void Should_mock_and_verify_method_with_no_return_with_multiple_args() | ||
{ | ||
var count = 0; | ||
var sub = Substitute.For<AnotherClass>(); | ||
var worker = new Worker(); | ||
|
||
sub.When("ProtectedMethodWithNoReturn", Arg.Any<string>(), Arg.Any<int>(), Arg.Any<char>()).Do(x => count++); | ||
|
||
worker.DoVoidWork(sub, 5, 'x'); | ||
Assert.That(count, Is.EqualTo(1)); | ||
sub.Received(1).Protected("ProtectedMethodWithNoReturn", Arg.Any<string>(), Arg.Any<int>(), Arg.Any<char>()); | ||
} | ||
|
||
private class Worker | ||
{ | ||
internal string DoWork(AnotherClass worker) | ||
{ | ||
return worker.DoWork(); | ||
} | ||
|
||
internal string DoMoreWork(AnotherClass worker, int i) | ||
{ | ||
return worker.DoWork(i); | ||
} | ||
|
||
internal string DoEvenMoreWork(AnotherClass worker, int i, char j) | ||
{ | ||
return worker.DoWork("worker", i, j); | ||
} | ||
|
||
internal void DoVoidWork(AnotherClass worker) | ||
{ | ||
worker.DoVoidWork(); | ||
} | ||
|
||
internal void DoVoidWork(AnotherClass worker, int i) | ||
{ | ||
worker.DoVoidWork(i); | ||
} | ||
|
||
internal void DoVoidWork(AnotherClass worker, int i, char j) | ||
{ | ||
worker.DoVoidWork("void worker", i, j); | ||
} | ||
} | ||
} |