0

使用 C# .NetFrameWork 4.7、Moq 1.4

我正在测试一种进行 2 次接口调用的方法。接口方法接受1个参数,该参数也是一个接口。由于很难对此进行测试,Moq 抛出了铸造错误。

具有以下签名的 2 次连续调用:

InterfaceUnderTest.MethodUnderTest(IEnumerable<IParamInterface>)

完整的代码示例可以在 github 上找到:https ://github.com/JayZhang727/UnitTestingInterfaceParams

以下是基本结构:

正在导入的我无法更改的接口和实现:

public interface IWorkInterface
{
    string DoWork(IEnumerable<IParamInterface> para);
}

public interface IParamInterface
{
    IParamDataInterface GetParamData();
}

public interface IParamDataInterface
{
    string Id { get; set; }
}

public class ParamClass<T> : IParamInterface where T : IParamDataInterface
{
    public T ParamData { get; set; }

    public ParamClass()
    {

    }

    public ParamClass(T para)
    {
        this.ParamData = para;
    }

    public IParamDataInterface GetParamData()
    {
        return ParamData;
    }
}

public class ParamClassA : IParamDataInterface
{
    public ParamClassA()
    {
        this.Id = "Id";
        this.ParamClassAVar = 123;
    }

    public string Id { get; set; }

    public int ParamClassAVar { get; set; }

}

public class ParamClassB : IParamDataInterface
{
    public ParamClassB()
    {
        this.Id = "Id";
        this.ParamClassBVar = "not 123";
    }

    public string Id { get; set; }

    public string ParamClassBVar { get; set; }

}

这是正在测试的类:

public ClassUnderTest(IWorkInterface workInt)
{
    this.WorkClient = workInt;
}

public IWorkInterface WorkClient { get; set; }

public string MethodUnderTest()
{
    var result = string.Empty;

    //Class A is an implementation of IParamDataInterface
    var a = new ParamClassA();
    var theParamA = new ParamClass<ParamClassA>(a);
    var listA = new List<IParamInterface>();
    listA.Add(theParamA);

    result = WorkClient.DoWork(listA);

    //Class B is also an implementation of IParamDataInterface
    var b = new ParamClassB();
    var theParamB = new ParamClass<ParamClassB>(b);
    var listB = new List<IParamInterface>();
    listB.Add(theParamB);

    result = result + WorkClient.DoWork(listB);

    return result;
}

这是我尝试过的测试调用:

private Mock<IWorkInterface> mockClient;

[TestMethod()]
public void DoWorkTest()
{
    mockClient = new Mock<IWorkInterface>(MockBehavior.Strict);
    var target = new ClassUnderTest(mockClient.Object);

    var mockSquence = new MockSequence();

    mockClient.InSequence(mockSquence).Setup(ec => ec.DoWork(It.Is<List<IParamInterface>>(el => ((ParamClassA)((ParamClass<ParamClassA>)el[0]).GetParamData()).ParamClassAVar == 123))).Returns("123");

    mockClient.InSequence(mockSquence).Setup(ec => ec.DoWork(It.Is<List<IParamInterface>>(el => ((ParamClassB)((ParamClass<ParamClassB>)el[0]).GetParamData()).ParamClassBVar == "not 123"))).Returns("not 123");

    //act
    target.MethodUnderTest();


    //assert
    mockClient.VerifyAll();
}

我收到关于无法从 A 类转换为 B 类的错误:

Message: Test method ClassUnderTesting.UnitTests.ClassUnderTestingTests.DoWorkTest threw exception: 
System.InvalidCastException: Unable to cast object of type 'ClassUnderTesting.ParamClass`1[ClassUnderTesting.ParamClassA]' to type 'ClassUnderTesting.ParamClass`1[ClassUnderTesting.ParamClassB]'.

起订量似乎没有按顺序进行,由于某种原因,第二个设置胜过第一个设置,并且顺序似乎没有帮助。有人知道我应该如何设置和测试这两个电话吗?

4

1 回答 1

1

我会在没有序列的情况下这样做。只需设置您的期望,以便在调用 DoWork 时ParamClassA返回“123”,而在调用时ParamClassB返回“not 123”。

请注意,如果没有给出第一个期望,它不会崩溃ParamClassA:它根本不匹配(同样是第二个 and ParamClassB)。

mockClient
    .Setup(ec => ec.DoWork(
        It.Is<List<IParamInterface>>(el => el[0] is ParamClass<ParamClassA> && ((ParamClass<ParamClassA>)el[0]).ParamData.ParamClassAVar == 123)
    ))
    .Returns("123");

mockClient
    .Setup(ec => ec.DoWork(
        It.Is<List<IParamInterface>>(el => el[0] is ParamClass<ParamClassB> && ((ParamClass<ParamClassB>)el[0]).ParamData.ParamClassBVar == "not 123")
    ))
    .Returns("not 123");
于 2019-02-07T15:36:03.650 回答