如何在 Swift 中使用 OCMock 模拟来自 AFNetworking 的响应?

How to mock response from AFNetworking with OCMock in Swift?

这是我获取网络客户端实例的方式:

let networkClient = DBNetworkClient(baseURL: NSURL(string: "http://mysite.pl/api"))

我也有一个方法:

func citiesWithParameters(parameters: [String: String], completionBlock: DBSearchOptionHandler) {

    GET("cities", parameters: parameters, success: { operation, response in

        if let error = NSError(response: response) {
            completionBlock([], error)
        } else {
            let cities = DBCity.parseCitiesWithDictionary(response as! NSDictionary)
            completionBlock(cities, nil)
        }

        }) { operation, error in

            completionBlock([], error)
    }
}

我是这样调用这个方法的:

networkClient.citiesWithParameters(parameters, completionBlock: { cities, error in 
    //do sth 
})

通过这种方式,我传递了一些参数,并从服务器获得了 REAL 响应。当我要求这个时,我想嘲笑那个回应。如何做到这一点?

func testCities() {

    let mockNetworkClient = OCMockObject.mockForClass(DBNetworkClient.classForCoder())        


    //what should I perform here to be able to do sth like this:

    let expectation = expectationWithDescription("")

    mockNetworkClient.citiesWithParameters(["a": "b"]) { cities, error in
        expectation.fulfill()
        XCTAssertNotNil(cities)
        XCTAssertNil(error) //since I know what is the response, because I mocked this
    }

    waitForExpectationsWithTimeout(10, handler: nil)
}

这就是我的方法 GETDBNetworkClient:

中的定义方式
override func GET(URLString: String!, parameters: AnyObject!, success: ((AFHTTPRequestOperation!, AnyObject!) -> Void)!, failure: ((AFHTTPRequestOperation!, NSError!) -> Void)!) -> AFHTTPRequestOperation! {

    return super.GET(URLString, parameters: parameters, success: { (operation, response) in

        print("GET \(operation.request.URL)")
        print("GET \(response)")

        success(operation, response)

        }, failure: { (operation, error) in

            print("GET \(operation.request.URL)")
            print("GET \(operation.responseObject)")

            failure(operation, error)
    })
}

一旦我能够,我将奖励 50 赏金 给帮助我做到这一点的人。

不幸的是,

Writing mock tests for AFNetworking 没有帮助。它对我不起作用。

我没有使用 Alamofire 的经验,因此我不知道在哪里声明了你的 GET 方法,但你绝对应该存根这个方法而不是 citiesWithParameters.

例如,如果它在您的 DBNetworkClient 中声明:

func testCities()
{
    //1
    class DBNetworkClientMocked: DBNetworkClient
    {
        //2
        override func GET(URLString: String!, parameters: AnyObject!, success: ((AFHTTPRequestOperation!, AnyObject!) -> Void)!, failure: ((AFHTTPRequestOperation!, NSError!) -> Void)!) -> AFHTTPRequestOperation! {

            //3
            success(nil, ["San Francisco", "London", "Sofia"])

        }
    }

    //4
    let sut = DBNetworkClientMocked()
    sut.citiesWithParameters(["a":"b"]) { cities, error in

        //5
        XCTAssertNotNil(cities)
        XCTAssertNil(error)

    }
}

那么这里发生了什么:

  1. 您将 class 定义为 DBNetworkClient 的子级,因此按照您发布的文章中的描述为其创建 'Mock'。在 class 中,我们将仅覆盖我们想要更改的方法(存根),而其他方法保持不变。这是以前用 OCMock 完成的,被称​​为部分模拟。
  2. 现在你存根它的 GET 方法是为了 return 特定数据,而不是来自服务器的实际数据
  3. 在这里您可以定义 return 您存根服务器的内容。可以是成功失败等
  4. 现在,在我们准备好模拟和存根之后,我们创建所谓的 Ssubject Under Test(sut)。请注意,如果 DBNetworkClient 有特定的构造函数,您必须调用此构造函数而不是默认构造函数 - ().
  5. 我们执行我们想要测试的方法。在它的回调中,我们放入了所有断言。

到目前为止一切顺利。但是,如果 GET 方法是 Alamofire 的一部分,您需要使用一种称为依赖注入的技术(您可以 google 它以获取更多信息)。

因此,如果 GET 在另一个 class 中声明并且仅在 citiesWithParameters 中引用,我们如何存根呢?让我们看看这个:

//1
class DBNetworkClient
{
    //2
    func citiesWithParameters(parameters: [String: String], networkWorker: Alamofire = Alamofire(), completionBlock: DBSearchOptionHandler) {

        //3
        networkWorker.GET("cities", parameters: parameters, success: { operation, response in

            if let error = NSError(response: response) {
                completionBlock([], error)
            } else {
                let cities = DBCity.parseCitiesWithDictionary(response as! NSDictionary)
                completionBlock(cities, nil)
            }

            }) { operation, error in

                completionBlock([], error)
        }

    }
}

func testCities()
{
    //4
    class AlamofireMock: Alamofire
    {
        override func GET(URLString: String!, parameters: AnyObject!, success: ((AFHTTPRequestOperation!, AnyObject!) -> Void)!, failure: ((AFHTTPRequestOperation!, NSError!) -> Void)!) -> AFHTTPRequestOperation! {

            success(nil, ["San Francisco", "London", "Sofia"])

        }
    }

    //5
    let sut = DBNetworkClient()
    sut.citiesWithParameters(["a":"b"], networkWorker: AlamofireMock()) { cities, error in

        //6
        XCTAssertNotNil(cities)
        XCTAssertNil(error)

    }
}
  1. 首先,我们必须稍微更改 citiesWithParameters 以接收更多参数。这个参数就是我们的依赖注入。它是具有 GET 方法的对象。在现实生活中的例子中,这将是唯一的协议会更好,因为 citiesWithParameters 除了这个对象能够发出请求之外不需要知道任何东西。
  2. 我已将 networkWorker 参数设置为默认值,否则您需要将所有调用更改为 citiesWithParameters 以满足新参数要求。
  3. 我们保持所有实现不变,只是现在我们调用我们注入的对象GET方法
  4. 现在回到测试中,这次我们将模拟 Alamofire。我们做了与前面示例完全相同的事情,只是这次模拟的 class 是 Alamofire 而不是 DBNetworkClient
  5. 当我们调用 citiesWithParameters 时,我们将模拟对象作为 networkWorker 传递。这样,我们存根的 GET 方法将被调用,我们将从 'fake' 服务器获得我们预期的数据。
  6. 我们的断言保持不变

请注意,这两个示例未使用 OCMock,而是完全依赖 Swift 功能! OCMock 是我们在伟大的动态语言中使用的一个很棒的工具 - Objective-C。然而在 Swift 上几乎完全没有活力和反思。这就是为什么即使在官方 OCMock 页面上我们也有以下声明:

Will there be a mock framework for Swift written in Swift? Maybe. As of now it doesn't look too likely, though, because mock frameworks depend heavily on access to the language runtime, and Swift does not seem to provide any.

提供的两个实现中缺少的是验证调用了 GET 方法。我会这样保留它,因为这不是最初的问题,您可以使用在 mocked class.

中声明的布尔值轻松实现它

更重要的是,我假设 Alamofire 中的 GET 方法是实例方法,而不是 class 方法。如果这不是真的,您可以在 DBNetworkClient 中声明新方法,它只调用 Alamofire.GET 并在 citiesWithParameters 中使用该方法。然后你可以像第一个例子那样存根这个方法,一切都会好起来的。