RequestInterceptorTests.swift 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575
  1. //
  2. // RequestInterceptorTests.swift
  3. //
  4. // Copyright (c) 2019 Alamofire Software Foundation (http://alamofire.org/)
  5. //
  6. // Permission is hereby granted, free of charge, to any person obtaining a copy
  7. // of this software and associated documentation files (the "Software"), to deal
  8. // in the Software without restriction, including without limitation the rights
  9. // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  10. // copies of the Software, and to permit persons to whom the Software is
  11. // furnished to do so, subject to the following conditions:
  12. //
  13. // The above copyright notice and this permission notice shall be included in
  14. // all copies or substantial portions of the Software.
  15. //
  16. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  17. // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  18. // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  19. // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  20. // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  21. // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  22. // THE SOFTWARE.
  23. //
  24. @testable import Alamofire
  25. import Foundation
  26. import XCTest
  27. private struct MockError: Error {}
  28. private struct RetryError: Error {}
  29. // MARK: -
  30. final class RetryResultTestCase: BaseTestCase {
  31. func testRetryRequiredProperty() {
  32. // Given, When
  33. let retry = RetryResult.retry
  34. let retryWithDelay = RetryResult.retryWithDelay(1.0)
  35. let doNotRetry = RetryResult.doNotRetry
  36. let doNotRetryWithError = RetryResult.doNotRetryWithError(MockError())
  37. // Then
  38. XCTAssertTrue(retry.retryRequired)
  39. XCTAssertTrue(retryWithDelay.retryRequired)
  40. XCTAssertFalse(doNotRetry.retryRequired)
  41. XCTAssertFalse(doNotRetryWithError.retryRequired)
  42. }
  43. func testDelayProperty() {
  44. // Given, When
  45. let retry = RetryResult.retry
  46. let retryWithDelay = RetryResult.retryWithDelay(1.0)
  47. let doNotRetry = RetryResult.doNotRetry
  48. let doNotRetryWithError = RetryResult.doNotRetryWithError(MockError())
  49. // Then
  50. XCTAssertEqual(retry.delay, nil)
  51. XCTAssertEqual(retryWithDelay.delay, 1.0)
  52. XCTAssertEqual(doNotRetry.delay, nil)
  53. XCTAssertEqual(doNotRetryWithError.delay, nil)
  54. }
  55. func testErrorProperty() {
  56. // Given, When
  57. let retry = RetryResult.retry
  58. let retryWithDelay = RetryResult.retryWithDelay(1.0)
  59. let doNotRetry = RetryResult.doNotRetry
  60. let doNotRetryWithError = RetryResult.doNotRetryWithError(MockError())
  61. // Then
  62. XCTAssertNil(retry.error)
  63. XCTAssertNil(retryWithDelay.error)
  64. XCTAssertNil(doNotRetry.error)
  65. XCTAssertTrue(doNotRetryWithError.error is MockError)
  66. }
  67. }
  68. // MARK: -
  69. final class AdapterTestCase: BaseTestCase {
  70. func testThatAdapterCallsAdaptHandler() {
  71. // Given
  72. let urlRequest = URLRequest(url: URL(string: "https://httpbin.org/get")!)
  73. let session = Session()
  74. var adapted = false
  75. let adapter = Adapter { request, _, completion in
  76. adapted = true
  77. completion(.success(request))
  78. }
  79. var result: Result<URLRequest, Error>!
  80. // When
  81. adapter.adapt(urlRequest, for: session) { result = $0 }
  82. // Then
  83. XCTAssertTrue(adapted)
  84. XCTAssertTrue(result.isSuccess)
  85. }
  86. func testThatAdapterCallsRequestRetrierDefaultImplementationInProtocolExtension() {
  87. // Given
  88. let url = URL(string: "https://httpbin.org/get")!
  89. let session = Session(startRequestsImmediately: false)
  90. let request = session.request(url)
  91. let adapter = Adapter { request, _, completion in
  92. completion(.success(request))
  93. }
  94. var result: RetryResult!
  95. // When
  96. adapter.retry(request, for: session, dueTo: MockError()) { result = $0 }
  97. // Then
  98. XCTAssertEqual(result, .doNotRetry)
  99. }
  100. func testThatAdapterCanBeImplementedAsynchronously() {
  101. // Given
  102. let urlRequest = URLRequest(url: URL(string: "https://httpbin.org/get")!)
  103. let session = Session()
  104. var adapted = false
  105. let adapter = Adapter { request, _, completion in
  106. adapted = true
  107. DispatchQueue.main.async {
  108. completion(.success(request))
  109. }
  110. }
  111. var result: Result<URLRequest, Error>!
  112. let completesExpectation = expectation(description: "adapter completes")
  113. // When
  114. adapter.adapt(urlRequest, for: session) {
  115. result = $0
  116. completesExpectation.fulfill()
  117. }
  118. waitForExpectations(timeout: timeout)
  119. // Then
  120. XCTAssertTrue(adapted)
  121. XCTAssertTrue(result.isSuccess)
  122. }
  123. }
  124. // MARK: -
  125. final class RetrierTestCase: BaseTestCase {
  126. func testThatRetrierCallsRetryHandler() {
  127. // Given
  128. let url = URL(string: "https://httpbin.org/get")!
  129. let session = Session(startRequestsImmediately: false)
  130. let request = session.request(url)
  131. var retried = false
  132. let retrier = Retrier { _, _, _, completion in
  133. retried = true
  134. completion(.retry)
  135. }
  136. var result: RetryResult!
  137. // When
  138. retrier.retry(request, for: session, dueTo: MockError()) { result = $0 }
  139. // Then
  140. XCTAssertTrue(retried)
  141. XCTAssertEqual(result, .retry)
  142. }
  143. func testThatRetrierCallsRequestAdapterDefaultImplementationInProtocolExtension() {
  144. // Given
  145. let urlRequest = URLRequest(url: URL(string: "https://httpbin.org/get")!)
  146. let session = Session()
  147. let retrier = Retrier { _, _, _, completion in
  148. completion(.retry)
  149. }
  150. var result: Result<URLRequest, Error>!
  151. // When
  152. retrier.adapt(urlRequest, for: session) { result = $0 }
  153. // Then
  154. XCTAssertTrue(result.isSuccess)
  155. }
  156. func testThatRetrierCanBeImplementedAsynchronously() {
  157. // Given
  158. let url = URL(string: "https://httpbin.org/get")!
  159. let session = Session(startRequestsImmediately: false)
  160. let request = session.request(url)
  161. var retried = false
  162. let retrier = Retrier { _, _, _, completion in
  163. retried = true
  164. DispatchQueue.main.async {
  165. completion(.retry)
  166. }
  167. }
  168. var result: RetryResult!
  169. let completesExpectation = expectation(description: "retrier completes")
  170. // When
  171. retrier.retry(request, for: session, dueTo: MockError()) {
  172. result = $0
  173. completesExpectation.fulfill()
  174. }
  175. waitForExpectations(timeout: timeout)
  176. // Then
  177. XCTAssertTrue(retried)
  178. XCTAssertEqual(result, .retry)
  179. }
  180. }
  181. // MARK: -
  182. final class InterceptorTestCase: BaseTestCase {
  183. func testAdaptHandlerAndRetryHandlerDefaultInitializer() {
  184. // Given
  185. let adaptHandler: AdaptHandler = { urlRequest, _, completion in completion(.success(urlRequest)) }
  186. let retryHandler: RetryHandler = { _, _, _, completion in completion(.doNotRetry) }
  187. // When
  188. let interceptor = Interceptor(adaptHandler: adaptHandler, retryHandler: retryHandler)
  189. // Then
  190. XCTAssertEqual(interceptor.adapters.count, 1)
  191. XCTAssertEqual(interceptor.retriers.count, 1)
  192. }
  193. func testAdapterAndRetrierDefaultInitializer() {
  194. // Given
  195. let adapter = Adapter { urlRequest, _, completion in completion(.success(urlRequest)) }
  196. let retrier = Retrier { _, _, _, completion in completion(.doNotRetry) }
  197. // When
  198. let interceptor = Interceptor(adapter: adapter, retrier: retrier)
  199. // Then
  200. XCTAssertEqual(interceptor.adapters.count, 1)
  201. XCTAssertEqual(interceptor.retriers.count, 1)
  202. }
  203. func testAdaptersAndRetriersDefaultInitializer() {
  204. // Given
  205. let adapter = Adapter { urlRequest, _, completion in completion(.success(urlRequest)) }
  206. let retrier = Retrier { _, _, _, completion in completion(.doNotRetry) }
  207. // When
  208. let interceptor = Interceptor(adapters: [adapter, adapter], retriers: [retrier, retrier])
  209. // Then
  210. XCTAssertEqual(interceptor.adapters.count, 2)
  211. XCTAssertEqual(interceptor.retriers.count, 2)
  212. }
  213. func testThatInterceptorCanAdaptRequestWithNoAdapters() {
  214. // Given
  215. let urlRequest = URLRequest(url: URL(string: "https://httpbin.org/get")!)
  216. let session = Session()
  217. let interceptor = Interceptor()
  218. var result: Result<URLRequest, Error>!
  219. // When
  220. interceptor.adapt(urlRequest, for: session) { result = $0 }
  221. // Then
  222. XCTAssertTrue(result.isSuccess)
  223. XCTAssertEqual(result.success, urlRequest)
  224. }
  225. func testThatInterceptorCanAdaptRequestWithOneAdapter() {
  226. // Given
  227. let urlRequest = URLRequest(url: URL(string: "https://httpbin.org/get")!)
  228. let session = Session()
  229. let adapter = Adapter { _, _, completion in completion(.failure(MockError())) }
  230. let interceptor = Interceptor(adapters: [adapter])
  231. var result: Result<URLRequest, Error>!
  232. // When
  233. interceptor.adapt(urlRequest, for: session) { result = $0 }
  234. // Then
  235. XCTAssertTrue(result.isFailure)
  236. XCTAssertTrue(result.failure is MockError)
  237. }
  238. func testThatInterceptorCanAdaptRequestWithMultipleAdapters() {
  239. // Given
  240. let urlRequest = URLRequest(url: URL(string: "https://httpbin.org/get")!)
  241. let session = Session()
  242. let adapter1 = Adapter { urlRequest, _, completion in completion(.success(urlRequest)) }
  243. let adapter2 = Adapter { _, _, completion in completion(.failure(MockError())) }
  244. let interceptor = Interceptor(adapters: [adapter1, adapter2])
  245. var result: Result<URLRequest, Error>!
  246. // When
  247. interceptor.adapt(urlRequest, for: session) { result = $0 }
  248. // Then
  249. XCTAssertTrue(result.isFailure)
  250. XCTAssertTrue(result.failure is MockError)
  251. }
  252. func testThatInterceptorCanAdaptRequestAsynchronously() {
  253. // Given
  254. let urlRequest = URLRequest(url: URL(string: "https://httpbin.org/get")!)
  255. let session = Session()
  256. let adapter = Adapter { _, _, completion in
  257. DispatchQueue.main.async {
  258. completion(.failure(MockError()))
  259. }
  260. }
  261. let interceptor = Interceptor(adapters: [adapter])
  262. var result: Result<URLRequest, Error>!
  263. let completesExpectation = expectation(description: "interceptor completes")
  264. // When
  265. interceptor.adapt(urlRequest, for: session) {
  266. result = $0
  267. completesExpectation.fulfill()
  268. }
  269. waitForExpectations(timeout: timeout)
  270. // Then
  271. XCTAssertTrue(result.isFailure)
  272. XCTAssertTrue(result.failure is MockError)
  273. }
  274. func testThatInterceptorCanRetryRequestWithNoRetriers() {
  275. // Given
  276. let url = URL(string: "https://httpbin.org/get")!
  277. let session = Session(startRequestsImmediately: false)
  278. let request = session.request(url)
  279. let interceptor = Interceptor()
  280. var result: RetryResult!
  281. // When
  282. interceptor.retry(request, for: session, dueTo: MockError()) { result = $0 }
  283. // Then
  284. XCTAssertEqual(result, .doNotRetry)
  285. }
  286. func testThatInterceptorCanRetryRequestWithOneRetrier() {
  287. // Given
  288. let url = URL(string: "https://httpbin.org/get")!
  289. let session = Session(startRequestsImmediately: false)
  290. let request = session.request(url)
  291. let retrier = Retrier { _, _, _, completion in completion(.retry) }
  292. let interceptor = Interceptor(retriers: [retrier])
  293. var result: RetryResult!
  294. // When
  295. interceptor.retry(request, for: session, dueTo: MockError()) { result = $0 }
  296. // Then
  297. XCTAssertEqual(result, .retry)
  298. }
  299. func testThatInterceptorCanRetryRequestWithMultipleRetriers() {
  300. // Given
  301. let url = URL(string: "https://httpbin.org/get")!
  302. let session = Session(startRequestsImmediately: false)
  303. let request = session.request(url)
  304. let retrier1 = Retrier { _, _, _, completion in completion(.doNotRetry) }
  305. let retrier2 = Retrier { _, _, _, completion in completion(.retry) }
  306. let interceptor = Interceptor(retriers: [retrier1, retrier2])
  307. var result: RetryResult!
  308. // When
  309. interceptor.retry(request, for: session, dueTo: MockError()) { result = $0 }
  310. // Then
  311. XCTAssertEqual(result, .retry)
  312. }
  313. func testThatInterceptorCanRetryRequestAsynchronously() {
  314. // Given
  315. let url = URL(string: "https://httpbin.org/get")!
  316. let session = Session(startRequestsImmediately: false)
  317. let request = session.request(url)
  318. let retrier = Retrier { _, _, _, completion in
  319. DispatchQueue.main.async {
  320. completion(.retry)
  321. }
  322. }
  323. let interceptor = Interceptor(retriers: [retrier])
  324. var result: RetryResult!
  325. let completesExpectation = expectation(description: "interceptor completes")
  326. // When
  327. interceptor.retry(request, for: session, dueTo: MockError()) {
  328. result = $0
  329. completesExpectation.fulfill()
  330. }
  331. waitForExpectations(timeout: timeout)
  332. // Then
  333. XCTAssertEqual(result, .retry)
  334. }
  335. func testThatInterceptorStopsIteratingThroughPendingRetriersWithRetryResult() {
  336. // Given
  337. let url = URL(string: "https://httpbin.org/get")!
  338. let session = Session(startRequestsImmediately: false)
  339. let request = session.request(url)
  340. var retrier2Called = false
  341. let retrier1 = Retrier { _, _, _, completion in completion(.retry) }
  342. let retrier2 = Retrier { _, _, _, completion in retrier2Called = true; completion(.doNotRetry) }
  343. let interceptor = Interceptor(retriers: [retrier1, retrier2])
  344. var result: RetryResult!
  345. // When
  346. interceptor.retry(request, for: session, dueTo: MockError()) { result = $0 }
  347. // Then
  348. XCTAssertEqual(result, .retry)
  349. XCTAssertFalse(retrier2Called)
  350. }
  351. func testThatInterceptorStopsIteratingThroughPendingRetriersWithRetryWithDelayResult() {
  352. // Given
  353. let url = URL(string: "https://httpbin.org/get")!
  354. let session = Session(startRequestsImmediately: false)
  355. let request = session.request(url)
  356. var retrier2Called = false
  357. let retrier1 = Retrier { _, _, _, completion in completion(.retryWithDelay(1.0)) }
  358. let retrier2 = Retrier { _, _, _, completion in retrier2Called = true; completion(.doNotRetry) }
  359. let interceptor = Interceptor(retriers: [retrier1, retrier2])
  360. var result: RetryResult!
  361. // When
  362. interceptor.retry(request, for: session, dueTo: MockError()) { result = $0 }
  363. // Then
  364. XCTAssertEqual(result, .retryWithDelay(1.0))
  365. XCTAssertEqual(result.delay, 1.0)
  366. XCTAssertFalse(retrier2Called)
  367. }
  368. func testThatInterceptorStopsIteratingThroughPendingRetriersWithDoNotRetryResult() {
  369. // Given
  370. let url = URL(string: "https://httpbin.org/get")!
  371. let session = Session(startRequestsImmediately: false)
  372. let request = session.request(url)
  373. var retrier2Called = false
  374. let retrier1 = Retrier { _, _, _, completion in completion(.doNotRetryWithError(RetryError())) }
  375. let retrier2 = Retrier { _, _, _, completion in retrier2Called = true; completion(.doNotRetry) }
  376. let interceptor = Interceptor(retriers: [retrier1, retrier2])
  377. var result: RetryResult!
  378. // When
  379. interceptor.retry(request, for: session, dueTo: MockError()) { result = $0 }
  380. // Then
  381. XCTAssertEqual(result, RetryResult.doNotRetryWithError(RetryError()))
  382. XCTAssertTrue(result.error is RetryError)
  383. XCTAssertFalse(retrier2Called)
  384. }
  385. }
  386. // MARK: - Functional Tests
  387. final class InterceptorRequestTests: BaseTestCase {
  388. func testThatRetryPolicyRetriesRequestTimeout() {
  389. // Given
  390. let interceptor = InspectorInterceptor(RetryPolicy(retryLimit: 1, exponentialBackoffScale: 0.1))
  391. let urlRequest = URLRequest.makeHTTPBinRequest(path: "delay/1", timeout: 0.01)
  392. let expect = expectation(description: "request completed")
  393. // When
  394. let request = AF.request(urlRequest, interceptor: interceptor).response { _ in
  395. expect.fulfill()
  396. }
  397. waitForExpectations(timeout: timeout)
  398. // Then
  399. XCTAssertEqual(request.tasks.count, 2, "There should be two tasks, one original, one retry.")
  400. XCTAssertEqual(interceptor.retryCalledCount, 2, "retry() should be called twice.")
  401. XCTAssertEqual(interceptor.retries, [.retryWithDelay(0.1), .doNotRetry], "RetryResults should retryWithDelay, doNotRetry")
  402. }
  403. }
  404. // MARK: - Helpers
  405. /// Class which captures the output of any underlying `RequestInterceptor`.
  406. final class InspectorInterceptor<Interceptor: RequestInterceptor>: RequestInterceptor {
  407. var onAdaptation: ((Result<URLRequest, Error>) -> Void)?
  408. var onRetry: ((RetryResult) -> Void)?
  409. private(set) var adaptations: [Result<URLRequest, Error>] = []
  410. private(set) var retries: [RetryResult] = []
  411. /// Number of times `retry` was called.
  412. var retryCalledCount: Int { return retries.count }
  413. let interceptor: Interceptor
  414. init(_ interceptor: Interceptor) {
  415. self.interceptor = interceptor
  416. }
  417. func adapt(_ urlRequest: URLRequest, for session: Session, completion: @escaping (Result<URLRequest, Error>) -> Void) {
  418. interceptor.adapt(urlRequest, for: session) { result in
  419. self.adaptations.append(result)
  420. completion(result)
  421. self.onAdaptation?(result)
  422. }
  423. }
  424. func retry(_ request: Request, for session: Session, dueTo error: Error, completion: @escaping (RetryResult) -> Void) {
  425. interceptor.retry(request, for: session, dueTo: error) { result in
  426. self.retries.append(result)
  427. completion(result)
  428. self.onRetry?(result)
  429. }
  430. }
  431. }
  432. extension RetryResult: Equatable {
  433. public static func ==(lhs: RetryResult, rhs: RetryResult) -> Bool {
  434. switch (lhs, rhs) {
  435. case (.retry, .retry),
  436. (.doNotRetry, .doNotRetry),
  437. (.doNotRetryWithError, .doNotRetryWithError):
  438. return true
  439. case let (.retryWithDelay(leftDelay), .retryWithDelay(rightDelay)):
  440. return leftDelay == rightDelay
  441. default:
  442. return false
  443. }
  444. }
  445. }