Skip to content

Commit acaca2d

Browse files
authoredOct 21, 2024
Added: ability to set basic authentication on requests (#778)
Motivation: As an HTTP library, async-http-client should have authentication support. Modifications: This adds a `setBasicAuth()` method to both HTTPClientRequest and `HTTPClient.Request` and their related unit tests. Result: Library users will be able to leverage this method to use basic authentication on their requests without implementing this in their own projects. Note: I also ran the tests (`swift test`) with the `docker.io/library/swift:6.0-focal` and `docker.io/library/swift:5.10.1-focal` to ensure linux compatibility. Signed-off-by: Agam Dua <[email protected]>
1 parent 0a9b723 commit acaca2d

File tree

5 files changed

+93
-0
lines changed

5 files changed

+93
-0
lines changed
 
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
//===----------------------------------------------------------------------===//
2+
//
3+
// This source file is part of the AsyncHTTPClient open source project
4+
//
5+
// Copyright (c) 2024 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+
import Foundation
16+
17+
@available(macOS 10.15, iOS 13.0, watchOS 6.0, tvOS 13.0, *)
18+
extension HTTPClientRequest {
19+
/// Set basic auth for a request.
20+
///
21+
/// - parameters:
22+
/// - username: the username to authenticate with
23+
/// - password: authentication password associated with the username
24+
public mutating func setBasicAuth(username: String, password: String) {
25+
self.headers.setBasicAuth(username: username, password: password)
26+
}
27+
}
+39
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,39 @@
1+
//===----------------------------------------------------------------------===//
2+
//
3+
// This source file is part of the AsyncHTTPClient open source project
4+
//
5+
// Copyright (c) 2024 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+
import Foundation
16+
import NIOHTTP1
17+
18+
/// Generates base64 encoded username + password for http basic auth.
19+
///
20+
/// - Parameters:
21+
/// - username: the username to authenticate with
22+
/// - password: authentication password associated with the username
23+
/// - Returns: encoded credentials to use the Authorization: Basic http header.
24+
func encodeBasicAuthCredentials(username: String, password: String) -> String {
25+
var value = Data()
26+
value.reserveCapacity(username.utf8.count + password.utf8.count + 1)
27+
value.append(contentsOf: username.utf8)
28+
value.append(UInt8(ascii: ":"))
29+
value.append(contentsOf: password.utf8)
30+
return value.base64EncodedString()
31+
}
32+
33+
extension HTTPHeaders {
34+
/// Sets the basic auth header
35+
mutating func setBasicAuth(username: String, password: String) {
36+
let encoded = encodeBasicAuthCredentials(username: username, password: password)
37+
self.replaceOrAdd(name: "Authorization", value: "Basic \(encoded)")
38+
}
39+
}

‎Sources/AsyncHTTPClient/HTTPHandler.swift

+9
Original file line numberDiff line numberDiff line change
@@ -285,6 +285,15 @@ extension HTTPClient {
285285

286286
return (head, metadata)
287287
}
288+
289+
/// Set basic auth for a request.
290+
///
291+
/// - parameters:
292+
/// - username: the username to authenticate with
293+
/// - password: authentication password associated with the username
294+
public mutating func setBasicAuth(username: String, password: String) {
295+
self.headers.setBasicAuth(username: username, password: password)
296+
}
288297
}
289298

290299
/// Represents an HTTP response.

‎Tests/AsyncHTTPClientTests/HTTPClientRequestTests.swift

+11
Original file line numberDiff line numberDiff line change
@@ -57,6 +57,17 @@ class HTTPClientRequestTests: XCTestCase {
5757
}
5858
}
5959

60+
func testBasicAuth() {
61+
XCTAsyncTest {
62+
var request = Request(url: "https://example.com/get")
63+
request.setBasicAuth(username: "foo", password: "bar")
64+
var preparedRequest: PreparedRequest?
65+
XCTAssertNoThrow(preparedRequest = try PreparedRequest(request))
66+
guard let preparedRequest = preparedRequest else { return }
67+
XCTAssertEqual(preparedRequest.head.headers.first(name: "Authorization")!, "Basic Zm9vOmJhcg==")
68+
}
69+
}
70+
6071
func testUnixScheme() {
6172
XCTAsyncTest {
6273
var request = Request(url: "unix://%2Fexample%2Ffolder.sock/some_path")

‎Tests/AsyncHTTPClientTests/HTTPClientTests.swift

+7
Original file line numberDiff line numberDiff line change
@@ -3668,4 +3668,11 @@ final class HTTPClientTests: XCTestCaseHTTPClientTestsBaseClass {
36683668
let response3 = try await client.execute(request, timeout: /* infinity */ .hours(99))
36693669
XCTAssertEqual(.ok, response3.status)
36703670
}
3671+
3672+
@available(macOS 10.15, iOS 13.0, watchOS 6.0, tvOS 13.0, *)
3673+
func testRequestBasicAuth() async throws {
3674+
var request = try HTTPClient.Request(url: self.defaultHTTPBinURLPrefix)
3675+
request.setBasicAuth(username: "foo", password: "bar")
3676+
XCTAssertEqual(request.headers.first(name: "Authorization"), "Basic Zm9vOmJhcg==")
3677+
}
36713678
}

0 commit comments

Comments
 (0)
Please sign in to comment.