如何单元/集成依赖NSNotificationCenter的测试类(在MonoTouch中)?

时间:2013-04-12 17:04:14

标签: c# objective-c testing xamarin.ios nunit

我正在使用MonoTouch编写一个iOS应用程序,它有很多信息可以在很多不同的屏幕上输入。可以在屏幕1上输入信息A,B和C,但是我需要在屏幕3上显示信息B以及信息X,Y,Z等的输入。用户可以从屏幕1跳到屏幕7,然后返回到屏幕3.

所以,我创建了一个静态类来接收所有信息(通过NSNotifications)并缓存它。它还从各种屏幕监听信息请求(同样,通过NSNotifications),从缓存中打包相关信息(在私有静态方法中),并发出另一个通知作为响应。这个类应该始终具有最新信息,因此无论用户跳转到哪个屏幕都应如此。

public static class InformationFlowController
{
    static List<NSObject> _observers;
    static Dictionary<SomeEnum, Object> _data;

    static InformationFlowController()
    {
        _observers = new List<NSObject>();
        _data = new Dictionary<SomeEnum, Object>();

        _observers.Add(NSNotificationCenter.DefaultCenter.AddObserver(Notifications.PayloadA, HandlePayloadA));
        ...
        _observers.Add(NSNotificationCenter.DefaultCenter.AddObserver(Notifications.RequestA, HandleRequestA));
    }

    // receive information A
    static void HandlePayloadA(NSNotification ntf)
    {
        var infoA = (InfoA)ntf.Object;
        if (A == null) { /* throw an exception */ }

        if (_data.ContainsKey(EnumA))
        {
            _data.Remove(EnumA);
        }

        _data.Add(EnumA, infoA);
    }

    // receive request for info A
    static void HandleRequestA(NSNotification ntf)
    {
        InfoA info = null;

        if (_data.ContainsKey(EnumA))
        {
            info = _data.GetValue(EnumA);
        }

        // its up to the receiver what it wants to do if it gets null
        NSNotificationCenter.DefaultCenter.PostNotificationName(Notifications.ResponseA, info);
    }
}

一切正常。

我的问题是如何对其进行单元/集成测试? 由于app / policy / yadda-yadda的性质,我必须有大约100%的代码覆盖率。这是我尝试过的以及我遇到过的事情。

我试图为此编写集成测试的问题是确保如果我请求信息A,我会得到包含信息A的响应。

[Test]
public void HandleRequestForATest()
{
    // setup the data in the IFC here

    NSNotificationCenter.DefaultCenter.AddObserver(Notifications.ResponseA, delegate(NSNotification ntf)
    {
        var info = ntf.Object as InfoA;
        Assert.IsNotNull(info, "Info is null!");
        // some other asserts to check the data within info
    }

    NSNotificationCenter.DefaultCenter.PostNotificationName(Notifications.RequestA, null);        
}

测试通过0断言。我在var info = ntf.Object as InfoA;上放了一个断点,它永远不会被触及。我想也许它可以在它收到响应之前得到GC,但是我在IFC.HandleRequestA方法中放了一个断点,它永远不会被调用。通过检查,我可以看到通知已经在DefaultCenter中注册,但它们似乎没有被解雇。

除此之外,我如何测试以确保在一定时间内收到通知,如果不是,则测试失败?我尝试添加一个类变量:

bool assertsCompleted = false;

并修改测试:

[Test]
public void HandleRequestForATest()
{
    // setup the data in the IFC here

    NSNotificationCenter.DefaultCenter.AddObserver(Notifications.ResponseA, delegate(NSNotification ntf)
    {
        var info = ntf.Object as InfoA;
        Assert.IsNotNull(info, "Info is null!");
        // some other asserts to check the data within info
        assertsCompleted = true;
    }

    NSNotificationCenter.DefaultCenter.PostNotificationName(Notifications.RequestA, null);        

    NSTimer.CreateTimer(2, delegate
    {
        if (!assertsCompleted)
        {
            Assert.IsTrue(false, "Notification not received or Asserts not complete!");
        }
    } 
}

我在if (!assertsCompleted)上放了一个断点,但也没有被击中。

请帮忙! :)

EDIT2 我将IFC类从静态更改为非静态(以及方法),并修复了未调用处理程序的问题。这意味着我必须在AppDelegate和TestFixtureSetUp中实例化它,但我认为我们可以忍受这个。 的 / EDIT2

1 个答案:

答案 0 :(得分:2)

你可以尝试这样的事情:

[Test]
public void HandleRequestForATest()
{
    // setup the data in the IFC here

    var resetEvent = new ManualResetEvent(false);
    InfoA info = null;

    var observer = NSNotificationCenter.DefaultCenter.AddObserver(Notifications.ResponseA, delegate(NSNotification ntf)
    {
        info = ntf.Object as InfoA;
        resetEvent.Set();
    }

    NSNotificationCenter.DefaultCenter.PostNotificationName(Notifications.RequestA, null);        

    Assert.IsTrue(resetEvent.WaitOne(250), "Reset event timed out!");
    Assert.IsNotNull(info, "Info is null!");
    // some other asserts to check the data within info

    //Make sure you do this! I'd put in try-finally block
    NSNotificationCenter.DefaultCenter.RemoveObserver(observer);
}