请有人看看下面的演示代码,让我知道我看到的是我的错误或一个Telerik问题?
我使用的是Telerik.JustMock诉2014.1.1519.1。以及Microsoft.VisualStudio.QualityTools.UnitTestFramework诉10.0.0.0。
正如代码注释所指出的,当id变量相等时(每个id调用一次),但当id变量不同时,我就会得到预期的结果。当我通过第一个测试时,我可以看到预期的调用正在进行,但是JustMock告诉我它们没有发出。
我会感激任何建设性的想法。希望这不是我睡眠不足的情况..。
[TestClass]
public class RunnerTests
{
[TestMethod]
public void MakeTwoCallsDifferentIdsFails()
{
int idOne=1;
int idTwo=2;
DataTable dt=new DataTable();
dt.Columns.Add("Id");
dt.Rows.Add(idOne);
dt.Rows.Add(idTwo);
IProcessor mock = Mock.Create<IProcessor>();
Runner runner = new Runner(mock);
runner.Process(dt);
Mock.Assert(()=>mock.Process(Arg.IsAny<MyArgs>()), Occurs.Exactly(2));
//The following two asserts fail (with 0 calls made to mock), regardless of sequence:
Mock.Assert(()=>mock.Process(Arg.Matches<MyArgs>
(d=>d.Id==idOne)),Occurs.Once());
Mock.Assert(()=>mock.Process(Arg.Matches<MyArgs>
(d=>d.Id==idTwo)),Occurs.Once());
}
[TestMethod]
public void MakeTwoCallsSameIdPasses()
{
//ids intentionally equal:
int idOne=1;
int idTwo=1;
DataTable dt=new DataTable();
dt.Columns.Add("Id");
dt.Rows.Add(idOne);
dt.Rows.Add(idTwo);
IProcessor mock = Mock.Create<IProcessor>();
Runner runner = new Runner(mock);
runner.Process(dt);
//all asserts pass:
Mock.Assert(()=>mock.Process(Arg.IsAny<MyArgs>()), Occurs.Exactly(2));
//The following two pass:
Mock.Assert(()=>mock.Process(Arg.Matches<MyArgs>
(d=>d.Id==idOne)),Occurs.Exactly(2));
Mock.Assert(()=>mock.Process(Arg.Matches<MyArgs>
(d=>d.Id==idTwo)),Occurs.Exactly(2));
}
}
public interface IProcessor
{
void Process(MyArgs args);
}
public class MyArgs
{
public void UpdateId(int newId)
{
this.Id = newId;
}
public int Id {get; private set;}
}
public class Runner
{
private IProcessor processor;
public Runner(IProcessor processor)
{
this.processor=processor;
}
public void Process(DataTable dt)
{
MyArgs args = new MyArgs();
foreach(DataRow row in dt.Rows)
{
int id = Convert.ToInt32(row["Id"]);
args.UpdateId(id);
processor.Process(args);
}
}
}编辑:在失败的测试方法中,如果我完全删除一个int变量并显式断言另一个变量只调用了一次,测试就会通过。只有当我把第二个,不同的价值加到一起的时候,事情才会变得更糟。
发布于 2015-03-31 17:10:27
由于无法让我们选择的模拟框架(JustMock)实现我想要的结果,所以我最终应用了接吻,忘记了这个测试的模拟框架,并进行了如下操作:
我改变了我的测试方法,如下所示:
[TestMethod]
public void TwoCallsDifferentIds()
{
int idOne = 1;
int idTwo = 2;
DataTable dt = new DataTable();
dt.Columns.Add("Id");
dt.Rows.Add(idOne);
dt.Rows.Add(idTwo);
FakeProcessor processor = new FakeProcessor();
Runner runner = new Runner(processor);
runner.Process(dt);
Assert.AreEqual(2, processor.MyArgsIds.Count);
Assert.AreEqual(1, processor.MyArgsIds[0]);
Assert.AreEqual(2, processor.MyArgsIds[1]);
}并添加了以下假的:
public class FakeProcessor : IProcessor
{
private IList<int> mList = new List<int>();
public IList<int> MyArgsIds
{
get { return mList; }
}
public void Process(MyArgs args)
{
mList.Add(args.Id);
}
}也许没那么聪明,但它给了我我想要的考试。
发布于 2015-03-31 07:22:35
模拟记录使用相同的MyArgs实例两次调用它。但是,实例内容在这两个调用之间是不同的这一事实已经丢失了。当您达到断言调用的程度时,记录的MyArgs参数在失败测试中有一个Id等于2。
如果将Process方法更改为
public void Process(DataTable dt)
{
foreach(DataRow row in dt.Rows)
{
MyArgs args = new MyArgs();
int id = Convert.ToInt32(row["Id"]);
args.UpdateId(id);
processor.Process(args);
}
}两项测试都会通过的。
编辑:
通过在安排中说明您的期望,您可以使用JustMock获得所需的行为,如下所示:
[TestMethod]
public void MakeTwoCallsDifferentIdsFails()
{
int idOne = 1;
int idTwo = 2;
DataTable dt = new DataTable();
dt.Columns.Add("Id");
dt.Rows.Add(idOne);
dt.Rows.Add(idTwo);
IProcessor mock = Mock.Create<IProcessor>();
Mock.Arrange(() => mock.Process(Arg.Matches<MyArgs>(d => d.Id == idOne))).OccursOnce();
Mock.Arrange(() => mock.Process(Arg.Matches<MyArgs>(d => d.Id == idTwo))).OccursOnce();
Runner runner = new Runner(mock);
runner.Process(dt);
Mock.Assert(mock);
}这样做的原因是,一旦正确的调用发生,出现的期望就被标记为满足了,所以Arg.Matches的谓词处理传递给方法的参数的活值,而不是在保存的参数实例可能发生变异时发布事实。
https://stackoverflow.com/questions/29352091
复制相似问题