Skip to content

Latest commit

 

History

History
584 lines (422 loc) · 26.4 KB

Testing-Network-Code.md

File metadata and controls

584 lines (422 loc) · 26.4 KB

5 Testing Network code

앱스토어에 있는 대부분의 앱은 한 가지 이상의 방법으로 네트워킹을 수행한다. 애플은 네트워크 요청을 위해 URLSession 클래스를 제공한다. 이때의 요청은 비동기다. 백그라운드 스레드에서 응답을 받는 것을 의미한다. 이런 방법이 없다면, UI는 멈추고, 서버에서 응답이 도착할 때까지 기다려야 한다.

이 장의 주요 주제는 비동기 API를 테스트하는 법이다. 비동기 API 요청을 테스트하는 두 가지 방법이 있다. 첫째는, 앱이 앱스토어 있을 때 사용하는 되는 방법으로 실제 서버를 이용하는 것이다. 둘째로, 이전 챕터에서 사용했던 stubs를 사용할 수 있다.

두 가지 모두 각자의 장점이 있다. 실제 서버로 테스트하는 것은 서버가 문서대로 구현되었는지를 추가로 테스트할 수 있다. 이런 테스트는 완성된 앱의 구현에 가깝고, 최종 버전에서 끝날 버그를 찾기도 쉽다.

한편, stubs을 통해 심지어 웹서비스를 만들기도 전에도 앱의 네트워크 레이어를 개발할 수 있다. API와 기대하는 응답에 대한 문서만 있으면 된다. 서버와의 상호작용에 의존하지 않기 때문에 테스트 속도는 아주 빠르다.

이 장에서 다음 주제를 다룰 것이다.

  • 실제 웹서비스를 이용한 테스트 구현
  • 가짜 웹서비스에 대한 로그인 요청 만들기
  • 에러 다루기

Implementing tests using a web service

이전 장에서 CLGeocoder에 대한 stubs를 작성했다. 이제 geocoder가 내장된 CoreLocation이 기대한 대로 잘 작동하는지 검증하는 테스트를 작성할 것이다. geocoder에서 좌표를 가져오는 것은 비동기이다. 비동기 인터페이스를 다룰 수 있는 테스트를 작성해야 한다는 것을 의미한다.

Controller에 해당 하는 파일을 그루핑하자. 테스트 타겟에 대해서도 똑같이 하자.

이제 테스트를 시작하자. 다음의 테스트를 InputViewControllerTests에 추가하자.

func test_Geocoder_FetchesCoordinates() {
  CLGeocoder().geocodeAddressString("Infinite Loop 1, Cupertino") {
    (placemarks, error) in
    
    let coordinate = placemarks?.first?.location?.coordinate
    guard let latitude = coordinate?.latitude else {
      XCTFail(); return
    }
    
    guard let longitude = coordinate?.longitude else {
      XCTFail(); return
    }
    
    XCTAssertEqualWithAccuracy(latitude, 37.3316, accuracy: 0.0001)
    XCTAssertEqualWithAccuracy(longitude, -122.0300, accuracy: 0.0001)
  }
}

테스트를 돌리자. 모두 성공한다. geocoder가 우리 생각처럼 동작하는 듯이 보인다. 잠깐만 기다리자. 우리는 적색 상태를 넘겼다. TDD에서는 실패하는 테스트를 먼저 가져야 한다. 그렇지 않으면 테스트가 실제 동작하는지 확신할 수 없다.

우리는 CLGeocoder의 소스에 접근할 수 없고, 테스트 실패를 위해 구현을 변경할 수도 없다. assertion을 바꾸는 것만 할 수 있다. 클로저 내의 assertion을 다음 코드로 변경하자.

테스트를 수행하자. 테스트는 여전히 성공이다. 브레이크 포인트를 걸어 무슨 일이 벌어지고 있는지 확인하자.

테스트를 다시 수행하자. 테스트 수행 중에 디버거는 해당 라인에서 멈추고, 디버거 콘솔을 열어 무슨 일이 진행 중인지 알 수 있다.

