Skip to content

Commit 5ce7377

Browse files
authored
Add HTTPClientReuqest.Prepared (#511)
* add HTTPClientReuqest.Prepared * make `prepared()` an init of `Prepared` * make all stored properties of `Prepared` `var`s
1 parent a956e7b commit 5ce7377

6 files changed

+148
-24
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,70 @@
1+
//===----------------------------------------------------------------------===//
2+
//
3+
// This source file is part of the AsyncHTTPClient open source project
4+
//
5+
// Copyright (c) 2021 Apple Inc. and the AsyncHTTPClient project authors
6+
// Licensed under Apache License v2.0
7+
//
8+
// See LICENSE.txt for license information
9+
// See CONTRIBUTORS.txt for the list of AsyncHTTPClient project authors
10+
//
11+
// SPDX-License-Identifier: Apache-2.0
12+
//
13+
//===----------------------------------------------------------------------===//
14+
15+
#if compiler(>=5.5) && canImport(_Concurrency)
16+
import NIOHTTP1
17+
18+
@available(macOS 12.0, iOS 15.0, watchOS 8.0, tvOS 15.0, *)
19+
extension HTTPClientRequest {
20+
struct Prepared {
21+
var poolKey: ConnectionPool.Key
22+
var requestFramingMetadata: RequestFramingMetadata
23+
var head: HTTPRequestHead
24+
var body: Body?
25+
}
26+
}
27+
28+
@available(macOS 12.0, iOS 15.0, watchOS 8.0, tvOS 15.0, *)
29+
extension HTTPClientRequest.Prepared {
30+
init(_ request: HTTPClientRequest) throws {
31+
let url = try DeconstructedURL(url: request.url)
32+
33+
var headers = request.headers
34+
headers.addHostIfNeeded(for: url)
35+
let metadata = try headers.validateAndSetTransportFraming(
36+
method: request.method,
37+
bodyLength: .init(request.body)
38+
)
39+
40+
self.init(
41+
poolKey: .init(url: url, tlsConfiguration: nil),
42+
requestFramingMetadata: metadata,
43+
head: .init(
44+
version: .http1_1,
45+
method: request.method,
46+
uri: url.uri,
47+
headers: headers
48+
),
49+
body: request.body
50+
)
51+
}
52+
}
53+
54+
@available(macOS 12.0, iOS 15.0, watchOS 8.0, tvOS 15.0, *)
55+
extension RequestBodyLength {
56+
init(_ body: HTTPClientRequest.Body?) {
57+
switch body?.mode {
58+
case .none:
59+
self = .fixed(length: 0)
60+
case .byteBuffer(let buffer):
61+
self = .fixed(length: buffer.readableBytes)
62+
case .sequence(nil, _), .asyncSequence(nil, _):
63+
self = .dynamic
64+
case .sequence(.some(let length), _), .asyncSequence(.some(let length), _):
65+
self = .fixed(length: length)
66+
}
67+
}
68+
}
69+
70+
#endif

Sources/AsyncHTTPClient/ConnectionPool.swift

+29-6
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,8 @@
1212
//
1313
//===----------------------------------------------------------------------===//
1414

15+
import NIOSSL
16+
1517
enum ConnectionPool {
1618
/// Used by the `ConnectionPool` to index its `HTTP1ConnectionProvider`s
1719
///
@@ -23,12 +25,14 @@ enum ConnectionPool {
2325
var connectionTarget: ConnectionTarget
2426
private var tlsConfiguration: BestEffortHashableTLSConfiguration?
2527

26-
init(_ request: HTTPClient.Request) {
27-
self.scheme = request.deconstructedURL.scheme
28-
self.connectionTarget = request.deconstructedURL.connectionTarget
29-
if let tls = request.tlsConfiguration {
30-
self.tlsConfiguration = BestEffortHashableTLSConfiguration(wrapping: tls)
31-
}
28+
init(
29+
scheme: Scheme,
30+
connectionTarget: ConnectionTarget,
31+
tlsConfiguration: BestEffortHashableTLSConfiguration? = nil
32+
) {
33+
self.scheme = scheme
34+
self.connectionTarget = connectionTarget
35+
self.tlsConfiguration = tlsConfiguration
3236
}
3337

3438
var description: String {
@@ -48,3 +52,22 @@ enum ConnectionPool {
4852
}
4953
}
5054
}
55+
56+
extension ConnectionPool.Key {
57+
init(url: DeconstructedURL, tlsConfiguration: TLSConfiguration?) {
58+
self.init(
59+
scheme: url.scheme,
60+
connectionTarget: url.connectionTarget,
61+
tlsConfiguration: tlsConfiguration.map {
62+
BestEffortHashableTLSConfiguration(wrapping: $0)
63+
}
64+
)
65+
}
66+
67+
init(_ request: HTTPClient.Request) {
68+
self.init(
69+
url: request.deconstructedURL,
70+
tlsConfiguration: request.tlsConfiguration
71+
)
72+
}
73+
}

Sources/AsyncHTTPClient/ConnectionTarget.swift

+22
Original file line numberDiff line numberDiff line change
@@ -40,3 +40,25 @@ enum ConnectionTarget: Equatable, Hashable {
4040
}
4141
}
4242
}
43+
44+
extension ConnectionTarget {
45+
/// The host name which will be send as an HTTP `Host` header.
46+
/// Only returns nil if the `self` is a `unixSocket`.
47+
var host: String? {
48+
switch self {
49+
case .ipAddress(let serialization, _): return serialization
50+
case .domain(let name, _): return name
51+
case .unixSocket: return nil
52+
}
53+
}
54+
55+
/// The host name which will be send as an HTTP host header.
56+
/// Only returns nil if the `self` is a `unixSocket`.
57+
var port: Int? {
58+
switch self {
59+
case .ipAddress(_, let address): return address.port!
60+
case .domain(_, let port): return port
61+
case .unixSocket: return nil
62+
}
63+
}
64+
}

Sources/AsyncHTTPClient/DeconstructedURL.swift

+7
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,13 @@ struct DeconstructedURL {
3131
}
3232

3333
extension DeconstructedURL {
34+
init(url: String) throws {
35+
guard let url = URL(string: url) else {
36+
throw HTTPClientError.invalidURL
37+
}
38+
try self.init(url: url)
39+
}
40+
3441
init(url: URL) throws {
3542
guard let schemeString = url.scheme else {
3643
throw HTTPClientError.emptyScheme

Sources/AsyncHTTPClient/HTTPHandler.swift

+3-18
Original file line numberDiff line numberDiff line change
@@ -200,20 +200,12 @@ extension HTTPClient {
200200

201201
/// Remote host, resolved from `URL`.
202202
public var host: String {
203-
switch self.deconstructedURL.connectionTarget {
204-
case .ipAddress(let serialization, _): return serialization
205-
case .domain(let name, _): return name
206-
case .unixSocket: return ""
207-
}
203+
self.deconstructedURL.connectionTarget.host ?? ""
208204
}
209205

210206
/// Resolved port.
211207
public var port: Int {
212-
switch self.deconstructedURL.connectionTarget {
213-
case .ipAddress(_, let address): return address.port!
214-
case .domain(_, let port): return port
215-
case .unixSocket: return self.deconstructedURL.scheme.defaultPort
216-
}
208+
self.deconstructedURL.connectionTarget.port ?? self.deconstructedURL.scheme.defaultPort
217209
}
218210

219211
/// Whether request will be executed using secure socket.
@@ -227,14 +219,7 @@ extension HTTPClient {
227219
headers: self.headers
228220
)
229221

230-
if !head.headers.contains(name: "host") {
231-
let port = self.port
232-
var host = self.host
233-
if !(port == 80 && self.deconstructedURL.scheme == .http), !(port == 443 && self.deconstructedURL.scheme == .https) {
234-
host += ":\(port)"
235-
}
236-
head.headers.add(name: "host", value: host)
237-
}
222+
head.headers.addHostIfNeeded(for: self.deconstructedURL)
238223

239224
let metadata = try head.headers.validateAndSetTransportFraming(method: self.method, bodyLength: .init(self.body))
240225

Sources/AsyncHTTPClient/RequestValidation.swift

+17
Original file line numberDiff line numberDiff line change
@@ -110,3 +110,20 @@ extension HTTPHeaders {
110110
}
111111
}
112112
}
113+
114+
extension HTTPHeaders {
115+
mutating func addHostIfNeeded(for url: DeconstructedURL) {
116+
// if no host header was set, let's use the url host
117+
guard !self.contains(name: "host"),
118+
var host = url.connectionTarget.host
119+
else {
120+
return
121+
}
122+
// if the request uses a non-default port, we need to add it after the host
123+
if let port = url.connectionTarget.port,
124+
port != url.scheme.defaultPort {
125+
host += ":\(port)"
126+
}
127+
self.add(name: "host", value: host)
128+
}
129+
}

0 commit comments

Comments
 (0)