| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367 | ////  RetryPolicy.swift////  Copyright (c) 2019 Alamofire Software Foundation (http://alamofire.org/)////  Permission is hereby granted, free of charge, to any person obtaining a copy//  of this software and associated documentation files (the "Software"), to deal//  in the Software without restriction, including without limitation the rights//  to use, copy, modify, merge, publish, distribute, sublicense, and/or sell//  copies of the Software, and to permit persons to whom the Software is//  furnished to do so, subject to the following conditions:////  The above copyright notice and this permission notice shall be included in//  all copies or substantial portions of the Software.////  THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR//  IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,//  FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE//  AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER//  LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,//  OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN//  THE SOFTWARE.//import Foundation/// A retry policy that retries requests using an exponential backoff for allowed HTTP methods and HTTP status codes/// as well as certain types of networking errors.open class RetryPolicy: RequestInterceptor {    /// The default retry limit for retry policies.    public static let defaultRetryLimit: UInt = 2    /// The default exponential backoff base for retry policies (must be a minimum of 2).    public static let defaultExponentialBackoffBase: UInt = 2    /// The default exponential backoff scale for retry policies.    public static let defaultExponentialBackoffScale: Double = 0.5    /// The default HTTP methods to retry.    /// See [RFC 2616 - Section 9.1.2](https://www.w3.org/Protocols/rfc2616/rfc2616-sec9.html) for more information.    public static let defaultRetryableHTTPMethods: Set<HTTPMethod> = [.delete, // [Delete](https://www.w3.org/Protocols/rfc2616/rfc2616-sec9.html#sec9.7) - not always idempotent                                                                      .get, // [GET](https://www.w3.org/Protocols/rfc2616/rfc2616-sec9.html#sec9.3) - generally idempotent                                                                      .head, // [HEAD](https://www.w3.org/Protocols/rfc2616/rfc2616-sec9.html#sec9.4) - generally idempotent                                                                      .options, // [OPTIONS](https://www.w3.org/Protocols/rfc2616/rfc2616-sec9.html#sec9.2) - inherently idempotent                                                                      .put, // [PUT](https://www.w3.org/Protocols/rfc2616/rfc2616-sec9.html#sec9.6) - not always idempotent                                                                      .trace // [TRACE](https://www.w3.org/Protocols/rfc2616/rfc2616-sec9.html#sec9.8) - inherently idempotent    ]    /// The default HTTP status codes to retry.    /// See [RFC 2616 - Section 10](https://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html#sec10) for more information.    public static let defaultRetryableHTTPStatusCodes: Set<Int> = [408, // [Request Timeout](https://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html#sec10.4.9)                                                                   500, // [Internal Server Error](https://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html#sec10.5.1)                                                                   502, // [Bad Gateway](https://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html#sec10.5.3)                                                                   503, // [Service Unavailable](https://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html#sec10.5.4)                                                                   504 // [Gateway Timeout](https://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html#sec10.5.5)    ]    /// The default URL error codes to retry.    public static let defaultRetryableURLErrorCodes: Set<URLError.Code> = [// [Security] App Transport Security disallowed a connection because there is no secure network connection.        //   - [Disabled] ATS settings do not change at runtime.        // .appTransportSecurityRequiresSecureConnection,        // [System] An app or app extension attempted to connect to a background session that is already connected to a        // process.        //   - [Enabled] The other process could release the background session.        .backgroundSessionInUseByAnotherProcess,                                                                                   // [System] The shared container identifier of the URL session configuration is needed but has not been set.        //   - [Disabled] Cannot change at runtime.        // .backgroundSessionRequiresSharedContainer,        // [System] The app is suspended or exits while a background data task is processing.        //   - [Enabled] App can be foregrounded or launched to recover.        .backgroundSessionWasDisconnected,                                                                                   // [Network] The URL Loading system received bad data from the server.        //   - [Enabled] Server could return valid data when retrying.        .badServerResponse,                                                                                   // [Resource] A malformed URL prevented a URL request from being initiated.        //   - [Disabled] URL was most likely constructed incorrectly.        // .badURL,        // [System] A connection was attempted while a phone call is active on a network that does not support        // simultaneous phone and data communication (EDGE or GPRS).        //   - [Enabled] Phone call could be ended to allow request to recover.        .callIsActive,                                                                                   // [Client] An asynchronous load has been canceled.        //   - [Disabled] Request was cancelled by the client.        // .cancelled,        // [File System] A download task couldn’t close the downloaded file on disk.        //   - [Disabled] File system error is unlikely to recover with retry.        // .cannotCloseFile,        // [Network] An attempt to connect to a host failed.        //   - [Enabled] Server or DNS lookup could recover during retry.        .cannotConnectToHost,                                                                                   // [File System] A download task couldn’t create the downloaded file on disk because of an I/O failure.        //   - [Disabled] File system error is unlikely to recover with retry.        // .cannotCreateFile,        // [Data] Content data received during a connection request had an unknown content encoding.        //   - [Disabled] Server is unlikely to modify the content encoding during a retry.        // .cannotDecodeContentData,        // [Data] Content data received during a connection request could not be decoded for a known content encoding.        //   - [Disabled] Server is unlikely to modify the content encoding during a retry.        // .cannotDecodeRawData,        // [Network] The host name for a URL could not be resolved.        //   - [Enabled] Server or DNS lookup could recover during retry.        .cannotFindHost,                                                                                   // [Network] A request to load an item only from the cache could not be satisfied.        //   - [Enabled] Cache could be populated during a retry.        .cannotLoadFromNetwork,                                                                                   // [File System] A download task was unable to move a downloaded file on disk.        //   - [Disabled] File system error is unlikely to recover with retry.        // .cannotMoveFile,        // [File System] A download task was unable to open the downloaded file on disk.        //   - [Disabled] File system error is unlikely to recover with retry.        // .cannotOpenFile,        // [Data] A task could not parse a response.        //   - [Disabled] Invalid response is unlikely to recover with retry.        // .cannotParseResponse,        // [File System] A download task was unable to remove a downloaded file from disk.        //   - [Disabled] File system error is unlikely to recover with retry.        // .cannotRemoveFile,        // [File System] A download task was unable to write to the downloaded file on disk.        //   - [Disabled] File system error is unlikely to recover with retry.        // .cannotWriteToFile,        // [Security] A client certificate was rejected.        //   - [Disabled] Client certificate is unlikely to change with retry.        // .clientCertificateRejected,        // [Security] A client certificate was required to authenticate an SSL connection during a request.        //   - [Disabled] Client certificate is unlikely to be provided with retry.        // .clientCertificateRequired,        // [Data] The length of the resource data exceeds the maximum allowed.        //   - [Disabled] Resource will likely still exceed the length maximum on retry.        // .dataLengthExceedsMaximum,        // [System] The cellular network disallowed a connection.        //   - [Enabled] WiFi connection could be established during retry.        .dataNotAllowed,                                                                                   // [Network] The host address could not be found via DNS lookup.        //   - [Enabled] DNS lookup could succeed during retry.        .dnsLookupFailed,                                                                                   // [Data] A download task failed to decode an encoded file during the download.        //   - [Enabled] Server could correct the decoding issue with retry.        .downloadDecodingFailedMidStream,                                                                                   // [Data] A download task failed to decode an encoded file after downloading.        //   - [Enabled] Server could correct the decoding issue with retry.        .downloadDecodingFailedToComplete,                                                                                   // [File System] A file does not exist.        //   - [Disabled] File system error is unlikely to recover with retry.        // .fileDoesNotExist,        // [File System] A request for an FTP file resulted in the server responding that the file is not a plain file,        // but a directory.        //   - [Disabled] FTP directory is not likely to change to a file during a retry.        // .fileIsDirectory,        // [Network] A redirect loop has been detected or the threshold for number of allowable redirects has been        // exceeded (currently 16).        //   - [Disabled] The redirect loop is unlikely to be resolved within the retry window.        // .httpTooManyRedirects,        // [System] The attempted connection required activating a data context while roaming, but international roaming        // is disabled.        //   - [Enabled] WiFi connection could be established during retry.        .internationalRoamingOff,                                                                                   // [Connectivity] A client or server connection was severed in the middle of an in-progress load.        //   - [Enabled] A network connection could be established during retry.        .networkConnectionLost,                                                                                   // [File System] A resource couldn’t be read because of insufficient permissions.        //   - [Disabled] Permissions are unlikely to be granted during retry.        // .noPermissionsToReadFile,        // [Connectivity] A network resource was requested, but an internet connection has not been established and        // cannot be established automatically.        //   - [Enabled] A network connection could be established during retry.        .notConnectedToInternet,                                                                                   // [Resource] A redirect was specified by way of server response code, but the server did not accompany this        // code with a redirect URL.        //   - [Disabled] The redirect URL is unlikely to be supplied during a retry.        // .redirectToNonExistentLocation,        // [Client] A body stream is needed but the client did not provide one.        //   - [Disabled] The client will be unlikely to supply a body stream during retry.        // .requestBodyStreamExhausted,        // [Resource] A requested resource couldn’t be retrieved.        //   - [Disabled] The resource is unlikely to become available during the retry window.        // .resourceUnavailable,        // [Security] An attempt to establish a secure connection failed for reasons that can’t be expressed more        // specifically.        //   - [Enabled] The secure connection could be established during a retry given the lack of specificity        //     provided by the error.        .secureConnectionFailed,                                                                                   // [Security] A server certificate had a date which indicates it has expired, or is not yet valid.        //   - [Enabled] The server certificate could become valid within the retry window.        .serverCertificateHasBadDate,                                                                                   // [Security] A server certificate was not signed by any root server.        //   - [Disabled] The server certificate is unlikely to change during the retry window.        // .serverCertificateHasUnknownRoot,        // [Security] A server certificate is not yet valid.        //   - [Enabled] The server certificate could become valid within the retry window.        .serverCertificateNotYetValid,                                                                                   // [Security] A server certificate was signed by a root server that isn’t trusted.        //   - [Disabled] The server certificate is unlikely to become trusted within the retry window.        // .serverCertificateUntrusted,        // [Network] An asynchronous operation timed out.        //   - [Enabled] The request timed out for an unknown reason and should be retried.        .timedOut        // [System] The URL Loading System encountered an error that it can’t interpret.        //   - [Disabled] The error could not be interpreted and is unlikely to be recovered from during a retry.        // .unknown,        // [Resource] A properly formed URL couldn’t be handled by the framework.        //   - [Disabled] The URL is unlikely to change during a retry.        // .unsupportedURL,        // [Client] Authentication is required to access a resource.        //   - [Disabled] The user authentication is unlikely to be provided by retrying.        // .userAuthenticationRequired,        // [Client] An asynchronous request for authentication has been canceled by the user.        //   - [Disabled] The user cancelled authentication and explicitly took action to not retry.        // .userCancelledAuthentication,        // [Resource] A server reported that a URL has a non-zero content length, but terminated the network connection        // gracefully without sending any data.        //   - [Disabled] The server is unlikely to provide data during the retry window.        // .zeroByteResource,    ]    /// The total number of times the request is allowed to be retried.    public let retryLimit: UInt    /// The base of the exponential backoff policy (should always be greater than or equal to 2).    public let exponentialBackoffBase: UInt    /// The scale of the exponential backoff.    public let exponentialBackoffScale: Double    /// The HTTP methods that are allowed to be retried.    public let retryableHTTPMethods: Set<HTTPMethod>    /// The HTTP status codes that are automatically retried by the policy.    public let retryableHTTPStatusCodes: Set<Int>    /// The URL error codes that are automatically retried by the policy.    public let retryableURLErrorCodes: Set<URLError.Code>    /// Creates an `ExponentialBackoffRetryPolicy` from the specified parameters.    ///    /// - Parameters:    ///   - retryLimit:               The total number of times the request is allowed to be retried. `2` by default.    ///   - exponentialBackoffBase:   The base of the exponential backoff policy. `2` by default.    ///   - exponentialBackoffScale:  The scale of the exponential backoff. `0.5` by default.    ///   - retryableHTTPMethods:     The HTTP methods that are allowed to be retried.    ///                               `RetryPolicy.defaultRetryableHTTPMethods` by default.    ///   - retryableHTTPStatusCodes: The HTTP status codes that are automatically retried by the policy.    ///                               `RetryPolicy.defaultRetryableHTTPStatusCodes` by default.    ///   - retryableURLErrorCodes:   The URL error codes that are automatically retried by the policy.    ///                               `RetryPolicy.defaultRetryableURLErrorCodes` by default.    public init(retryLimit: UInt = RetryPolicy.defaultRetryLimit,                exponentialBackoffBase: UInt = RetryPolicy.defaultExponentialBackoffBase,                exponentialBackoffScale: Double = RetryPolicy.defaultExponentialBackoffScale,                retryableHTTPMethods: Set<HTTPMethod> = RetryPolicy.defaultRetryableHTTPMethods,                retryableHTTPStatusCodes: Set<Int> = RetryPolicy.defaultRetryableHTTPStatusCodes,                retryableURLErrorCodes: Set<URLError.Code> = RetryPolicy.defaultRetryableURLErrorCodes) {        precondition(exponentialBackoffBase >= 2, "The `exponentialBackoffBase` must be a minimum of 2.")        self.retryLimit = retryLimit        self.exponentialBackoffBase = exponentialBackoffBase        self.exponentialBackoffScale = exponentialBackoffScale        self.retryableHTTPMethods = retryableHTTPMethods        self.retryableHTTPStatusCodes = retryableHTTPStatusCodes        self.retryableURLErrorCodes = retryableURLErrorCodes    }    open func retry(_ request: Request,                    for session: Session,                    dueTo error: Error,                    completion: @escaping (RetryResult) -> Void) {        if            request.retryCount < retryLimit,            let httpMethod = request.request?.method,            retryableHTTPMethods.contains(httpMethod),            shouldRetry(response: request.response, error: error) {            let timeDelay = pow(Double(exponentialBackoffBase), Double(request.retryCount)) * exponentialBackoffScale            completion(.retryWithDelay(timeDelay))        } else {            completion(.doNotRetry)        }    }    private func shouldRetry(response: HTTPURLResponse?, error: Error) -> Bool {        if let statusCode = response?.statusCode, retryableHTTPStatusCodes.contains(statusCode) {            return true        } else {            let errorCode = (error as? URLError)?.code            let afErrorCode = (error.asAFError?.underlyingError as? URLError)?.code            if let code = errorCode ?? afErrorCode, retryableURLErrorCodes.contains(code) {                return true            } else {                return false            }        }    }}// MARK: -/// A retry policy that automatically retries idempotent requests for network connection lost errors. For more/// information about retrying network connection lost errors, please refer to Apple's/// [technical document](https://developer.apple.com/library/content/qa/qa1941/_index.html).open class ConnectionLostRetryPolicy: RetryPolicy {    /// Creates a `ConnectionLostRetryPolicy` instance from the specified parameters.    ///    /// - Parameters:    ///   - retryLimit:              The total number of times the request is allowed to be retried.    ///                              `RetryPolicy.defaultRetryLimit` by default.    ///   - exponentialBackoffBase:  The base of the exponential backoff policy.    ///                              `RetryPolicy.defaultExponentialBackoffBase` by default.    ///   - exponentialBackoffScale: The scale of the exponential backoff.    ///                              `RetryPolicy.defaultExponentialBackoffScale` by default.    ///   - retryableHTTPMethods:    The idempotent http methods to retry.    ///                              `RetryPolicy.defaultRetryableHTTPMethods` by default.    public init(retryLimit: UInt = RetryPolicy.defaultRetryLimit,                exponentialBackoffBase: UInt = RetryPolicy.defaultExponentialBackoffBase,                exponentialBackoffScale: Double = RetryPolicy.defaultExponentialBackoffScale,                retryableHTTPMethods: Set<HTTPMethod> = RetryPolicy.defaultRetryableHTTPMethods) {        super.init(retryLimit: retryLimit,                   exponentialBackoffBase: exponentialBackoffBase,                   exponentialBackoffScale: exponentialBackoffScale,                   retryableHTTPMethods: retryableHTTPMethods,                   retryableHTTPStatusCodes: [],                   retryableURLErrorCodes: [.networkConnectionLost])    }}
 |