디버거는 브레이크 포인트에 절대 닿지 않는다.

geocodeAddressString(_:completionHandler:)이 비동기이기 때문이다. 해당 클로저는 언젠가 미래에 다른 쓰레드에서 호출되고, 테스트 실행은 진행된다. 콜백 블럭이 실행되기도 전에 테스트는 끝나고, assertion은 끝내 호출되지 않는다. 테스트가 비동기로 동작하도록 바꿔야 한다.

다음 코드로 바꿔보자.

func test_Geocoder_FetchesCoordinates() {
  let geocoderAnswered = expectation(description: "Geocoder")
  
  CLGeocoder().geocodeAddressString("Infinite Loop 1, Cupertino") {
    (placemarks, error) in
    
    let coordinate = placemarks?.first?.location?.coordinate
    guard let latitude = coordinate?.latitude else {
      XCTFail(); return
    }
    
    guard let longitude = coordinate?.longitude else {
      XCTFail(); return
    }

    XCTAssertEqualWithAccuracy(latitude, 0.0, accuracy: 0.0001)
    XCTAssertEqualWithAccuracy(longitude, 0.0, accuracy: 0.0001)
    
    geocoderAnswered.fulfill()
  }
  waitForExpectations(timeout: 3, handler: nil)
}

expectation(description:)를 이용해 expectation을 만들었다. 테스트 마지막에 3초 타임아웃과 함께 waitForExpectations(timeout:handler:)를 호출한다. 테스트 러너가 이 지점에서 멈추고 expectation이 만족되거나 타임아웃이 끝날때까지 기다리는 것을 의미한다. 만약 모든 타임아웃이 지나도록 모든 expectation이 만족되지 않는다면 테스트를 실패한다. 콜백 클로저에서 assertion이 호출된 이후에 expectation을 만족한다.

이제 테스트를 다시 해보자. assertion에 맞지 않는 값을 넣었기 떄문에 마지막 테스트는 실패한다. 다음으로 바꾸자.

XCTAssertEqualWithAccuracy(latitude, 37.3316, accuracy: 0.0001)
XCTAssertEqualWithAccuracy(longitude, -122.0300, accuracy: 0.0001)

테스트를 다시 돌리자. 모든 테스트가 통과하고, GLGeocoder는 예상대로 동작한다.

우리는 방금 XCTest로 비동기 API를 테스트 하는 법을 봤다. 이는 다양한 측면의 iOS 개발에 사용될 수 있다(예를 들어 NSNotification 보내기, 웹서버에서 데이터 가져오기, 백그라운드에서 데이터베이스에 데이터 쓰기 등). 비동기 작업일 발생할 때마다, 우리는 expectation을 추가하고, 비동기 콜백이 실행되었을 때 이것이 만족되도록 만들 수 있다.

이는 매우 강력하다. 하지만 단위 테스트는 빠르고, 신뢰할 수 있어야 한다는 것을 잊지 말자. 웹서버를 테스트에 이용하는 것은 테스트를 약하고, 느리게 만든다. 만약 웹서버가 3초 이상을 필요로 한다면, 테스트를 실패한다. 그리고 테스트를 수행할 때 항상 인터넷이 연결되어 있어야 한다.

다음 섹션에서 우리는 stubs을 이용해 테스트를 강하고 빠르게 만들 것이다. 완성된 웹서버 없이도 앱의 네트워크 계층을 만들 수 있다는 것은 추가적이 이점이다. 완성된 API 문서만 있으면 된다.

Implementing a login request

동료가 웹서비스를 만들고 있지만 아직 끝내지 않았다고 가정하자. 하지만 우리는 이미 API가 어떻게 생겼는지 알고 있다. 로그인을 위한 endpoint가 있을 것이다. https://asesometodos.com/login 일 것이다. 사용자이름과 비밀번호 두 개를 인자로 받고, 다음 API요청에 사용될 token을 반환한다.

우리는 로그인 요청에서 받은 토큰이 토큰 struct에 들어가는지 테스트를 통해 검증해야 한다.

