Skip to content

Commit 854430f

Browse files
committed
Support async generators as body
Fixes #1735
1 parent 45230e3 commit 854430f

File tree

5 files changed

+60
-5
lines changed

5 files changed

+60
-5
lines changed

documentation/3-streams.md

+25
Original file line numberDiff line numberDiff line change
@@ -110,6 +110,31 @@ An object representing how much data have been downloaded.
110110

111111
An object representing how much data have been uploaded.
112112

113+
**Note:**
114+
> - When a chunk is greater than `highWaterMark`, the progress won't be emitted. The body needs to be split into chunks.
115+
116+
```js
117+
import got from 'got';
118+
119+
const body = Buffer.alloc(1024 * 1024); // 1MB
120+
121+
function* chunkify(buffer, chunkSize = 64 * 1024) {
122+
for (let pos = 0; pos < buffer.byteLength; pos += chunkSize) {
123+
yield buffer.subarray(pos, pos + chunkSize)
124+
}
125+
}
126+
127+
const stream = got.stream.post('https://httpbin.org/anything', {
128+
body: chunkify(body)
129+
});
130+
131+
stream.resume();
132+
133+
stream.on('uploadProgress', progress => {
134+
console.log(progress);
135+
});
136+
```
137+
113138
### `stream.timings`
114139

115140
**Type: [`Timings`](typescript.md#timings)**

documentation/4-pagination.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -188,7 +188,7 @@ The reason `filter` looks exactly the same like `shouldContinue` is that the lat
188188
The `filter` function is needed as well, because in the same response we can get results with different timestamps.
189189

190190
```js
191-
import got from '../../dist/source/index.js';
191+
import got from 'got';
192192
import Bourne from '@hapi/bourne';
193193

194194
const max = Date.now() - 1000 * 86400 * 7;

documentation/examples/h2c.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
import http2 from 'http2-wrapper';
2-
import got from 'got';
2+
import got from '../../dist/source/index.js';
33

44
let sessions = {};
55
const getSession = ({origin}) => {

source/core/index.ts

+25
Original file line numberDiff line numberDiff line change
@@ -918,13 +918,38 @@ export default class Request extends Duplex implements RequestEvents<Request> {
918918
this.emit('request', request);
919919
}
920920

921+
private async _asyncWrite(chunk: any): Promise<void> {
922+
return new Promise((resolve, reject) => {
923+
super.write(chunk, error => {
924+
if (error) {
925+
reject(error);
926+
return;
927+
}
928+
929+
resolve();
930+
});
931+
});
932+
}
933+
921934
private _sendBody() {
922935
// Send body
923936
const {body} = this.options;
924937
const currentRequest = this.redirectUrls.length === 0 ? this : this._request ?? this;
925938

926939
if (is.nodeStream(body)) {
927940
body.pipe(currentRequest);
941+
} else if (is.generator(body)) {
942+
(async () => {
943+
try {
944+
for await (const chunk of body) {
945+
await this._asyncWrite(chunk);
946+
}
947+
948+
super.end();
949+
} catch (error) {
950+
this._beforeError(error);
951+
}
952+
})();
928953
} else {
929954
this._unlockWrite();
930955

source/core/options.ts

+8-3
Original file line numberDiff line numberDiff line change
@@ -845,6 +845,11 @@ export default class Options {
845845

846846
this._merging = true;
847847

848+
// Always merge `isStream` first
849+
if ('isStream' in options) {
850+
this.isStream = options.isStream!;
851+
}
852+
848853
try {
849854
let push = false;
850855

@@ -1087,12 +1092,12 @@ export default class Options {
10871092
10881093
Since Got 12, the `content-length` is not automatically set when `body` is a `fs.createReadStream`.
10891094
*/
1090-
get body(): string | Buffer | Readable | undefined {
1095+
get body(): string | Buffer | Readable | Generator | AsyncGenerator | undefined {
10911096
return this._internals.body;
10921097
}
10931098

1094-
set body(value: string | Buffer | Readable | undefined) {
1095-
assert.any([is.string, is.buffer, is.nodeStream, is.undefined], value);
1099+
set body(value: string | Buffer | Readable | Generator | AsyncGenerator | undefined) {
1100+
assert.any([is.string, is.buffer, is.nodeStream, is.generator, is.asyncGenerator, is.undefined], value);
10961101

10971102
if (is.nodeStream(value)) {
10981103
assert.truthy(value.readable);

0 commit comments

Comments
 (0)