RequestInterceptorTests.swift 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754
  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. import Foundation
  25. import Testing
  26. @testable import Alamofire
  27. private struct MockError: Error {}
  28. private struct RetryError: Error {}
  29. // MARK: -
  30. @Suite
  31. struct RetryResultTests {
  32. @Test
  33. func retryRequiredProperty() async throws {
  34. // Given, When
  35. let retry = RetryResult.retry
  36. let retryWithDelay = RetryResult.retryWithDelay(1.0)
  37. let doNotRetry = RetryResult.doNotRetry
  38. let doNotRetryWithError = RetryResult.doNotRetryWithError(MockError())
  39. // Then
  40. #expect(retry.retryRequired == true)
  41. #expect(retryWithDelay.retryRequired == true)
  42. #expect(doNotRetry.retryRequired == false)
  43. #expect(doNotRetryWithError.retryRequired == false)
  44. }
  45. @Test
  46. func delayProperty() async throws {
  47. // Given, When
  48. let retry = RetryResult.retry
  49. let retryWithDelay = RetryResult.retryWithDelay(1.0)
  50. let doNotRetry = RetryResult.doNotRetry
  51. let doNotRetryWithError = RetryResult.doNotRetryWithError(MockError())
  52. // Then
  53. #expect(retry.delay == nil)
  54. #expect(retryWithDelay.delay == 1.0)
  55. #expect(doNotRetry.delay == nil)
  56. #expect(doNotRetryWithError.delay == nil)
  57. }
  58. @Test
  59. func errorProperty() {
  60. // Given, When
  61. let retry = RetryResult.retry
  62. let retryWithDelay = RetryResult.retryWithDelay(1.0)
  63. let doNotRetry = RetryResult.doNotRetry
  64. let doNotRetryWithError = RetryResult.doNotRetryWithError(MockError())
  65. // Then
  66. #expect(retry.error == nil)
  67. #expect(retryWithDelay.error == nil)
  68. #expect(doNotRetry.error == nil)
  69. #expect(doNotRetryWithError.error is MockError)
  70. }
  71. }
  72. // MARK: -
  73. @Suite
  74. struct AdapterTests {
  75. @Test
  76. func thatAdapterCallsAdaptHandler() async throws {
  77. // Given
  78. let urlRequest = Endpoint().urlRequest
  79. let session = Session()
  80. var adapted = false
  81. let adapter = Adapter { request, _, completion in
  82. adapted = true
  83. completion(.success(request))
  84. }
  85. var result: Result<URLRequest, any Error>!
  86. // When
  87. adapter.adapt(urlRequest, for: session) { result = $0 }
  88. // Then
  89. #expect(adapted)
  90. #expect(result.isSuccess)
  91. }
  92. @Test
  93. func thatAdapterCallsAdaptHandlerWithStateAPI() async throws {
  94. // Given
  95. let urlRequest = Endpoint().urlRequest
  96. let session = Session()
  97. let requestID = UUID()
  98. let interceptor = InspectorInterceptor(.adapter { @Sendable request, _, completionHandler in
  99. completionHandler(.success(request))
  100. })
  101. let state = RequestAdapterState(requestID: requestID, session: session)
  102. // When
  103. interceptor.adapt(urlRequest, using: state) { _ in }
  104. let adaptation = interceptor.adaptations.first
  105. // Then
  106. #expect(interceptor.adaptations.count == 1)
  107. #expect(adaptation?.result.isSuccess == true)
  108. #expect(adaptation?.urlRequest == urlRequest)
  109. #expect(adaptation?.state.requestID == requestID)
  110. #expect(adaptation.map(\.state.sessionID) == ObjectIdentifier(session))
  111. }
  112. @Test
  113. func thatAdapterCallsRequestRetrierDefaultImplementationInProtocolExtension() async throws {
  114. // Given
  115. let session = Session(startRequestsImmediately: false)
  116. let request = session.request(.default)
  117. let adapter = InspectorInterceptor(.adapter { @Sendable request, _, completion in
  118. completion(.success(request))
  119. })
  120. // When
  121. adapter.retry(request, for: session, dueTo: MockError()) { _ in }
  122. // Then
  123. #expect(adapter.retryResults.first == .doNotRetry)
  124. }
  125. @Test
  126. func thatAdapterCanBeImplementedAsynchronously() async throws {
  127. // Given
  128. let urlRequest = Endpoint().urlRequest
  129. let session = Session()
  130. let adapter = InspectorInterceptor(.adapter { @Sendable request, _, completion in
  131. DispatchQueue.main.async {
  132. completion(.success(request))
  133. }
  134. })
  135. // When
  136. await withCheckedContinuation { continuation in
  137. adapter.adapt(urlRequest, for: session) { _ in
  138. continuation.resume()
  139. }
  140. }
  141. // Then
  142. #expect(adapter.adaptations.count == 1)
  143. #expect(adapter.adaptations.first?.result.isSuccess == true)
  144. }
  145. }
  146. // MARK: -
  147. @Suite
  148. struct RetrierTests {
  149. @Test
  150. func thatRetrierCallsRetryHandler() async throws {
  151. // Given
  152. let session = Session(startRequestsImmediately: false)
  153. let request = session.request(.default)
  154. let retrier = InspectorInterceptor(Retrier { _, _, _, completion in
  155. completion(.retry)
  156. })
  157. // When
  158. retrier.retry(request, for: session, dueTo: MockError()) { _ in }
  159. // Then
  160. #expect(retrier.retryCalledCount == 1)
  161. #expect(retrier.retryResults.first == .retry)
  162. }
  163. @Test
  164. func thatRetrierCallsRequestAdapterDefaultImplementationInProtocolExtension() async throws {
  165. // Given
  166. let urlRequest = Endpoint().urlRequest
  167. let session = Session()
  168. let retrier = InspectorInterceptor(Retrier { _, _, _, completion in
  169. completion(.retry)
  170. })
  171. // When
  172. retrier.adapt(urlRequest, for: session) { _ in }
  173. // Then
  174. #expect(retrier.adaptations.first?.result.isSuccess == true)
  175. }
  176. @Test
  177. func thatRetrierCanBeImplementedAsynchronously() async throws {
  178. // Given
  179. let session = Session(startRequestsImmediately: false)
  180. let request = session.request(.default)
  181. let retrier = InspectorInterceptor(Retrier { _, _, _, completion in
  182. DispatchQueue.main.async {
  183. completion(.retry)
  184. }
  185. })
  186. // When
  187. await withCheckedContinuation { continuation in
  188. retrier.retry(request, for: session, dueTo: MockError()) { _ in
  189. continuation.resume()
  190. }
  191. }
  192. // Then
  193. #expect(retrier.retryCalledCount == 1)
  194. #expect(retrier.retryResults.first == .retry)
  195. }
  196. }
  197. // MARK: -
  198. @Suite
  199. struct InterceptorTests {
  200. @Test
  201. func adaptHandlerAndRetryHandlerDefaultInitializer() {
  202. // Given
  203. let adaptHandler: AdaptHandler = { urlRequest, _, completion in completion(.success(urlRequest)) }
  204. let retryHandler: RetryHandler = { _, _, _, completion in completion(.doNotRetry) }
  205. // When
  206. let interceptor = Interceptor(adaptHandler: adaptHandler, retryHandler: retryHandler)
  207. // Then
  208. #expect(interceptor.adapters.count == 1)
  209. #expect(interceptor.retriers.count == 1)
  210. }
  211. @Test
  212. func adapterAndRetrierDefaultInitializer() {
  213. // Given
  214. let adapter = Adapter { urlRequest, _, completion in completion(.success(urlRequest)) }
  215. let retrier = Retrier { _, _, _, completion in completion(.doNotRetry) }
  216. // When
  217. let interceptor = Interceptor(adapter: adapter, retrier: retrier)
  218. // Then
  219. #expect(interceptor.adapters.count == 1)
  220. #expect(interceptor.retriers.count == 1)
  221. }
  222. @Test
  223. func adaptersAndRetriersDefaultInitializer() {
  224. // Given
  225. let adapter = Adapter { urlRequest, _, completion in completion(.success(urlRequest)) }
  226. let retrier = Retrier { _, _, _, completion in completion(.doNotRetry) }
  227. // When
  228. let interceptor = Interceptor(adapters: [adapter, adapter], retriers: [retrier, retrier])
  229. // Then
  230. #expect(interceptor.adapters.count == 2)
  231. #expect(interceptor.retriers.count == 2)
  232. }
  233. @Test
  234. func thatInterceptorCanBeComposedOfMultipleRequestInterceptors() {
  235. // Given
  236. let adapter = Adapter { request, _, completion in completion(.success(request)) }
  237. let retrier = Retrier { _, _, _, completion in completion(.doNotRetry) }
  238. let inner = Interceptor(adapter: adapter, retrier: retrier)
  239. // When
  240. let interceptor = Interceptor(interceptors: [inner])
  241. // Then
  242. #expect(interceptor.adapters.count == 1)
  243. #expect(interceptor.retriers.count == 1)
  244. }
  245. @Test
  246. func thatInterceptorCanAdaptRequestWithNoAdapters() {
  247. // Given
  248. let urlRequest = Endpoint().urlRequest
  249. let session = Session()
  250. let interceptor = Interceptor()
  251. var result: Result<URLRequest, any Error>!
  252. // When
  253. interceptor.adapt(urlRequest, for: session) { result = $0 }
  254. // Then
  255. #expect(result.isSuccess == true)
  256. #expect(result.success == urlRequest)
  257. }
  258. @Test
  259. func thatInterceptorCanAdaptRequestWithOneAdapter() {
  260. // Given
  261. let urlRequest = Endpoint().urlRequest
  262. let session = Session()
  263. let adapter = Adapter { _, _, completion in completion(.failure(MockError())) }
  264. let interceptor = Interceptor(adapters: [adapter])
  265. var result: Result<URLRequest, any Error>!
  266. // When
  267. interceptor.adapt(urlRequest, for: session) { result = $0 }
  268. // Then
  269. #expect(result.isFailure)
  270. #expect(result.failure is MockError)
  271. }
  272. @Test
  273. func thatInterceptorCanAdaptRequestWithMultipleAdapters() {
  274. // Given
  275. let urlRequest = Endpoint().urlRequest
  276. let session = Session()
  277. let adapter1 = Adapter { urlRequest, _, completion in completion(.success(urlRequest)) }
  278. let adapter2 = Adapter { _, _, completion in completion(.failure(MockError())) }
  279. let interceptor = Interceptor(adapters: [adapter1, adapter2])
  280. var result: Result<URLRequest, any Error>!
  281. // When
  282. interceptor.adapt(urlRequest, for: session) { result = $0 }
  283. // Then
  284. #expect(result.isFailure)
  285. #expect(result.failure is MockError)
  286. }
  287. @Test
  288. func thatInterceptorCanAdaptRequestWithMultipleAdaptersUsingStateAPI() {
  289. // Given
  290. let urlRequest = Endpoint().urlRequest
  291. let session = Session()
  292. let adapter1 = Adapter { urlRequest, _, completion in completion(.success(urlRequest)) }
  293. let adapter2 = Adapter { _, _, completion in completion(.failure(MockError())) }
  294. let interceptor = Interceptor(adapters: [adapter1, adapter2])
  295. let state = RequestAdapterState(requestID: UUID(), session: session)
  296. var result: Result<URLRequest, any Error>!
  297. // When
  298. interceptor.adapt(urlRequest, using: state) { result = $0 }
  299. // Then
  300. #expect(result.isFailure)
  301. #expect(result.failure is MockError)
  302. }
  303. @Test
  304. func thatInterceptorCanAdaptRequestAsynchronously() async throws {
  305. // Given
  306. let urlRequest = Endpoint().urlRequest
  307. let session = Session()
  308. let adapter = Adapter { _, _, completion in
  309. DispatchQueue.main.async {
  310. completion(.failure(MockError()))
  311. }
  312. }
  313. let interceptor = InspectorInterceptor(Interceptor(adapters: [adapter]))
  314. // When
  315. await withCheckedContinuation { continuation in
  316. interceptor.adapt(urlRequest, for: session) { _ in
  317. continuation.resume()
  318. }
  319. }
  320. // Then
  321. #expect(interceptor.adaptations.first?.result.isFailure == true)
  322. #expect(interceptor.adaptations.first?.result.failure is MockError)
  323. }
  324. @Test
  325. func thatInterceptorCanRetryRequestWithNoRetriers() {
  326. // Given
  327. let session = Session(startRequestsImmediately: false)
  328. let request = session.request(.default)
  329. let interceptor = Interceptor()
  330. var result: RetryResult!
  331. // When
  332. interceptor.retry(request, for: session, dueTo: MockError()) { result = $0 }
  333. // Then
  334. #expect(result == .doNotRetry)
  335. }
  336. @Test
  337. func thatInterceptorCanRetryRequestWithOneRetrier() {
  338. // Given
  339. let session = Session(startRequestsImmediately: false)
  340. let request = session.request(.default)
  341. let retrier = Retrier { _, _, _, completion in completion(.retry) }
  342. let interceptor = Interceptor(retriers: [retrier])
  343. var result: RetryResult!
  344. // When
  345. interceptor.retry(request, for: session, dueTo: MockError()) { result = $0 }
  346. // Then
  347. #expect(result == .retry)
  348. }
  349. @Test
  350. func thatInterceptorCanRetryRequestWithMultipleRetriers() {
  351. // Given
  352. let session = Session(startRequestsImmediately: false)
  353. let request = session.request(.default)
  354. let retrier1 = Retrier { _, _, _, completion in completion(.doNotRetry) }
  355. let retrier2 = Retrier { _, _, _, completion in completion(.retry) }
  356. let interceptor = Interceptor(retriers: [retrier1, retrier2])
  357. var result: RetryResult!
  358. // When
  359. interceptor.retry(request, for: session, dueTo: MockError()) { result = $0 }
  360. // Then
  361. #expect(result == .retry)
  362. }
  363. @Test
  364. func thatInterceptorCanRetryRequestAsynchronously() async throws {
  365. // Given
  366. let session = Session(startRequestsImmediately: false)
  367. let request = session.request(.default)
  368. let retrier = Retrier { _, _, _, completion in
  369. DispatchQueue.main.async {
  370. completion(.retry)
  371. }
  372. }
  373. let interceptor = InspectorInterceptor(Interceptor(retriers: [retrier]))
  374. // When
  375. await withCheckedContinuation { continuation in
  376. interceptor.retry(request, for: session, dueTo: MockError()) { _ in
  377. continuation.resume()
  378. }
  379. }
  380. // Then
  381. #expect(interceptor.retryResults.first == .retry)
  382. }
  383. @Test
  384. func thatInterceptorStopsIteratingThroughPendingRetriersWithRetryResult() {
  385. // Given
  386. let session = Session(startRequestsImmediately: false)
  387. let request = session.request(.default)
  388. var retrier2Called = false
  389. let retrier1 = Retrier { _, _, _, completion in completion(.retry) }
  390. let retrier2 = Retrier { _, _, _, completion in retrier2Called = true; completion(.doNotRetry) }
  391. let interceptor = Interceptor(retriers: [retrier1, retrier2])
  392. var result: RetryResult!
  393. // When
  394. interceptor.retry(request, for: session, dueTo: MockError()) { result = $0 }
  395. // Then
  396. #expect(result == .retry)
  397. #expect(retrier2Called == false)
  398. }
  399. @Test
  400. func thatInterceptorStopsIteratingThroughPendingRetriersWithRetryWithDelayResult() {
  401. // Given
  402. let session = Session(startRequestsImmediately: false)
  403. let request = session.request(.default)
  404. var retrier2Called = false
  405. let retrier1 = Retrier { _, _, _, completion in completion(.retryWithDelay(1.0)) }
  406. let retrier2 = Retrier { _, _, _, completion in retrier2Called = true; completion(.doNotRetry) }
  407. let interceptor = Interceptor(retriers: [retrier1, retrier2])
  408. var result: RetryResult!
  409. // When
  410. interceptor.retry(request, for: session, dueTo: MockError()) { result = $0 }
  411. // Then
  412. #expect(result == .retryWithDelay(1.0))
  413. #expect(result.delay == 1.0)
  414. #expect(retrier2Called == false)
  415. }
  416. @Test
  417. func thatInterceptorStopsIteratingThroughPendingRetriersWithDoNotRetryResult() {
  418. // Given
  419. let session = Session(startRequestsImmediately: false)
  420. let request = session.request(.default)
  421. var retrier2Called = false
  422. let retrier1 = Retrier { _, _, _, completion in completion(.doNotRetryWithError(RetryError())) }
  423. let retrier2 = Retrier { _, _, _, completion in retrier2Called = true; completion(.doNotRetry) }
  424. let interceptor = Interceptor(retriers: [retrier1, retrier2])
  425. var result: RetryResult!
  426. // When
  427. interceptor.retry(request, for: session, dueTo: MockError()) { result = $0 }
  428. // Then
  429. #expect(result == RetryResult.doNotRetryWithError(RetryError()))
  430. #expect(result.error is RetryError)
  431. #expect(retrier2Called == false)
  432. }
  433. }
  434. // MARK: - Functional Tests
  435. @Suite
  436. struct InterceptorRequestTests {
  437. @Test
  438. func thatRetryPolicyRetriesRequestTimeout() async throws {
  439. // Given
  440. let interceptor = InspectorInterceptor(RetryPolicy(retryLimit: 1, exponentialBackoffScale: 0.1))
  441. let urlRequest = Endpoint.delay(1).modifying(\.timeout, to: 0.01)
  442. // When
  443. let request = AF.request(urlRequest, interceptor: interceptor)
  444. await request.finished()
  445. // Then
  446. #expect(request.tasks.count == 2, "There should be two tasks, one original, one retry.")
  447. #expect(interceptor.retryCalledCount == 2, "retry() should be called twice.")
  448. #expect(interceptor.retryResults == [.retryWithDelay(0.1), .doNotRetry], "RetryResults should be .retryWithDelay(0.1), .doNotRetry")
  449. }
  450. }
  451. extension DataRequest {
  452. func finished() async {
  453. await withCheckedContinuation { continuation in
  454. response { _ in continuation.resume() }
  455. }
  456. }
  457. }
  458. // MARK: - Static Accessors
  459. @Suite
  460. struct StaticAccessorTests {
  461. func consumeRequestAdapter(_ requestAdapter: any RequestAdapter) {
  462. _ = requestAdapter
  463. }
  464. func consumeRequestRetrier(_ requestRetrier: any RequestRetrier) {
  465. _ = requestRetrier
  466. }
  467. func consumeRequestInterceptor(_ requestInterceptor: any RequestInterceptor) {
  468. _ = requestInterceptor
  469. }
  470. @Test
  471. func thatAdapterCanBeCreatedStaticallyFromProtocol() {
  472. // Given, When, Then
  473. consumeRequestAdapter(.adapter { request, _, completion in completion(.success(request)) })
  474. }
  475. @Test
  476. func thatRetrierCanBeCreatedStaticallyFromProtocol() {
  477. // Given, When, Then
  478. consumeRequestRetrier(.retrier { _, _, _, completion in completion(.doNotRetry) })
  479. }
  480. @Test
  481. func thatInterceptorCanBeCreatedStaticallyFromProtocol() {
  482. // Given, When, Then
  483. consumeRequestInterceptor(.interceptor())
  484. }
  485. @Test
  486. func thatRetryPolicyCanBeCreatedStaticallyFromProtocol() {
  487. // Given, When, Then
  488. consumeRequestInterceptor(.retryPolicy())
  489. }
  490. @Test
  491. func thatConnectionLostRetryPolicyCanBeCreatedStaticallyFromProtocol() {
  492. // Given, When, Then
  493. consumeRequestInterceptor(.connectionLostRetryPolicy())
  494. }
  495. }
  496. // MARK: - Helpers
  497. /// Class which captures the output of any underlying `RequestInterceptor`.
  498. final class InspectorInterceptor<Interceptor: RequestInterceptor>: RequestInterceptor, Sendable {
  499. private struct State {
  500. let onAdaptation: ((_ result: Result<URLRequest, any Error>) -> Void)?
  501. let onRetry: ((_ retryResult: RetryResult) -> Void)?
  502. var adaptations: [Adaptation] = []
  503. var retries: [Retry] = []
  504. }
  505. private let state: Protected<State>
  506. struct Adaptation {
  507. var urlRequest: URLRequest
  508. var state: State
  509. var result: Result<URLRequest, any Error>
  510. var date: Date
  511. struct State {
  512. var requestID: UUID
  513. var sessionID: ObjectIdentifier
  514. }
  515. }
  516. struct Retry {
  517. var result: RetryResult
  518. var date: Date
  519. }
  520. /// Underlying interceptor.
  521. let interceptor: Interceptor
  522. /// Result of performed adaptations.
  523. var adaptations: [Adaptation] { state.read(\.adaptations) }
  524. /// Retry events.
  525. var retries: [Retry] { state.read(\.retries) }
  526. /// Result of performed retries.
  527. var retryResults: [RetryResult] { retries.map(\.result) }
  528. /// Number of times `retry` was called.
  529. var retryCalledCount: Int { state.read(\.retries.count) }
  530. init(_ interceptor: Interceptor,
  531. onAdaptation: ((_ result: Result<URLRequest, any Error>) -> Void)? = nil,
  532. onRetry: ((_ retryResult: RetryResult) -> Void)? = nil) {
  533. self.interceptor = interceptor
  534. state = Protected(State(onAdaptation: onAdaptation, onRetry: onRetry))
  535. }
  536. func adapt(_ urlRequest: URLRequest, for session: Session, completion: @escaping (Result<URLRequest, any Error>) -> Void) {
  537. let sessionID = ObjectIdentifier(session)
  538. interceptor.adapt(urlRequest, for: session) { result in
  539. let onAdaptation = self.state.write { state in
  540. state.adaptations.append(.init(urlRequest: urlRequest, state: .init(requestID: .zero, sessionID: sessionID), result: result, date: .now))
  541. return state.onAdaptation
  542. }
  543. completion(result)
  544. onAdaptation?(result)
  545. }
  546. }
  547. func adapt(_ urlRequest: URLRequest, using adapterState: RequestAdapterState, completion: @escaping @Sendable (Result<URLRequest, any Error>) -> Void) {
  548. let requestID = adapterState.requestID
  549. let sessionID = ObjectIdentifier(adapterState.session)
  550. interceptor.adapt(urlRequest, using: adapterState) { result in
  551. let onAdaptation = self.state.write { state in
  552. state.adaptations.append(.init(urlRequest: urlRequest, state: .init(requestID: requestID, sessionID: sessionID), result: result, date: .now))
  553. return state.onAdaptation
  554. }
  555. completion(result)
  556. onAdaptation?(result)
  557. }
  558. }
  559. func retry(_ request: Request, for session: Session, dueTo error: any Error, completion: @escaping @Sendable (RetryResult) -> Void) {
  560. interceptor.retry(request, for: session, dueTo: error) { result in
  561. let onRetry = self.state.write { state in
  562. state.retries.append(.init(result: result, date: .now))
  563. return state.onRetry
  564. }
  565. completion(result)
  566. onRetry?(result)
  567. }
  568. }
  569. }
  570. extension UUID {
  571. static let zero = UUID(uuidString: "00000000-0000-0000-0000-000000000000")!
  572. }
  573. /// Retry a request once, allowing the second to succeed using the method path.
  574. final class SingleRetrier: RequestInterceptor, Sendable {
  575. private let state = Protected(false)
  576. private var hasRetried: Bool { state.read(\.self) }
  577. func adapt(_ urlRequest: URLRequest, for session: Session, completion: @escaping @Sendable (Result<URLRequest, any Error>) -> Void) {
  578. if hasRetried {
  579. let method = urlRequest.method ?? .get
  580. let endpoint = Endpoint(path: .method(method),
  581. method: method,
  582. headers: urlRequest.headers)
  583. completion(.success(endpoint.urlRequest))
  584. } else {
  585. completion(.success(urlRequest))
  586. }
  587. }
  588. func retry(_ request: Request, for session: Session, dueTo error: any Error, completion: @escaping @Sendable (RetryResult) -> Void) {
  589. completion(hasRetried ? .doNotRetry : .retry)
  590. state.write(true)
  591. }
  592. }
  593. extension Alamofire.RetryResult: Swift.Equatable {
  594. public static func ==(lhs: RetryResult, rhs: RetryResult) -> Bool {
  595. switch (lhs, rhs) {
  596. case (.retry, .retry),
  597. (.doNotRetry, .doNotRetry),
  598. (.doNotRetryWithError, .doNotRetryWithError):
  599. true
  600. case let (.retryWithDelay(leftDelay), .retryWithDelay(rightDelay)):
  601. leftDelay == rightDelay
  602. default:
  603. false
  604. }
  605. }
  606. }