APIClientTests를 만들자.

우리는 로그인 기능을 몇 개의 작은 기능으로 나눌 것이다. 이전에 언급했듯 로그인은 사용자 이름과 비밀번호를 쿼리 파라메터로 넘기면서, https://asesometodos.com/login에 대한 HTTPS 요청을 만들어야 한다.

다음 코드를 APIClientTests에 추가하자.

func test_Login_UsesExpectedURL() {
  let sut = APIClient()
}

정적 분석기가 APIClient가 필요하다 말할 것이다. APIClient.swift를 추가하자.

class APIClient {
  
}

이거면 정적분석기를 행복하게 만드는 데에 충분할 것이다.

서버 쪽이 아직 끝나지 않았기 때문에 네트워크 요청을 가장하는 가짜 URL session을 주입할 수 있어야 한다. 다음 코드를 추가하자.

let mockURLSession = MockURLSession()

클래스가 없기 때문에 컴파일은 불가능하다. 다음 코드를 APIClientTests.swift의 클래스 정의 밖에 추가하자.

extension APIClientTests {
  class MockURLSession {
    var url: URL?
    func dataTask(
      with url: URL,
      completionHandler: @escaping (Data?, URLResponse?, Error?) -> Void)
      -> URLSessionTask {
        
        self.url = url
        return URLSession.shared.dataTask(with: url)
    }
  }
}

이 모의 클래스는 네트워크 요청 구현에 사용할 dataTask(with:completionHandler:) 메서드를 구현하고 있다. 이 모의 클래스는 URL을 캐치한다. 이를 통해 테스트에서 URL을 확인할 수 있다. 다음에는 모의 클래스를 실제 구현에 주입하고 싶을 것이다. 다음 코드를 test_Login_UsesExpectedURL()의 마지막에 추가하자.

sut.session = mockURLSession

APIClient를 열어서 session 프로퍼티를 추가해서 컴파일 되도록 만들자.

lazy var session: URLSession = URLSession.shared

다시 테스트를 수행하자. 여전히 컴파일이 안 된다. APIClientTests.MockURLSessionURLSession에 할당할 수 없기 때문이다. URLSession나 모의 클래스의 인스턴스를 할당하려면 session 의 타입을 바꿔야 한다. 프로토콜을 이용하면 된다. 다음 코드를 APIClient.swift에 추가하자.

protocol SessionProtocol {
  func dataTask(
    with url: URL,
    completionHandler: @escaping (Data?, URLResponse?, Error?) -> Void)
    -> URLSessionTask
}

URLSession은 이미 이 프로토콜을 구현하고 있다. 다음 코드를 APIClient.swift에 추가해서 프로토콜을 준수하도록 만들자.

extension URLSession: SessionProtocol {}

다음, 모의 클래스가 해당 프로토콜을 준수한다는 것을 컴파일러에게 알려야 한다. 다음처럼 바꾸자.

class MockURLSession: SessionProtocol {
  // ...
}

마지막으로, session 프로퍼티의 타입을 바꾸자. APIClient에서 URLSession 타입을 SessionProtocol으로 바꾸자.

lazy var session: SessionProtocol = URLSession.shared

테스트를 수행하자. 이제 테스트는 컴파일되고, 다음으로 가자. APIClient는 로그인 메서드가 필요하다. 다음을 test_Login_UsesExpectedURL()에 추가하자.

let completion = { (token: Token?, error: Error?) in }
    sut.loginUser(withName: "dasdom",
                  password: "1234",
                  completion: completion)

loginUser(withName:password:completion:)Token이 없기 때문에 컴파일은 불가능하다. APIClient를 열어서 다음을 추가하자.

func loginUser(withName username: String,
                 password: String,
                 completion: @escaping (Token?, Error?) -> Void) {   
}

Token.swift 파일을 만들고 다음을 추가하자.

struct Token { }

다시 컴파일하면 성공할 것이다.

