正在调用委托方法的XCTest测试

时间:2014-05-31 16:01:40

标签: ios objective-c macos xctest

我一直在尝试测试我创建的一些使用NSNetServer类执行网络操作的类。我有一些问题确保调用委托方法。

我尝试了很多方法,包括:

在其他操作发生时,使用[NSThread sleepForTimeInterval:5.0f];[[NSRunLoop currentRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:5.0f]];暂停。 NSRunLoop方法在第一次调用时工作(如下面的示例代码所示),但在第二次调用时崩溃。我知道"正确"做事的方式,但我不知道"纠正"方式是。

使用NSConditionNSConditionLock类,它们似乎锁定代码并且永远不会调用回调。

对回调方法中更改的变量使用while循环,与上面相同。

以下是带有一些额外注释的代码,为简单起见,删除了一些测试:

- (void)testCheckCredentials
{
    [self.server start];
    // Create a client
    self.nsb = [[NSNetServiceBrowser alloc] init];
    // Set the delegate to self
    self.nsb.delegate = self;
    // Search for the server
    [self.nsb searchForServicesOfType:self.protocol inDomain:@""];
    // Wait for the service to be found and resolved
    [[NSRunLoop currentRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:self.timeout]];
    XCTAssertTrue(self.serviceWasFound, @"Service was not found");
    // Open the connection to the server
    XCTAssertTrue([self.serverConnection open], @"Connection to server failed to open");
    // Wait for the client to connect
    /* This is where it crashes */
    [[NSRunLoop currentRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:self.timeout]];
    XCTAssertTrue(self.clientDidConnect, @"Client did not connect");
    /* Further, more class-specific tests */
}

- (void)netServiceBrowser:(NSNetServiceBrowser *)aNetServiceBrowser didFindService:(NSNetService *)service moreComing:(BOOL)moreComing
{
    NSLog(@"Found a service: %@ (%@)", service.name, service.domain);
    if ([self.serverName isEqualToString:service.name]) {
        self.serviceWasFound = YES;
    }
}

- (void)clientDidConnect:(RCFClientConnection *)client
{
    XCTAssertNotNil(client, @"Connected client is nil");
    self.clientConnection = client;
    self.clientDidConnect = YES;
}

我还尝试在lock对象上执行NSCondition

[self.nsb searchForServicesOfType:self.protocol inDomain:@""];
// Wait for the service to be found and resolved
[self.lock lockWhenCondition:1];
XCTAssertTrue(self.serviceWasFound, @"Service was not found");

self.serviceWasFound = YES;
[self.lock unlockWithCondition:1]

使用lock方法时,永远不会调用netServiceBrowser:didFindService:moreComing:方法,使用时也是如此:

while (!self.serviceWasFound) {};

我还在学习Objective-C,但我完全坚持这个问题。

2 个答案:

答案 0 :(得分:14)

为了处理调用异步执行方法和函数的测试组件,XCTest在Xcode 6中得到了增强,包括使用新API和类XCTestExpectation对象处理块的能力。这些对象响应新的XCTest方法,这些方法允许测试方法等待,直到异步调用返回或达到超时。

以下是上述摘录的Apple文档链接。 Writing Tests of Asynchronous Operations

@interface sampleAPITests : XCTestCase<APIRequestClassDelegate>{
APIRequestClass *apiRequester;
XCTestExpectation *serverRespondExpectation;
}
@end

//implementation test class
- (void)testAPIConnectivity {
// This is an example of a functional test case.
serverRespondExpectation = [self expectationWithDescription:@"server responded"];
[apiRequester sendAPIRequestForMethod:nil withParams:nil];//send request to server to get tap info
apiRequester.delegate = self;
[self waitForExpectationsWithTimeout:1 handler:^(NSError *error) {
    if (error) {
        NSLog(@"Server Timeout Error: %@", error);
    }
   nslog(@"execute here after delegate called  or timeout");
}];
XCTAssert(YES, @"Pass");
}

//Delegate implementation
- (void) request:(APIRequest *)request didReceiveResponse:(NSDictionary *)jsonResponse success:(BOOL)success{
[serverRespondExpectation fulfill];
XCTAssertNotNil(jsonResponse,@"json object returned from server is nil");
}

答案 1 :(得分:1)

许多测试库支持测试异步和线程操作。

它们的工作方式基本相同:

  • 等待x秒以发生条件。
  • 可选择在第一个条件之后执行一组断言,如果不满足其中一个,则失败。
  • 如果在指定(或默认)时间内未发生所需条件,则失败。

以下是一些提供这些功能的库:

  • expecta匹配库。
  • Kiwi测试框架。
  • Typhoon DI框架有一个用于执行异步集成测试的实用程序。

据我所知,所有这些库都使用运行循环方法,因为其他库可能会导致死锁。为了使这种方法可靠地工作:

  • 在测试基于块的回调时,您可以创建内联块。
  • 对于委托回调,您应该创建一个单独的存根(最简单的协议实现)或使用mock(使用模拟库创建的“魔术”实现),并将其设置为委托。如果将测试实例本身设置为委托,则可能会出现问题。

修改

从Xcode6开始,有一个新的断言XCTestExpectation用于异步测试。

可以使用CocoaPods安装上述每个库。