Skip to content

Commit c4a63c1

Browse files
authored
Update README.md for async/await (#554)
1 parent 06b9f98 commit c4a63c1

File tree

1 file changed

+86
-7
lines changed

1 file changed

+86
-7
lines changed

README.md

+86-7
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22
This package provides simple HTTP Client library built on top of SwiftNIO.
33

44
This library provides the following:
5+
- First class support for Swift Concurrency (since version 1.9.0)
56
- Asynchronous and non-blocking request methods
67
- Simple follow-redirects (cookie headers are dropped)
78
- Streaming body download
@@ -11,7 +12,7 @@ This library provides the following:
1112

1213
---
1314

14-
**NOTE**: You will need [Xcode 11.4](https://apps.apple.com/gb/app/xcode/id497799835?mt=12) or [Swift 5.2](https://swift.org/download/#swift-52) to try out `AsyncHTTPClient`.
15+
**NOTE**: You will need [Xcode 13.2](https://apps.apple.com/gb/app/xcode/id497799835?mt=12) or [Swift 5.5.2](https://swift.org/download/#swift-552) to try out `AsyncHTTPClient`s new async/await APIs.
1516

1617
---
1718

@@ -21,7 +22,7 @@ This library provides the following:
2122
Add the following entry in your <code>Package.swift</code> to start using <code>HTTPClient</code>:
2223

2324
```swift
24-
.package(url: "https://github.com/swift-server/async-http-client.git", from: "1.0.0")
25+
.package(url: "https://github.com/swift-server/async-http-client.git", from: "1.9.0")
2526
```
2627
and `AsyncHTTPClient` dependency to your target:
2728
```swift
@@ -40,7 +41,21 @@ If your application does not use SwiftNIO yet, it is acceptable to use `eventLoo
4041
import AsyncHTTPClient
4142

4243
let httpClient = HTTPClient(eventLoopGroupProvider: .createNew)
43-
httpClient.get(url: "https://swift.org").whenComplete { result in
44+
45+
/// MARK: - Using Swift Concurrency
46+
let request = HTTPClientRequest(url: "https://apple.com/")
47+
let response = try await httpClient.execute(request, timeout: .seconds(30))
48+
print("HTTP head", response)
49+
if response.status == .ok {
50+
let body = try await response.body.collect(upTo: 1024 * 1024) // 1 MB
51+
// handle body
52+
} else {
53+
// handle remote error
54+
}
55+
56+
57+
/// MARK: - Using SwiftNIO EventLoopFuture
58+
httpClient.get(url: "https://apple.com/").whenComplete { result in
4459
switch result {
4560
case .failure(let error):
4661
// process error
@@ -58,7 +73,35 @@ You should always shut down `HTTPClient` instances you created using `try httpCl
5873

5974
## Usage guide
6075

61-
Most common HTTP methods are supported out of the box. In case you need to have more control over the method, or you want to add headers or body, use the `HTTPRequest` struct:
76+
The default HTTP Method is `GET`. In case you need to have more control over the method, or you want to add headers or body, use the `HTTPClientRequest` struct:
77+
78+
#### Using Swift Concurrency
79+
80+
```swift
81+
import AsyncHTTPClient
82+
83+
let httpClient = HTTPClient(eventLoopGroupProvider: .createNew)
84+
do {
85+
var request = HTTPClientRequest(url: "https://apple.com/")
86+
request.method = .POST
87+
request.headers.add(name: "User-Agent", value: "Swift HTTPClient")
88+
request.body = .bytes(ByteBuffer(string: "some data"))
89+
90+
let response = try await httpClient.execute(request, timeout: .seconds(30))
91+
if response.status == .ok {
92+
// handle response
93+
} else {
94+
// handle remote error
95+
}
96+
} catch {
97+
// handle error
98+
}
99+
// it's important to shutdown the httpClient after all requests are done, even if one failed
100+
try await httpClient.shutdown()
101+
```
102+
103+
#### Using SwiftNIO EventLoopFuture
104+
62105
```swift
63106
import AsyncHTTPClient
64107

@@ -67,7 +110,7 @@ defer {
67110
try? httpClient.syncShutdown()
68111
}
69112

70-
var request = try HTTPClient.Request(url: "https://swift.org", method: .POST)
113+
var request = try HTTPClient.Request(url: "https://apple.com/", method: .POST)
71114
request.headers.add(name: "User-Agent", value: "Swift HTTPClient")
72115
request.body = .string("some-body")
73116

@@ -105,7 +148,43 @@ httpClient.execute(request: request, deadline: .now() + .milliseconds(1))
105148
```
106149

107150
### Streaming
108-
When dealing with larger amount of data, it's critical to stream the response body instead of aggregating in-memory. Handling a response stream is done using a delegate protocol. The following example demonstrates how to count the number of bytes in a streaming response body:
151+
When dealing with larger amount of data, it's critical to stream the response body instead of aggregating in-memory.
152+
The following example demonstrates how to count the number of bytes in a streaming response body:
153+
154+
#### Using Swift Concurrency
155+
```swift
156+
let httpClient = HTTPClient(eventLoopGroupProvider: .createNew)
157+
do {
158+
let request = HTTPClientRequest(url: "https://apple.com/")
159+
let response = try await httpClient.execute(request, timeout: .seconds(30))
160+
print("HTTP head", response)
161+
162+
// if defined, the content-length headers announces the size of the body
163+
let expectedBytes = response.headers.first(name: "content-length").flatMap(Int.init)
164+
165+
var receivedBytes = 0
166+
// asynchronously iterates over all body fragments
167+
// this loop will automatically propagate backpressure correctly
168+
for try await buffer in response.body {
169+
// for this example, we are just interested in the size of the fragment
170+
receivedBytes += buffer.readableBytes
171+
172+
if let expectedBytes = expectedBytes {
173+
// if the body size is known, we calculate a progress indicator
174+
let progress = Double(receivedBytes) / Double(expectedBytes)
175+
print("progress: \(Int(progress * 100))%")
176+
}
177+
}
178+
print("did receive \(receivedBytes) bytes")
179+
} catch {
180+
print("request failed:", error)
181+
}
182+
// it is important to shutdown the httpClient after all requests are done, even if one failed
183+
try await httpClient.shutdown()
184+
```
185+
186+
#### Using HTTPClientResponseDelegate and SwiftNIO EventLoopFuture
187+
109188
```swift
110189
import NIOCore
111190
import NIOHTTP1
@@ -158,7 +237,7 @@ class CountingDelegate: HTTPClientResponseDelegate {
158237
}
159238
}
160239

161-
let request = try HTTPClient.Request(url: "https://swift.org")
240+
let request = try HTTPClient.Request(url: "https://apple.com/")
162241
let delegate = CountingDelegate()
163242

164243
httpClient.execute(request: request, delegate: delegate).futureResult.whenSuccess { count in

0 commit comments

Comments
 (0)