로그인 메서드가 예상하는 host를 사용하는지 확인하는 테스트를 만들자. test_Login_UsesExpectedURL() 마지막에 다음을 추가하자.

guard let url = mockURLSession.url else { XCTFail(); return }
let urlComponents = URLComponents(url: url,
                                  resolvingAgainstBaseURL: true)
XCTAssertEqual(urlComponents?.host, "awesometodos.com")

이 코드는 mockURLSession으로부터 URL component를 가져오고(우리의 모의 session은 URL을 캐치한다), URL의 host가 awesometodos.com 인지 검증한다.

테스트를 수행하면 테스트는 실패한다. 다음을 추가하자.

loginUser(withName:password:completion:):

  guard let url = URL(string: "https://awesometodos.com") else { 
    fatalError() 
  }
  session.dataTask(with: url) { (data, response, error) in 
  }

테스트를 다시 수행하자. 모든 테스틑 성공하고, 리팩토링할 것도 없다. 다음, URL의 path에 대한 테스트를 추가하자. 다음 코드를 test_Login_UsesExpectedURL() 마지막에 추가하자.

XCTAssertEqual(urlComponents?.path, "/login")

테스트를 다시 통과하기 위해 다음으로 변경하자.

guard let url = URL(string: "https://awesometodos.com/login") else { 
  fatalError() 
}

모든 테스트는 성공한다. 다음, URL query 인자를 검증하자. test_Login_UsesExpectedURL() 마지막에 다음을 추가하자.

XCTAssertEqual(urlComponents?.query, 
  "username=dasdom&password=1234")

테스트는 실패한다. 다음으로 변경하자.

let query = "username=\(username)&password=\(password)" 
guard let url = URL(string: "https://awesometodos.com/login?\(query)") else {
  fatalError()
}

테스트는 다시 성공한다. 하지만 이전에 웹서비스 작업을 해봤다면, 코드에 문제가 있다는 것을 깨달았을 것이다. URL에서 몇몇 문자는 특별한 의미를 갖는다. 예를 들어, &은 URL 쿼리를 나눈다. 하지만 사용자는 이 문자를 비밀번호에 사용할 수 있다. 쿼리 아이템을 엔코딩 해야 한다. 실제 코드를 변경하도록 테스트를 리팩토링하자. 먼저, 사용자이름과 비밀번호에서 특수문자를 사용할 수 있도록 loginUser(withName:password:completion:) 요청을 바꾸자.

sut.loginUser(withName:"dasdöm", 
              password: "%&34", 
              completion: completion)

다음, 쿼리에 대한 검증을 다음 코드로 교체하자.

let allowedCharacters = CharacterSet(
  charactersIn: "/%&=?$#+-~@<>|\\*,.()[]{}^!").inverted

guard let expectedUsername = "dasdöm".addingPercentEncoding(
  withAllowedCharacters: allowedCharacters) else { fatalError() }

guard let expectedPassword = "%&34".addingPercentEncoding(
  withAllowedCharacters: allowedCharacters) else { fatalError() }

XCTAssertEqual(urlComponents?.percentEncodedQuery,
              "username=\(expectedUsername)&password=\(expectedPassword)")

이 수정으로 엔코딩된 사용자 이름과 비밀번호가 URL 쿼리에 사용되었는지 검증한다. 테스트를 수행하자. 이 경우 URL이 문자열이 될 수 없어서 에러가 발생하고, 테스트는 멈춘다. loginUser(withName:password:completion:)를 변경해서 해결하자.

let allowedCharacters = CharacterSet(
    charactersIn: "/%&=?$#+-~@<>|\\*,.()[]{}^!").inverted
    
guard let encodedUsername = username.addingPercentEncoding(withAllowedCharacters: allowedCharacters) else { fatalError() }

guard let encodedPassword = password.addingPercentEncoding(withAllowedCharacters: allowedCharacters) else { fatalError() }

let query = "username=\(encodedUsername)&password=\(encodedPassword)"
guard let url = URL(string: "https://awesometodos.com/login?\(query)") else {
  fatalError()
}

