| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337 |
- // Alamofire.swift
- //
- // Copyright (c) 2014–2015 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
- extension Manager {
- private enum Uploadable {
- case Data(NSURLRequest, NSData)
- case File(NSURLRequest, NSURL)
- case Stream(NSURLRequest, NSInputStream)
- }
- private func upload(uploadable: Uploadable) -> Request {
- var uploadTask: NSURLSessionUploadTask!
- var HTTPBodyStream: NSInputStream?
- switch uploadable {
- case .Data(let request, let data):
- dispatch_sync(queue) {
- uploadTask = self.session.uploadTaskWithRequest(request, fromData: data)
- }
- case .File(let request, let fileURL):
- dispatch_sync(queue) {
- uploadTask = self.session.uploadTaskWithRequest(request, fromFile: fileURL)
- }
- case .Stream(let request, var stream):
- dispatch_sync(queue) {
- uploadTask = self.session.uploadTaskWithStreamedRequest(request)
- }
- HTTPBodyStream = stream
- }
- let request = Request(session: session, task: uploadTask)
- if HTTPBodyStream != nil {
- request.delegate.taskNeedNewBodyStream = { _, _ in
- return HTTPBodyStream
- }
- }
- delegate[request.delegate.task] = request.delegate
- if startRequestsImmediately {
- request.resume()
- }
- return request
- }
- // MARK: File
- /**
- Creates a request for uploading a file to the specified URL request.
- If `startRequestsImmediately` is `true`, the request will have `resume()` called before being returned.
- :param: URLRequest The URL request
- :param: file The file to upload
- :returns: The created upload request.
- */
- public func upload(URLRequest: URLRequestConvertible, file: NSURL) -> Request {
- return upload(.File(URLRequest.URLRequest, file))
- }
- /**
- Creates a request for uploading a file to the specified URL request.
- If `startRequestsImmediately` is `true`, the request will have `resume()` called before being returned.
- :param: method The HTTP method.
- :param: URLString The URL string.
- :param: file The file to upload
- :returns: The created upload request.
- */
- public func upload(method: Method, _ URLString: URLStringConvertible, file: NSURL) -> Request {
- return upload(URLRequest(method, URLString), file: file)
- }
- // MARK: Data
- /**
- Creates a request for uploading data to the specified URL request.
- If `startRequestsImmediately` is `true`, the request will have `resume()` called before being returned.
- :param: URLRequest The URL request
- :param: data The data to upload
- :returns: The created upload request.
- */
- public func upload(URLRequest: URLRequestConvertible, data: NSData) -> Request {
- return upload(.Data(URLRequest.URLRequest, data))
- }
- /**
- Creates a request for uploading data to the specified URL request.
- If `startRequestsImmediately` is `true`, the request will have `resume()` called before being returned.
- :param: method The HTTP method.
- :param: URLString The URL string.
- :param: data The data to upload
- :returns: The created upload request.
- */
- public func upload(method: Method, _ URLString: URLStringConvertible, data: NSData) -> Request {
- return upload(URLRequest(method, URLString), data: data)
- }
- // MARK: Stream
- /**
- Creates a request for uploading a stream to the specified URL request.
- If `startRequestsImmediately` is `true`, the request will have `resume()` called before being returned.
- :param: URLRequest The URL request
- :param: stream The stream to upload
- :returns: The created upload request.
- */
- public func upload(URLRequest: URLRequestConvertible, stream: NSInputStream) -> Request {
- return upload(.Stream(URLRequest.URLRequest, stream))
- }
- /**
- Creates a request for uploading a stream to the specified URL request.
- If `startRequestsImmediately` is `true`, the request will have `resume()` called before being returned.
- :param: method The HTTP method.
- :param: URLString The URL string.
- :param: stream The stream to upload.
- :returns: The created upload request.
- */
- public func upload(method: Method, _ URLString: URLStringConvertible, stream: NSInputStream) -> Request {
- return upload(URLRequest(method, URLString), stream: stream)
- }
- // MARK: MultipartFormData
- /// Default memory threshold used when encoding `MultipartFormData`.
- public static let MultipartFormDataEncodingMemoryThreshold: UInt64 = 10 * 1024 * 1024
- /**
- Defines whether the `MultipartFormData` encoding was successful and contains result of the encoding as
- associated values.
- - Success: Represents a successful `MultipartFormData` encoding and contains the new `Request` along with
- streaming information.
- - Failure: Used to represent a failure in the `MultipartFormData` encoding and also contains the encoding
- error.
- */
- public enum MultipartFormDataEncodingResult {
- case Success(request: Request, streamingFromDisk: Bool, streamFileURL: NSURL?)
- case Failure(NSError)
- }
- /**
- Encodes the `MultipartFormData` and creates a request to upload the result to the specified URL request.
- It is important to understand the memory implications of uploading `MultipartFormData`. If the cummulative
- payload is small, encoding the data in-memory and directly uploading to a server is the by far the most
- efficient approach. However, if the payload is too large, encoding the data in-memory could cause your app to
- be terminated. Larger payloads must first be written to disk using input and output streams to keep the memory
- footprint low, then the data can be uploaded as a stream from the resulting file. Streaming from disk MUST be
- used for larger payloads such as video content.
- The `encodingMemoryThreshold` parameter allows Alamofire to automatically determine whether to encode in-memory
- or stream from disk. If the content length of the `MultipartFormData` is below the `encodingMemoryThreshold`,
- encoding takes place in-memory. If the content length exceeds the threshold, the data is streamed to disk
- during the encoding process. Then the result is uploaded as data or as a stream depending on which encoding
- technique was used.
- If `startRequestsImmediately` is `true`, the request will have `resume()` called before being returned.
- :param: method The HTTP method.
- :param: URLString The URL string.
- :param: multipartFormData The closure used to append body parts to the `MultipartFormData`.
- :param: encodingMemoryThreshold The encoding memory threshold in bytes. `MultipartFormDataEncodingMemoryThreshold`
- by default.
- :param: encodingCompletion The closure called when the `MultipartFormData` encoding is complete.
- */
- public func upload(
- method: Method,
- _ URLString: URLStringConvertible,
- multipartFormData: MultipartFormData -> Void,
- encodingMemoryThreshold: UInt64 = Manager.MultipartFormDataEncodingMemoryThreshold,
- encodingCompletion: (MultipartFormDataEncodingResult -> Void)?)
- {
- let urlRequest = URLRequest(method, URLString)
- return upload(
- urlRequest,
- multipartFormData: multipartFormData,
- encodingMemoryThreshold: encodingMemoryThreshold,
- encodingCompletion: encodingCompletion
- )
- }
- /**
- Encodes the `MultipartFormData` and creates a request to upload the result to the specified URL request.
- It is important to understand the memory implications of uploading `MultipartFormData`. If the cummulative
- payload is small, encoding the data in-memory and directly uploading to a server is the by far the most
- efficient approach. However, if the payload is too large, encoding the data in-memory could cause your app to
- be terminated. Larger payloads must first be written to disk using input and output streams to keep the memory
- footprint low, then the data can be uploaded as a stream from the resulting file. Streaming from disk MUST be
- used for larger payloads such as video content.
- The `encodingMemoryThreshold` parameter allows Alamofire to automatically determine whether to encode in-memory
- or stream from disk. If the content length of the `MultipartFormData` is below the `encodingMemoryThreshold`,
- encoding takes place in-memory. If the content length exceeds the threshold, the data is streamed to disk
- during the encoding process. Then the result is uploaded as data or as a stream depending on which encoding
- technique was used.
- If `startRequestsImmediately` is `true`, the request will have `resume()` called before being returned.
- :param: URLRequest The URL request.
- :param: multipartFormData The closure used to append body parts to the `MultipartFormData`.
- :param: encodingMemoryThreshold The encoding memory threshold in bytes. `MultipartFormDataEncodingMemoryThreshold`
- by default.
- :param: encodingCompletion The closure called when the `MultipartFormData` encoding is complete.
- */
- public func upload(
- URLRequest: URLRequestConvertible,
- multipartFormData: MultipartFormData -> Void,
- encodingMemoryThreshold: UInt64 = Manager.MultipartFormDataEncodingMemoryThreshold,
- encodingCompletion: (MultipartFormDataEncodingResult -> Void)?)
- {
- dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0)) {
- let formData = MultipartFormData()
- multipartFormData(formData)
- let URLRequestWithContentType = URLRequest.URLRequest.mutableCopy() as! NSMutableURLRequest
- URLRequestWithContentType.setValue(formData.contentType, forHTTPHeaderField: "Content-Type")
- if formData.contentLength < encodingMemoryThreshold {
- let encodingResult = formData.encode()
- dispatch_async(dispatch_get_main_queue()) {
- switch encodingResult {
- case .Success(let data):
- let encodingResult = MultipartFormDataEncodingResult.Success(
- request: self.upload(URLRequestWithContentType, data: data),
- streamingFromDisk: false,
- streamFileURL: nil
- )
- encodingCompletion?(encodingResult)
- case .Failure(let error):
- encodingCompletion?(.Failure(error))
- }
- }
- } else {
- let fileManager = NSFileManager.defaultManager()
- let tempDirectoryURL = NSURL(fileURLWithPath: NSTemporaryDirectory())!
- let directoryURL = tempDirectoryURL.URLByAppendingPathComponent("com.alamofire.manager/multipart.form.data")
- let fileName = NSUUID().UUIDString
- let fileURL = directoryURL.URLByAppendingPathComponent(fileName)
- var error: NSError?
- if fileManager.createDirectoryAtURL(directoryURL, withIntermediateDirectories: true, attributes: nil, error: &error) {
- formData.writeEncodedDataToDisk(fileURL) { error in
- dispatch_async(dispatch_get_main_queue()) {
- if let error = error {
- encodingCompletion?(.Failure(error))
- } else {
- let encodingResult = MultipartFormDataEncodingResult.Success(
- request: self.upload(URLRequestWithContentType, file: fileURL),
- streamingFromDisk: true,
- streamFileURL: fileURL
- )
- encodingCompletion?(encodingResult)
- }
- }
- }
- } else {
- dispatch_async(dispatch_get_main_queue()) {
- encodingCompletion?(.Failure(error!))
- }
- }
- }
- }
- }
- }
- // MARK: -
- extension Request {
- // MARK: - UploadTaskDelegate
- class UploadTaskDelegate: DataTaskDelegate {
- var uploadTask: NSURLSessionUploadTask? { return task as? NSURLSessionUploadTask }
- var uploadProgress: ((Int64, Int64, Int64) -> Void)!
- // MARK: - NSURLSessionTaskDelegate
- // MARK: Override Closures
- var taskDidSendBodyData: ((NSURLSession, NSURLSessionTask, Int64, Int64, Int64) -> Void)?
- // MARK: Delegate Methods
- func URLSession(session: NSURLSession, task: NSURLSessionTask, didSendBodyData bytesSent: Int64, totalBytesSent: Int64, totalBytesExpectedToSend: Int64) {
- if let taskDidSendBodyData = self.taskDidSendBodyData {
- taskDidSendBodyData(session, task, bytesSent, totalBytesSent, totalBytesExpectedToSend)
- } else {
- progress.totalUnitCount = totalBytesExpectedToSend
- progress.completedUnitCount = totalBytesSent
- uploadProgress?(bytesSent, totalBytesSent, totalBytesExpectedToSend)
- }
- }
- }
- }
|