61 lines
1.3 KiB
TypeScript
61 lines
1.3 KiB
TypeScript
// Copyright 2021-2022 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
import type { Readable } from 'stream';
|
|
|
|
import * as Bytes from '../Bytes';
|
|
import { clearTimeoutIfNecessary } from './clearTimeoutIfNecessary';
|
|
import { explodePromise } from './explodePromise';
|
|
|
|
export type OptionsType = Readonly<{
|
|
name: string;
|
|
timeout: number;
|
|
abortController: { abort(): void };
|
|
}>;
|
|
|
|
export class StreamTimeoutError extends Error {}
|
|
|
|
export function getStreamWithTimeout(
|
|
stream: Readable,
|
|
{ name, timeout, abortController }: OptionsType
|
|
): Promise<Uint8Array> {
|
|
const { promise, resolve, reject } = explodePromise<Uint8Array>();
|
|
|
|
const chunks = new Array<Uint8Array>();
|
|
|
|
let timer: NodeJS.Timeout | undefined;
|
|
|
|
const clearTimer = () => {
|
|
clearTimeoutIfNecessary(timer);
|
|
timer = undefined;
|
|
};
|
|
|
|
const reset = () => {
|
|
clearTimer();
|
|
|
|
timer = setTimeout(() => {
|
|
abortController.abort();
|
|
reject(new StreamTimeoutError(`getStreamWithTimeout(${name}) timed out`));
|
|
}, timeout);
|
|
};
|
|
|
|
stream.on('data', chunk => {
|
|
reset();
|
|
|
|
chunks.push(chunk);
|
|
});
|
|
|
|
stream.on('end', () => {
|
|
clearTimer();
|
|
resolve(Bytes.concatenate(chunks));
|
|
});
|
|
|
|
stream.on('error', error => {
|
|
clearTimer();
|
|
reject(error);
|
|
});
|
|
|
|
reset();
|
|
|
|
return promise;
|
|
}
|