session.dataTask(with: url) { (data, response, error) in
  
}

이 코드로 URL을 만들기 전에 사용자 이름과 비밀번호를 엔코딩할 수 있다. 테스트를 돌리면 성공한다.

지금의 테스트는 쿼리 아이템의 순서에 의존적이다. URL에서 순서는 상관없기 때문에 이는 좋은 생각이 아니다. 이는 URL에 정확해도 실패할 수 있다는 것을 의미한다. 그러니 다음 테스트로 가기 전에 리팩토링하자. 힌트: URLComponentsqueryItems라는 프로퍼티를 가지고 있다. 이걸 써보자.

지금 현재, MockURLSession은 요청의 URL을 캐치한다. 로그인 코드를 테스트하려면, 테스트에서 data task의 completion handler를 요청할 수 있어야 한다. 이 방법으로 로그인 코드가 예상한 방식의 반환 데이터로 진행하는지 보장할 수 있다. dataTaskresume이 호출될 때 completion handler를 캐치하면서 이 작업을 수행할 수 있다.

먼저 data task에 대한 mock을 만들어야 한다. 다음 모의 클래스를 APIClientTests에 추가하자.

class MockTask: URLSessionDataTask {
  private let data: Data?
  private let urlResponse: URLResponse?
  private let responseError: Error?
  
  typealias CompletionHandler = (Data?, URLResponse?, Error?) -> Void
  var completionHandler: CompletionHandler?
  
  init(data: Data?, urlResponse: URLResponse?, error: Error?) {
    self.data = data
    self.urlResponse = urlResponse
    self.responseError = error
  }
  
  override func resume() {
    DispatchQueue.main.async {
      self.completionHandler?(self.data,
                              self.urlResponse,
                              self.responseError)
    }
  }
}

이 코드는 네 개의 프로퍼티를 정의한다. 처음 세 개의 프로퍼티는 completion handler에서 사용될 값을 설정한다. 네 번째 프로퍼티는 resume()이 호출될 때 실행되는 completion handler이다.

추가로 이 mock은 두 개의 메서드를 갖는다. init은 completion handler에 사용되는 값을 받고, resume을 재정의한다. resume 메서드에서 completion handler는 메인큐로 보내진다. completion handler가 주변 코드에 대해 비동기인 것을 확인하기 위해 수행된다.

MockURLSession은 모의 data task를 만들어야 하고, dataTask(with:completionHandler:)가 호출될 때 모의 task를 리턴해야 한다. MockURLSession 클래스를 다음처럼 수정하자.

class MockURLSession: SessionProtocol {
  var url: URL?
  private let dataTask: MockTask
  
  init(data: Data?, urlResponse: URLResponse?, error: Error?) {
    dataTask = MockTask(data: data,
                        urlResponse: urlResponse,
                        error: error)
  }
  
  func dataTask(
    with url: URL,
    completionHandler: @escaping (Data?, URLResponse?, Error?) -> Void)
    -> URLSessionDataTask {
      
      self.url = url
      dataTask.completionHandler = completionHandler
      return dataTask
  }
}

data task를 만들고 저장하는 init 메서드를 추가했다. completion handler가 호출될 때 사용되는 인자들을 받는다. dataTask(with:completionHandler:)에서 completion handler를 모의 data task에 저장하고, 이를 반환한다.

이렇게 준비되면, completion handler의 요청은 모의 data task의 resume이 호출될 때 실행된다. completion handler의 인자들은 MockURLSession이 인스턴화될 때 할당된다.

init메서드를 MockURLSession에 추가 했기 때문에, 이전 테스트의 모의 URL Session은 더 이상 컴파일되지 않는다. let mockURLSession = MockURLSession(data: nil, urlResponse: nil, error: nil)으로 바꾸자.

이제 테스트할 준비가 됐으니 다음을 추가하자.

