AutorÃa | Ultima modificación | Ver Log |
//// ParameterEncoding.swift//// Copyright (c) 2014-2018 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 dictionary of parameters to apply to a `URLRequest`.public typealias Parameters = [String: Any]/// A type used to define how a set of parameters are applied to a `URLRequest`.public protocol ParameterEncoding {/// Creates a `URLRequest` by encoding parameters and applying them on the passed request.////// - Parameters:/// - urlRequest: `URLRequestConvertible` value onto which parameters will be encoded./// - parameters: `Parameters` to encode onto the request.////// - Returns: The encoded `URLRequest`./// - Throws: Any `Error` produced during parameter encoding.func encode(_ urlRequest: URLRequestConvertible, with parameters: Parameters?) throws -> URLRequest}// MARK: -/// Creates a url-encoded query string to be set as or appended to any existing URL query string or set as the HTTP/// body of the URL request. Whether the query string is set or appended to any existing URL query string or set as/// the HTTP body depends on the destination of the encoding.////// The `Content-Type` HTTP header field of an encoded request with HTTP body is set to/// `application/x-www-form-urlencoded; charset=utf-8`.////// There is no published specification for how to encode collection types. By default the convention of appending/// `[]` to the key for array values (`foo[]=1&foo[]=2`), and appending the key surrounded by square brackets for/// nested dictionary values (`foo[bar]=baz`) is used. Optionally, `ArrayEncoding` can be used to omit the/// square brackets appended to array keys.////// `BoolEncoding` can be used to configure how boolean values are encoded. The default behavior is to encode/// `true` as 1 and `false` as 0.public struct URLEncoding: ParameterEncoding {// MARK: Helper Types/// Defines whether the url-encoded query string is applied to the existing query string or HTTP body of the/// resulting URL request.public enum Destination {/// Applies encoded query string result to existing query string for `GET`, `HEAD` and `DELETE` requests and/// sets as the HTTP body for requests with any other HTTP method.case methodDependent/// Sets or appends encoded query string result to existing query string.case queryString/// Sets encoded query string result as the HTTP body of the URL request.case httpBodyfunc encodesParametersInURL(for method: HTTPMethod) -> Bool {switch self {case .methodDependent: return [.get, .head, .delete].contains(method)case .queryString: return truecase .httpBody: return false}}}/// Configures how `Array` parameters are encoded.public enum ArrayEncoding {/// An empty set of square brackets is appended to the key for every value. This is the default behavior.case brackets/// No brackets are appended. The key is encoded as is.case noBrackets/// Brackets containing the item index are appended. This matches the jQuery and Node.js behavior.case indexInBracketsfunc encode(key: String, atIndex index: Int) -> String {switch self {case .brackets:return "\(key)[]"case .noBrackets:return keycase .indexInBrackets:return "\(key)[\(index)]"}}}/// Configures how `Bool` parameters are encoded.public enum BoolEncoding {/// Encode `true` as `1` and `false` as `0`. This is the default behavior.case numeric/// Encode `true` and `false` as string literals.case literalfunc encode(value: Bool) -> String {switch self {case .numeric:return value ? "1" : "0"case .literal:return value ? "true" : "false"}}}// MARK: Properties/// Returns a default `URLEncoding` instance with a `.methodDependent` destination.public static var `default`: URLEncoding { URLEncoding() }/// Returns a `URLEncoding` instance with a `.queryString` destination.public static var queryString: URLEncoding { URLEncoding(destination: .queryString) }/// Returns a `URLEncoding` instance with an `.httpBody` destination.public static var httpBody: URLEncoding { URLEncoding(destination: .httpBody) }/// The destination defining where the encoded query string is to be applied to the URL request.public let destination: Destination/// The encoding to use for `Array` parameters.public let arrayEncoding: ArrayEncoding/// The encoding to use for `Bool` parameters.public let boolEncoding: BoolEncoding// MARK: Initialization/// Creates an instance using the specified parameters.////// - Parameters:/// - destination: `Destination` defining where the encoded query string will be applied. `.methodDependent` by/// default./// - arrayEncoding: `ArrayEncoding` to use. `.brackets` by default./// - boolEncoding: `BoolEncoding` to use. `.numeric` by default.public init(destination: Destination = .methodDependent,arrayEncoding: ArrayEncoding = .brackets,boolEncoding: BoolEncoding = .numeric) {self.destination = destinationself.arrayEncoding = arrayEncodingself.boolEncoding = boolEncoding}// MARK: Encodingpublic func encode(_ urlRequest: URLRequestConvertible, with parameters: Parameters?) throws -> URLRequest {var urlRequest = try urlRequest.asURLRequest()guard let parameters = parameters else { return urlRequest }if let method = urlRequest.method, destination.encodesParametersInURL(for: method) {guard let url = urlRequest.url else {throw AFError.parameterEncodingFailed(reason: .missingURL)}if var urlComponents = URLComponents(url: url, resolvingAgainstBaseURL: false), !parameters.isEmpty {let percentEncodedQuery = (urlComponents.percentEncodedQuery.map { $0 + "&" } ?? "") + query(parameters)urlComponents.percentEncodedQuery = percentEncodedQueryurlRequest.url = urlComponents.url}} else {if urlRequest.headers["Content-Type"] == nil {urlRequest.headers.update(.contentType("application/x-www-form-urlencoded; charset=utf-8"))}urlRequest.httpBody = Data(query(parameters).utf8)}return urlRequest}/// Creates a percent-escaped, URL encoded query string components from the given key-value pair recursively.////// - Parameters:/// - key: Key of the query component./// - value: Value of the query component.////// - Returns: The percent-escaped, URL encoded query string components.public func queryComponents(fromKey key: String, value: Any) -> [(String, String)] {var components: [(String, String)] = []switch value {case let dictionary as [String: Any]:for (nestedKey, value) in dictionary {components += queryComponents(fromKey: "\(key)[\(nestedKey)]", value: value)}case let array as [Any]:for (index, value) in array.enumerated() {components += queryComponents(fromKey: arrayEncoding.encode(key: key, atIndex: index), value: value)}case let number as NSNumber:if number.isBool {components.append((escape(key), escape(boolEncoding.encode(value: number.boolValue))))} else {components.append((escape(key), escape("\(number)")))}case let bool as Bool:components.append((escape(key), escape(boolEncoding.encode(value: bool))))default:components.append((escape(key), escape("\(value)")))}return components}/// Creates a percent-escaped string following RFC 3986 for a query string key or value.////// - Parameter string: `String` to be percent-escaped.////// - Returns: The percent-escaped `String`.public func escape(_ string: String) -> String {string.addingPercentEncoding(withAllowedCharacters: .afURLQueryAllowed) ?? string}private func query(_ parameters: [String: Any]) -> String {var components: [(String, String)] = []for key in parameters.keys.sorted(by: <) {let value = parameters[key]!components += queryComponents(fromKey: key, value: value)}return components.map { "\($0)=\($1)" }.joined(separator: "&")}}// MARK: -/// Uses `JSONSerialization` to create a JSON representation of the parameters object, which is set as the body of the/// request. The `Content-Type` HTTP header field of an encoded request is set to `application/json`.public struct JSONEncoding: ParameterEncoding {// MARK: Properties/// Returns a `JSONEncoding` instance with default writing options.public static var `default`: JSONEncoding { JSONEncoding() }/// Returns a `JSONEncoding` instance with `.prettyPrinted` writing options.public static var prettyPrinted: JSONEncoding { JSONEncoding(options: .prettyPrinted) }/// The options for writing the parameters as JSON data.public let options: JSONSerialization.WritingOptions// MARK: Initialization/// Creates an instance using the specified `WritingOptions`.////// - Parameter options: `JSONSerialization.WritingOptions` to use.public init(options: JSONSerialization.WritingOptions = []) {self.options = options}// MARK: Encodingpublic func encode(_ urlRequest: URLRequestConvertible, with parameters: Parameters?) throws -> URLRequest {var urlRequest = try urlRequest.asURLRequest()guard let parameters = parameters else { return urlRequest }do {let data = try JSONSerialization.data(withJSONObject: parameters, options: options)if urlRequest.headers["Content-Type"] == nil {urlRequest.headers.update(.contentType("application/json"))}urlRequest.httpBody = data} catch {throw AFError.parameterEncodingFailed(reason: .jsonEncodingFailed(error: error))}return urlRequest}/// Encodes any JSON compatible object into a `URLRequest`.////// - Parameters:/// - urlRequest: `URLRequestConvertible` value into which the object will be encoded./// - jsonObject: `Any` value (must be JSON compatible` to be encoded into the `URLRequest`. `nil` by default.////// - Returns: The encoded `URLRequest`./// - Throws: Any `Error` produced during encoding.public func encode(_ urlRequest: URLRequestConvertible, withJSONObject jsonObject: Any? = nil) throws -> URLRequest {var urlRequest = try urlRequest.asURLRequest()guard let jsonObject = jsonObject else { return urlRequest }do {let data = try JSONSerialization.data(withJSONObject: jsonObject, options: options)if urlRequest.headers["Content-Type"] == nil {urlRequest.headers.update(.contentType("application/json"))}urlRequest.httpBody = data} catch {throw AFError.parameterEncodingFailed(reason: .jsonEncodingFailed(error: error))}return urlRequest}}// MARK: -extension NSNumber {fileprivate var isBool: Bool {// Use Obj-C type encoding to check whether the underlying type is a `Bool`, as it's guaranteed as part of// swift-corelibs-foundation, per [this discussion on the Swift forums](https://forums.swift.org/t/alamofire-on-linux-possible-but-not-release-ready/34553/22).String(cString: objCType) == "c"}}