Skip to content

Commit fe3589d

Browse files
Added support for the new http+unix and https+unix schemes
Motivation: We can now not only make connections over sockets using the existing unix: baseURL approach, but also by %-escaping the socket path as the URL's hostname, and using the http+unix: or https+unix: schemes for plain HTTP or HTTP over TLS support. Modifications: Added references to the new url schemes where necessary, aand added a missingSocketPath error for when the socket path is not encoded as the hostname. Result: The tests now pass, and connections can be made over non-special URLs, supporting TLS if necessary.
1 parent d27bda2 commit fe3589d

File tree

3 files changed

+25
-4
lines changed

3 files changed

+25
-4
lines changed

Sources/AsyncHTTPClient/ConnectionPool.swift

+8-2
Original file line numberDiff line numberDiff line change
@@ -113,6 +113,10 @@ final class ConnectionPool {
113113
self.scheme = .https
114114
case "unix":
115115
self.scheme = .unix
116+
case "http+unix":
117+
self.scheme = .http_unix
118+
case "https+unix":
119+
self.scheme = .https_unix
116120
default:
117121
fatalError("HTTPClient.Request scheme should already be a valid one")
118122
}
@@ -130,10 +134,12 @@ final class ConnectionPool {
130134
case http
131135
case https
132136
case unix
137+
case http_unix
138+
case https_unix
133139

134140
var requiresTLS: Bool {
135141
switch self {
136-
case .https:
142+
case .https, .https_unix:
137143
return true
138144
default:
139145
return false
@@ -455,7 +461,7 @@ class HTTP1ConnectionProvider {
455461
case .http, .https:
456462
let address = HTTPClient.resolveAddress(host: self.key.host, port: self.key.port, proxy: self.configuration.proxy)
457463
channel = bootstrap.connect(host: address.host, port: address.port)
458-
case .unix:
464+
case .unix, .http_unix, .https_unix:
459465
channel = bootstrap.connect(unixDomainSocketPath: self.key.unixPath)
460466
}
461467

Sources/AsyncHTTPClient/HTTPClient.swift

+3
Original file line numberDiff line numberDiff line change
@@ -726,6 +726,7 @@ public struct HTTPClientError: Error, Equatable, CustomStringConvertible {
726726
private enum Code: Equatable {
727727
case invalidURL
728728
case emptyHost
729+
case missingSocketPath
729730
case alreadyShutdown
730731
case emptyScheme
731732
case unsupportedScheme(String)
@@ -758,6 +759,8 @@ public struct HTTPClientError: Error, Equatable, CustomStringConvertible {
758759
public static let invalidURL = HTTPClientError(code: .invalidURL)
759760
/// URL does not contain host.
760761
public static let emptyHost = HTTPClientError(code: .emptyHost)
762+
/// URL does not contain a socketPath as a host for http(s)+unix shemes.
763+
public static let missingSocketPath = HTTPClientError(code: .missingSocketPath)
761764
/// Client is shutdown and cannot be used for new requests.
762765
public static let alreadyShutdown = HTTPClientError(code: .alreadyShutdown)
763766
/// URL does not contain scheme.

Sources/AsyncHTTPClient/HTTPHandler.swift

+14-2
Original file line numberDiff line numberDiff line change
@@ -101,6 +101,8 @@ extension HTTPClient {
101101
enum Kind {
102102
enum UnixScheme {
103103
case baseURL
104+
case http_unix
105+
case https_unix
104106
}
105107

106108
/// Remote host request.
@@ -109,12 +111,14 @@ extension HTTPClient {
109111
case unixSocket(_ scheme: UnixScheme)
110112

111113
private static var hostSchemes = ["http", "https"]
112-
private static var unixSchemes = ["unix"]
114+
private static var unixSchemes = ["unix", "http+unix", "https+unix"]
113115

114116
init(forScheme scheme: String) throws {
115117
switch scheme {
116118
case "http", "https": self = .host
117119
case "unix": self = .unixSocket(.baseURL)
120+
case "http+unix": self = .unixSocket(.http_unix)
121+
case "https+unix": self = .unixSocket(.https_unix)
118122
default:
119123
throw HTTPClientError.unsupportedScheme(scheme)
120124
}
@@ -136,6 +140,11 @@ extension HTTPClient {
136140
switch self {
137141
case .unixSocket(.baseURL):
138142
return url.baseURL?.path ?? url.path
143+
case .unixSocket:
144+
guard let socketPath = url.host else {
145+
throw HTTPClientError.missingSocketPath
146+
}
147+
return socketPath
139148
case .host:
140149
return ""
141150
}
@@ -147,6 +156,8 @@ extension HTTPClient {
147156
return url.uri
148157
case .unixSocket(.baseURL):
149158
return url.baseURL != nil ? url.uri : "/"
159+
case .unixSocket:
160+
return url.uri
150161
}
151162
}
152163

@@ -217,6 +228,7 @@ extension HTTPClient {
217228
/// - `emptyScheme` if URL does not contain HTTP scheme.
218229
/// - `unsupportedScheme` if URL does contains unsupported HTTP scheme.
219230
/// - `emptyHost` if URL does not contains a host.
231+
/// - `missingSocketPath` if URL does not contains a socketPath as an encoded host.
220232
public init(url: URL, method: HTTPMethod = .GET, headers: HTTPHeaders = HTTPHeaders(), body: Body? = nil) throws {
221233
guard let scheme = url.scheme?.lowercased() else {
222234
throw HTTPClientError.emptyScheme
@@ -237,7 +249,7 @@ extension HTTPClient {
237249

238250
/// Whether request will be executed using secure socket.
239251
public var useTLS: Bool {
240-
return self.scheme == "https"
252+
return self.scheme == "https" || self.scheme == "https+unix"
241253
}
242254

243255
/// Resolved port.

0 commit comments

Comments
 (0)