func test_Login_WhenSuccessful_CreatesToken() {
  let sut = APIClient()
  let jsonData = "{\"token\" : \"1234567890\"}".data(using: .utf8)
  let mockURLSession = MockURLSession(data: jsonData,
                                      urlResponse: nil,
                                      error: nil)
  sut.session = mockURLSession
  
  let tokenExpectation = expectation(description: "Token")
  var catchedToken: Token? = nil
  sut.loginUser(withName: "Foo", password: "Bar") { (token, error) in
    catchedToken = token
    tokenExpectation.fulfill()
  }
  
  waitForExpectations(timeout: 1) { (error) in
    XCTAssertEqual(catchedToken.id, "1234567890")
  }
}

먼저, 모의 URL Session이 있는 sut가 간단한 JSON 을 반환하도록 준비하자. 그 다음, expectation을 생성하고, login 메서드를 호출하자. 어쨌든 completion handler에서 간단한 JSON을 반환히기 때문에 사용자 이름이나 비밀번호는 상관없다. 테스트 마지막에, expectation이 만족될때까지 기다렸다가 토큰이 예상한 id를 가지고 있는지 검증한다.

Tokenid를 가지고 있지 않기 때문에 컴파일 안 된다. 프로퍼티를 추가하자.

let it: String

completion handler가 아무것도 하지 않기 때문에 테스트는 실패한다. session.dataTask(with:completionHandler:)를 다음으로 교체하자.

session.dataTask(with: url) { (data, response, error) in
  guard let data = data else { return  }
  let dict = try! JSONSerialization.jsonObject(
    with: data,
  options: []) as? [String: String]
  
  let token: Token?
  if let tokenString = dict?["token"] {
    token = Token(id: tokenString)
  } else {
    token = nil
  }
  completion(token, nil)
}.resume()

이제 생성된 data task에 resume() 호출한다. 그렇지 않으면, completion handler는 호출되지 않고, 테스트를 통과할 수 없다.

이 코드는 응답 데이터로부터 dictionary를 얻고 'token' key의 문자열로부터 Token 인스턴스를 생성한다. 생성된 토큰은 login 메서드의 completion handler로 전달 된다.

모든 테스트를 통과한다. 코드가 나빠보이지만 리팩토링 할 것은 없다. 스위프트 코드에서 느낌표를 볼때마다, 이것이 정말 필요한 것인지 개발자가 개으른 것인지 알아야 한다. 이전 코드에서 우리는 try!로 에러 처리를 넘어갔다. 실제 코드를 가이드하는 대신 테스트를 통해 이 코드를 리팩토링하자.

Handling errors

jsonObject(with:options:)를 사용하는 대신 try!를 사용하는 것은 컴파일러에게 '이것은 절대 실패하지 않을 거야' 라고 말하는 것이다. 잘못된 데이터를 이용하는 테스트를 작성하고, 에러가 발생하는지 검증해보자.

func test_Login_WhenJSONIsInvalid_ReturnsError() {
  let sut = APIClient()
  let mockURLSession = MockURLSession(data: Data(), 
                                      urlResponse: nil,
                                      error: nil)
  sut.session = mockURLSession
  let errorExpectation = expectation(description: "Error")
  var catchedError: Error? = nil
  sut.loginUser(withName: "Foo", password: "Bar") { (token, error) in
    catchedError = error
    errorExpectation.fulfill()
  }
  
  waitForExpectations(timeout: 1) { (error) in
    XCTAssertNotNil(catchedError)
  }
}

이 테스트에서 빈 데이터 객체를 completion handler에 넘긴다.

deserialization를 실패하기 때문에 에러가 발생하고 코드는 멈춘다. 이 에러를 다룰 수 있도록 코드를 바꿔보자.

guard let data = data else { return }
do {
  let dict = try JSONSerialization.jsonObject(
    with: data,
    options: []) as? [String:String]
  
  let token: Token?
  if let tokenString = dict?["token"] {
    token = Token(id: tokenString)
  } else {
    token = nil
  }
  completion(token, nil)
} catch {
  completion(nil, error)
}

이 코드로 에러를 캐치할 수 있고, 그것을 로그인 메서드의 completion 블럭으로 넘길 수 있다. 테스트는 모두 성공한다.

