You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
660 lines
24 KiB
660 lines
24 KiB
// |
|
// Request.swift |
|
// |
|
// Copyright (c) 2014 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 type that can inspect and optionally adapt a `URLRequest` in some manner if necessary. |
|
public protocol RequestAdapter { |
|
/// Inspects and adapts the specified `URLRequest` in some manner if necessary and returns the result. |
|
/// |
|
/// - parameter urlRequest: The URL request to adapt. |
|
/// |
|
/// - throws: An `Error` if the adaptation encounters an error. |
|
/// |
|
/// - returns: The adapted `URLRequest`. |
|
func adapt(_ urlRequest: URLRequest) throws -> URLRequest |
|
} |
|
|
|
// MARK: - |
|
|
|
/// A closure executed when the `RequestRetrier` determines whether a `Request` should be retried or not. |
|
public typealias RequestRetryCompletion = (_ shouldRetry: Bool, _ timeDelay: TimeInterval) -> Void |
|
|
|
/// A type that determines whether a request should be retried after being executed by the specified session manager |
|
/// and encountering an error. |
|
public protocol RequestRetrier { |
|
/// Determines whether the `Request` should be retried by calling the `completion` closure. |
|
/// |
|
/// This operation is fully asynchronous. Any amount of time can be taken to determine whether the request needs |
|
/// to be retried. The one requirement is that the completion closure is called to ensure the request is properly |
|
/// cleaned up after. |
|
/// |
|
/// - parameter manager: The session manager the request was executed on. |
|
/// - parameter request: The request that failed due to the encountered error. |
|
/// - parameter error: The error encountered when executing the request. |
|
/// - parameter completion: The completion closure to be executed when retry decision has been determined. |
|
func should(_ manager: SessionManager, retry request: Request, with error: Error, completion: @escaping RequestRetryCompletion) |
|
} |
|
|
|
// MARK: - |
|
|
|
protocol TaskConvertible { |
|
func task(session: URLSession, adapter: RequestAdapter?, queue: DispatchQueue) throws -> URLSessionTask |
|
} |
|
|
|
/// A dictionary of headers to apply to a `URLRequest`. |
|
public typealias HTTPHeaders = [String: String] |
|
|
|
// MARK: - |
|
|
|
/// Responsible for sending a request and receiving the response and associated data from the server, as well as |
|
/// managing its underlying `URLSessionTask`. |
|
open class Request { |
|
|
|
// MARK: Helper Types |
|
|
|
/// A closure executed when monitoring upload or download progress of a request. |
|
public typealias ProgressHandler = (Progress) -> Void |
|
|
|
enum RequestTask { |
|
case data(TaskConvertible?, URLSessionTask?) |
|
case download(TaskConvertible?, URLSessionTask?) |
|
case upload(TaskConvertible?, URLSessionTask?) |
|
case stream(TaskConvertible?, URLSessionTask?) |
|
} |
|
|
|
// MARK: Properties |
|
|
|
/// The delegate for the underlying task. |
|
open internal(set) var delegate: TaskDelegate { |
|
get { |
|
taskDelegateLock.lock() ; defer { taskDelegateLock.unlock() } |
|
return taskDelegate |
|
} |
|
set { |
|
taskDelegateLock.lock() ; defer { taskDelegateLock.unlock() } |
|
taskDelegate = newValue |
|
} |
|
} |
|
|
|
/// The underlying task. |
|
open var task: URLSessionTask? { return delegate.task } |
|
|
|
/// The session belonging to the underlying task. |
|
public let session: URLSession |
|
|
|
/// The request sent or to be sent to the server. |
|
open var request: URLRequest? { return task?.originalRequest } |
|
|
|
/// The response received from the server, if any. |
|
open var response: HTTPURLResponse? { return task?.response as? HTTPURLResponse } |
|
|
|
/// The number of times the request has been retried. |
|
open internal(set) var retryCount: UInt = 0 |
|
|
|
let originalTask: TaskConvertible? |
|
|
|
var startTime: CFAbsoluteTime? |
|
var endTime: CFAbsoluteTime? |
|
|
|
var validations: [() -> Void] = [] |
|
|
|
private var taskDelegate: TaskDelegate |
|
private var taskDelegateLock = NSLock() |
|
|
|
// MARK: Lifecycle |
|
|
|
init(session: URLSession, requestTask: RequestTask, error: Error? = nil) { |
|
self.session = session |
|
|
|
switch requestTask { |
|
case .data(let originalTask, let task): |
|
taskDelegate = DataTaskDelegate(task: task) |
|
self.originalTask = originalTask |
|
case .download(let originalTask, let task): |
|
taskDelegate = DownloadTaskDelegate(task: task) |
|
self.originalTask = originalTask |
|
case .upload(let originalTask, let task): |
|
taskDelegate = UploadTaskDelegate(task: task) |
|
self.originalTask = originalTask |
|
case .stream(let originalTask, let task): |
|
taskDelegate = TaskDelegate(task: task) |
|
self.originalTask = originalTask |
|
} |
|
|
|
delegate.error = error |
|
delegate.queue.addOperation { self.endTime = CFAbsoluteTimeGetCurrent() } |
|
} |
|
|
|
// MARK: Authentication |
|
|
|
/// Associates an HTTP Basic credential with the request. |
|
/// |
|
/// - parameter user: The user. |
|
/// - parameter password: The password. |
|
/// - parameter persistence: The URL credential persistence. `.ForSession` by default. |
|
/// |
|
/// - returns: The request. |
|
@discardableResult |
|
open func authenticate( |
|
user: String, |
|
password: String, |
|
persistence: URLCredential.Persistence = .forSession) |
|
-> Self |
|
{ |
|
let credential = URLCredential(user: user, password: password, persistence: persistence) |
|
return authenticate(usingCredential: credential) |
|
} |
|
|
|
/// Associates a specified credential with the request. |
|
/// |
|
/// - parameter credential: The credential. |
|
/// |
|
/// - returns: The request. |
|
@discardableResult |
|
open func authenticate(usingCredential credential: URLCredential) -> Self { |
|
delegate.credential = credential |
|
return self |
|
} |
|
|
|
/// Returns a base64 encoded basic authentication credential as an authorization header tuple. |
|
/// |
|
/// - parameter user: The user. |
|
/// - parameter password: The password. |
|
/// |
|
/// - returns: A tuple with Authorization header and credential value if encoding succeeds, `nil` otherwise. |
|
open class func authorizationHeader(user: String, password: String) -> (key: String, value: String)? { |
|
guard let data = "\(user):\(password)".data(using: .utf8) else { return nil } |
|
|
|
let credential = data.base64EncodedString(options: []) |
|
|
|
return (key: "Authorization", value: "Basic \(credential)") |
|
} |
|
|
|
// MARK: State |
|
|
|
/// Resumes the request. |
|
open func resume() { |
|
guard let task = task else { delegate.queue.isSuspended = false ; return } |
|
|
|
if startTime == nil { startTime = CFAbsoluteTimeGetCurrent() } |
|
|
|
task.resume() |
|
|
|
NotificationCenter.default.post( |
|
name: Notification.Name.Task.DidResume, |
|
object: self, |
|
userInfo: [Notification.Key.Task: task] |
|
) |
|
} |
|
|
|
/// Suspends the request. |
|
open func suspend() { |
|
guard let task = task else { return } |
|
|
|
task.suspend() |
|
|
|
NotificationCenter.default.post( |
|
name: Notification.Name.Task.DidSuspend, |
|
object: self, |
|
userInfo: [Notification.Key.Task: task] |
|
) |
|
} |
|
|
|
/// Cancels the request. |
|
open func cancel() { |
|
guard let task = task else { return } |
|
|
|
task.cancel() |
|
|
|
NotificationCenter.default.post( |
|
name: Notification.Name.Task.DidCancel, |
|
object: self, |
|
userInfo: [Notification.Key.Task: task] |
|
) |
|
} |
|
} |
|
|
|
// MARK: - CustomStringConvertible |
|
|
|
extension Request: CustomStringConvertible { |
|
/// The textual representation used when written to an output stream, which includes the HTTP method and URL, as |
|
/// well as the response status code if a response has been received. |
|
open var description: String { |
|
var components: [String] = [] |
|
|
|
if let HTTPMethod = request?.httpMethod { |
|
components.append(HTTPMethod) |
|
} |
|
|
|
if let urlString = request?.url?.absoluteString { |
|
components.append(urlString) |
|
} |
|
|
|
if let response = response { |
|
components.append("(\(response.statusCode))") |
|
} |
|
|
|
return components.joined(separator: " ") |
|
} |
|
} |
|
|
|
// MARK: - CustomDebugStringConvertible |
|
|
|
extension Request: CustomDebugStringConvertible { |
|
/// The textual representation used when written to an output stream, in the form of a cURL command. |
|
open var debugDescription: String { |
|
return cURLRepresentation() |
|
} |
|
|
|
func cURLRepresentation() -> String { |
|
var components = ["$ curl -v"] |
|
|
|
guard let request = self.request, |
|
let url = request.url, |
|
let host = url.host |
|
else { |
|
return "$ curl command could not be created" |
|
} |
|
|
|
if let httpMethod = request.httpMethod, httpMethod != "GET" { |
|
components.append("-X \(httpMethod)") |
|
} |
|
|
|
if let credentialStorage = self.session.configuration.urlCredentialStorage { |
|
let protectionSpace = URLProtectionSpace( |
|
host: host, |
|
port: url.port ?? 0, |
|
protocol: url.scheme, |
|
realm: host, |
|
authenticationMethod: NSURLAuthenticationMethodHTTPBasic |
|
) |
|
|
|
if let credentials = credentialStorage.credentials(for: protectionSpace)?.values { |
|
for credential in credentials { |
|
guard let user = credential.user, let password = credential.password else { continue } |
|
components.append("-u \(user):\(password)") |
|
} |
|
} else { |
|
if let credential = delegate.credential, let user = credential.user, let password = credential.password { |
|
components.append("-u \(user):\(password)") |
|
} |
|
} |
|
} |
|
|
|
if session.configuration.httpShouldSetCookies { |
|
if |
|
let cookieStorage = session.configuration.httpCookieStorage, |
|
let cookies = cookieStorage.cookies(for: url), !cookies.isEmpty |
|
{ |
|
let string = cookies.reduce("") { $0 + "\($1.name)=\($1.value);" } |
|
|
|
#if swift(>=3.2) |
|
components.append("-b \"\(string[..<string.index(before: string.endIndex)])\"") |
|
#else |
|
components.append("-b \"\(string.substring(to: string.characters.index(before: string.endIndex)))\"") |
|
#endif |
|
} |
|
} |
|
|
|
var headers: [AnyHashable: Any] = [:] |
|
|
|
session.configuration.httpAdditionalHeaders?.filter { $0.0 != AnyHashable("Cookie") } |
|
.forEach { headers[$0.0] = $0.1 } |
|
|
|
request.allHTTPHeaderFields?.filter { $0.0 != "Cookie" } |
|
.forEach { headers[$0.0] = $0.1 } |
|
|
|
components += headers.map { |
|
let escapedValue = String(describing: $0.value).replacingOccurrences(of: "\"", with: "\\\"") |
|
|
|
return "-H \"\($0.key): \(escapedValue)\"" |
|
} |
|
|
|
if let httpBodyData = request.httpBody, let httpBody = String(data: httpBodyData, encoding: .utf8) { |
|
var escapedBody = httpBody.replacingOccurrences(of: "\\\"", with: "\\\\\"") |
|
escapedBody = escapedBody.replacingOccurrences(of: "\"", with: "\\\"") |
|
|
|
components.append("-d \"\(escapedBody)\"") |
|
} |
|
|
|
components.append("\"\(url.absoluteString)\"") |
|
|
|
return components.joined(separator: " \\\n\t") |
|
} |
|
} |
|
|
|
// MARK: - |
|
|
|
/// Specific type of `Request` that manages an underlying `URLSessionDataTask`. |
|
open class DataRequest: Request { |
|
|
|
// MARK: Helper Types |
|
|
|
struct Requestable: TaskConvertible { |
|
let urlRequest: URLRequest |
|
|
|
func task(session: URLSession, adapter: RequestAdapter?, queue: DispatchQueue) throws -> URLSessionTask { |
|
do { |
|
let urlRequest = try self.urlRequest.adapt(using: adapter) |
|
return queue.sync { session.dataTask(with: urlRequest) } |
|
} catch { |
|
throw AdaptError(error: error) |
|
} |
|
} |
|
} |
|
|
|
// MARK: Properties |
|
|
|
/// The request sent or to be sent to the server. |
|
open override var request: URLRequest? { |
|
if let request = super.request { return request } |
|
if let requestable = originalTask as? Requestable { return requestable.urlRequest } |
|
|
|
return nil |
|
} |
|
|
|
/// The progress of fetching the response data from the server for the request. |
|
open var progress: Progress { return dataDelegate.progress } |
|
|
|
var dataDelegate: DataTaskDelegate { return delegate as! DataTaskDelegate } |
|
|
|
// MARK: Stream |
|
|
|
/// Sets a closure to be called periodically during the lifecycle of the request as data is read from the server. |
|
/// |
|
/// This closure returns the bytes most recently received from the server, not including data from previous calls. |
|
/// If this closure is set, data will only be available within this closure, and will not be saved elsewhere. It is |
|
/// also important to note that the server data in any `Response` object will be `nil`. |
|
/// |
|
/// - parameter closure: The code to be executed periodically during the lifecycle of the request. |
|
/// |
|
/// - returns: The request. |
|
@discardableResult |
|
open func stream(closure: ((Data) -> Void)? = nil) -> Self { |
|
dataDelegate.dataStream = closure |
|
return self |
|
} |
|
|
|
// MARK: Progress |
|
|
|
/// Sets a closure to be called periodically during the lifecycle of the `Request` as data is read from the server. |
|
/// |
|
/// - parameter queue: The dispatch queue to execute the closure on. |
|
/// - parameter closure: The code to be executed periodically as data is read from the server. |
|
/// |
|
/// - returns: The request. |
|
@discardableResult |
|
open func downloadProgress(queue: DispatchQueue = DispatchQueue.main, closure: @escaping ProgressHandler) -> Self { |
|
dataDelegate.progressHandler = (closure, queue) |
|
return self |
|
} |
|
} |
|
|
|
// MARK: - |
|
|
|
/// Specific type of `Request` that manages an underlying `URLSessionDownloadTask`. |
|
open class DownloadRequest: Request { |
|
|
|
// MARK: Helper Types |
|
|
|
/// A collection of options to be executed prior to moving a downloaded file from the temporary URL to the |
|
/// destination URL. |
|
public struct DownloadOptions: OptionSet { |
|
/// Returns the raw bitmask value of the option and satisfies the `RawRepresentable` protocol. |
|
public let rawValue: UInt |
|
|
|
/// A `DownloadOptions` flag that creates intermediate directories for the destination URL if specified. |
|
public static let createIntermediateDirectories = DownloadOptions(rawValue: 1 << 0) |
|
|
|
/// A `DownloadOptions` flag that removes a previous file from the destination URL if specified. |
|
public static let removePreviousFile = DownloadOptions(rawValue: 1 << 1) |
|
|
|
/// Creates a `DownloadFileDestinationOptions` instance with the specified raw value. |
|
/// |
|
/// - parameter rawValue: The raw bitmask value for the option. |
|
/// |
|
/// - returns: A new log level instance. |
|
public init(rawValue: UInt) { |
|
self.rawValue = rawValue |
|
} |
|
} |
|
|
|
/// A closure executed once a download request has successfully completed in order to determine where to move the |
|
/// temporary file written to during the download process. The closure takes two arguments: the temporary file URL |
|
/// and the URL response, and returns a two arguments: the file URL where the temporary file should be moved and |
|
/// the options defining how the file should be moved. |
|
public typealias DownloadFileDestination = ( |
|
_ temporaryURL: URL, |
|
_ response: HTTPURLResponse) |
|
-> (destinationURL: URL, options: DownloadOptions) |
|
|
|
enum Downloadable: TaskConvertible { |
|
case request(URLRequest) |
|
case resumeData(Data) |
|
|
|
func task(session: URLSession, adapter: RequestAdapter?, queue: DispatchQueue) throws -> URLSessionTask { |
|
do { |
|
let task: URLSessionTask |
|
|
|
switch self { |
|
case let .request(urlRequest): |
|
let urlRequest = try urlRequest.adapt(using: adapter) |
|
task = queue.sync { session.downloadTask(with: urlRequest) } |
|
case let .resumeData(resumeData): |
|
task = queue.sync { session.downloadTask(withResumeData: resumeData) } |
|
} |
|
|
|
return task |
|
} catch { |
|
throw AdaptError(error: error) |
|
} |
|
} |
|
} |
|
|
|
// MARK: Properties |
|
|
|
/// The request sent or to be sent to the server. |
|
open override var request: URLRequest? { |
|
if let request = super.request { return request } |
|
|
|
if let downloadable = originalTask as? Downloadable, case let .request(urlRequest) = downloadable { |
|
return urlRequest |
|
} |
|
|
|
return nil |
|
} |
|
|
|
/// The resume data of the underlying download task if available after a failure. |
|
open var resumeData: Data? { return downloadDelegate.resumeData } |
|
|
|
/// The progress of downloading the response data from the server for the request. |
|
open var progress: Progress { return downloadDelegate.progress } |
|
|
|
var downloadDelegate: DownloadTaskDelegate { return delegate as! DownloadTaskDelegate } |
|
|
|
// MARK: State |
|
|
|
/// Cancels the request. |
|
override open func cancel() { |
|
cancel(createResumeData: true) |
|
} |
|
|
|
/// Cancels the request. |
|
/// |
|
/// - parameter createResumeData: Determines whether resume data is created via the underlying download task or not. |
|
open func cancel(createResumeData: Bool) { |
|
if createResumeData { |
|
downloadDelegate.downloadTask.cancel { self.downloadDelegate.resumeData = $0 } |
|
} else { |
|
downloadDelegate.downloadTask.cancel() |
|
} |
|
|
|
NotificationCenter.default.post( |
|
name: Notification.Name.Task.DidCancel, |
|
object: self, |
|
userInfo: [Notification.Key.Task: task as Any] |
|
) |
|
} |
|
|
|
// MARK: Progress |
|
|
|
/// Sets a closure to be called periodically during the lifecycle of the `Request` as data is read from the server. |
|
/// |
|
/// - parameter queue: The dispatch queue to execute the closure on. |
|
/// - parameter closure: The code to be executed periodically as data is read from the server. |
|
/// |
|
/// - returns: The request. |
|
@discardableResult |
|
open func downloadProgress(queue: DispatchQueue = DispatchQueue.main, closure: @escaping ProgressHandler) -> Self { |
|
downloadDelegate.progressHandler = (closure, queue) |
|
return self |
|
} |
|
|
|
// MARK: Destination |
|
|
|
/// Creates a download file destination closure which uses the default file manager to move the temporary file to a |
|
/// file URL in the first available directory with the specified search path directory and search path domain mask. |
|
/// |
|
/// - parameter directory: The search path directory. `.DocumentDirectory` by default. |
|
/// - parameter domain: The search path domain mask. `.UserDomainMask` by default. |
|
/// |
|
/// - returns: A download file destination closure. |
|
open class func suggestedDownloadDestination( |
|
for directory: FileManager.SearchPathDirectory = .documentDirectory, |
|
in domain: FileManager.SearchPathDomainMask = .userDomainMask) |
|
-> DownloadFileDestination |
|
{ |
|
return { temporaryURL, response in |
|
let directoryURLs = FileManager.default.urls(for: directory, in: domain) |
|
|
|
if !directoryURLs.isEmpty { |
|
return (directoryURLs[0].appendingPathComponent(response.suggestedFilename!), []) |
|
} |
|
|
|
return (temporaryURL, []) |
|
} |
|
} |
|
} |
|
|
|
// MARK: - |
|
|
|
/// Specific type of `Request` that manages an underlying `URLSessionUploadTask`. |
|
open class UploadRequest: DataRequest { |
|
|
|
// MARK: Helper Types |
|
|
|
enum Uploadable: TaskConvertible { |
|
case data(Data, URLRequest) |
|
case file(URL, URLRequest) |
|
case stream(InputStream, URLRequest) |
|
|
|
func task(session: URLSession, adapter: RequestAdapter?, queue: DispatchQueue) throws -> URLSessionTask { |
|
do { |
|
let task: URLSessionTask |
|
|
|
switch self { |
|
case let .data(data, urlRequest): |
|
let urlRequest = try urlRequest.adapt(using: adapter) |
|
task = queue.sync { session.uploadTask(with: urlRequest, from: data) } |
|
case let .file(url, urlRequest): |
|
let urlRequest = try urlRequest.adapt(using: adapter) |
|
task = queue.sync { session.uploadTask(with: urlRequest, fromFile: url) } |
|
case let .stream(_, urlRequest): |
|
let urlRequest = try urlRequest.adapt(using: adapter) |
|
task = queue.sync { session.uploadTask(withStreamedRequest: urlRequest) } |
|
} |
|
|
|
return task |
|
} catch { |
|
throw AdaptError(error: error) |
|
} |
|
} |
|
} |
|
|
|
// MARK: Properties |
|
|
|
/// The request sent or to be sent to the server. |
|
open override var request: URLRequest? { |
|
if let request = super.request { return request } |
|
|
|
guard let uploadable = originalTask as? Uploadable else { return nil } |
|
|
|
switch uploadable { |
|
case .data(_, let urlRequest), .file(_, let urlRequest), .stream(_, let urlRequest): |
|
return urlRequest |
|
} |
|
} |
|
|
|
/// The progress of uploading the payload to the server for the upload request. |
|
open var uploadProgress: Progress { return uploadDelegate.uploadProgress } |
|
|
|
var uploadDelegate: UploadTaskDelegate { return delegate as! UploadTaskDelegate } |
|
|
|
// MARK: Upload Progress |
|
|
|
/// Sets a closure to be called periodically during the lifecycle of the `UploadRequest` as data is sent to |
|
/// the server. |
|
/// |
|
/// After the data is sent to the server, the `progress(queue:closure:)` APIs can be used to monitor the progress |
|
/// of data being read from the server. |
|
/// |
|
/// - parameter queue: The dispatch queue to execute the closure on. |
|
/// - parameter closure: The code to be executed periodically as data is sent to the server. |
|
/// |
|
/// - returns: The request. |
|
@discardableResult |
|
open func uploadProgress(queue: DispatchQueue = DispatchQueue.main, closure: @escaping ProgressHandler) -> Self { |
|
uploadDelegate.uploadProgressHandler = (closure, queue) |
|
return self |
|
} |
|
} |
|
|
|
// MARK: - |
|
|
|
#if !os(watchOS) |
|
|
|
/// Specific type of `Request` that manages an underlying `URLSessionStreamTask`. |
|
@available(iOS 9.0, macOS 10.11, tvOS 9.0, *) |
|
open class StreamRequest: Request { |
|
enum Streamable: TaskConvertible { |
|
case stream(hostName: String, port: Int) |
|
case netService(NetService) |
|
|
|
func task(session: URLSession, adapter: RequestAdapter?, queue: DispatchQueue) throws -> URLSessionTask { |
|
let task: URLSessionTask |
|
|
|
switch self { |
|
case let .stream(hostName, port): |
|
task = queue.sync { session.streamTask(withHostName: hostName, port: port) } |
|
case let .netService(netService): |
|
task = queue.sync { session.streamTask(with: netService) } |
|
} |
|
|
|
return task |
|
} |
|
} |
|
} |
|
|
|
#endif
|
|
|