다음, 실제 코드에서 데이터가 nil일 때, completion handler가 에러와 함께 호출되는지를 확인해야 한다.

func test_Login_WhenDataIsNil_ReturnsError() {
  
  let sut = APIClient()
  let mockURLSession = MockURLSession(data: nil,
                                      urlResponse: nil,
                                      error: nil)
  sut.session = mockURLSession
  
  let errorExpectation = expectation(description: "Error")
  var catchedError: Error? = nil
  sut.loginUser(withName: "Foo", password: "Bar") { (token, error) in
    catchedError = error
    errorExpectation.fulfill()
  }
  
  waitForExpectations(timeout: 1) { (error) in
    XCTAssertNotNil(catchedError)
  }
}

테스트가 실패하는 것을 확인하자.

테스트를 통과하려면, 던질 에러를 정의해야 한다. 다음 APIClient.swift enum을 추가하자.

enum WebServiceError: Error {
  case DataEmpryError
}

completion handler의 시작도 바꿔보자.

guard let data = data else { 
  completion(nil, WebserviceError.DataEmptyError) 
  return 
}

테스트는 성공하고, 리팩토링할 것도 없다.

처리해야 할 한 가지 에러가 있다. data task의 completion handler는 error 매개변수로 호출된다. 웹서비스는 이 매개변수로 발생하는 모든 에러를 반환한다. 우리 코드는 이 코드를 처리해야 한다. 다음 테스트를 추가하자.

func test_Login_WhenResponseHasError_ReturnsError() {
  
  let sut = APIClient()
  let error = NSError(domain: "SomeError",
                      code: 1234,
                      userInfo: nil)
  let jsonData = "{\"token\": \"1234567890\"}".data(using: .utf8)
  let mockURLSession = MockURLSession(data: jsonData,
                                      urlResponse: nil,
                                      error: error)
  sut.session = mockURLSession
  
  let errorExpectation = expectation(description: "Error")
  var catchedError: Error? = nil
  sut.loginUser(withName: "Foo", password: "Bar") { (token, error) in
    catchedError = error
    errorExpectation.fulfill()
  }
  
  waitForExpectations(timeout: 1) { (error) in
    XCTAssertNotNil(catchedError)
  }
}

모의 URL session을 유효한 응답으로 초기화했다. 이 테스트에서 data로 nil을 전달하면, 심지어 응답 에러를 처리하는 코드를 적지 않았더라도 이미 통과했을 것이다.

테스트를 통과하기 위해, ResponseError case를 WebserviceError에 추가하자. 그리고 다음 코드를 data task의 completion handler의 처음에 추가하자.

guard error == nil else { completion(nil, error); return }

테스트는 성공하고, 리팩토링할 것은 없다.

APIClient에는 몇 가지 남은 테스트와 구현 코드가 있다. 예를 들어, 웹어플리케이션의 투두 아이템에 접근하도록 하려면 아이템을 서버에서 가져오고, 포스트하는 테스트를 추가할 수 있다.

Summary

이 장에서 우리는 XCTest가 제공하는 expectation을 사용해 테스트를 작성했다. 또한, 서버를 모의하기 위해 stub를 사용했다. 우리는 우리 목표(가능한 적은 버그를 가진 완성된 앱)에 근접하도록하는 두 가지 방법에 대해 살펴봤다.

우리는 우리의 모의 URL session의 session data task의 completion handler를 캐치하기 위해 의존성 주입을 사용했다. 이 방법으로, 테스트 데이터를 실제 코드에 넣을 수 있었고, 실제 코드가 예상대로 구현되어는지를 검증할 수 있었다. completion handler가 받은 데이터를 제어해서, 모든 종류의 에러를 흉내낼 수 있었고, 실제 구현 코드가 에러를 다룰 수 있게 했다.

다음 장에서는 이전 장의 다른 부분들을 모을 것이고, 마지막엔 앱이 돌아가는지 확인할 것이다.