botframework-streaming
Version:
Streaming library for the Microsoft Bot Framework
128 lines (111 loc) • 4.37 kB
text/typescript
/**
* @module botframework-streaming
*/
/**
* Copyright (c) Microsoft Corporation. All rights reserved.
* Licensed under the MIT License.
*/
import { INodeBuffer } from './interfaces/INodeBuffer';
import type { PayloadAssembler } from './assemblers';
import { PayloadTypes } from './payloads';
import type { SubscribableStream } from './subscribableStream';
/**
* A stream of fixed or infinite length containing content to be decoded.
*/
export class ContentStream {
id: string;
private readonly assembler: PayloadAssembler;
private stream: SubscribableStream;
/**
* Initializes a new instance of the [ContentStream](xref:botframework-streaming.ContentStream) class.
*
* @param id The ID assigned to this instance.
* @param assembler The [PayloadAssembler](xref:botframework-streaming.PayloadAssembler) assigned to this instance.
*/
constructor(id: string, assembler: PayloadAssembler) {
if (!assembler) {
throw Error('Null Argument Exception');
}
this.id = id;
this.assembler = assembler;
}
/**
* Gets the name of the type of the object contained within this [ContentStream](xref:botframework-streaming.ContentStream).
*
* @returns The [PayloadType](xref:botframework-streaming.PayloadType) of this [ContentStream](xref:botframework-streaming.ContentStream).
*/
get contentType(): string | PayloadTypes {
return this.assembler.payloadType;
}
/**
* Gets the length of this [ContentStream](xref:botframework-streaming.ContentStream).
*
* @returns A number representing the length of this [ContentStream](xref:botframework-streaming.ContentStream).
*/
get length(): number {
return this.assembler.contentLength;
}
/**
* Gets the data contained within this [ContentStream](xref:botframework-streaming.ContentStream).
*
* @returns This [ContentStream's](xref:botframework-streaming.ContentStream) [SubscribableStream](xref:botframework-streaming.SubscribableStream).
*/
getStream(): SubscribableStream {
if (!this.stream) {
this.stream = this.assembler.getPayloadStream();
}
return this.stream;
}
/**
* Closes the assembler.
*/
cancel(): void {
this.assembler.close();
}
/**
* Gets the [SubscribableStream](xref:botframework-streaming.SubscribableStream) content as a string.
*
* @returns A string Promise with [SubscribableStream](xref:botframework-streaming.SubscribableStream) content.
*/
async readAsString(): Promise<string> {
const { bufferArray } = await this.readAll();
return (bufferArray || []).map((result) => result.toString('utf8')).join('');
}
/**
* Gets the [SubscribableStream](xref:botframework-streaming.SubscribableStream) content as a typed JSON object.
*
* @returns A typed object Promise with `SubscribableStream` content.
*/
async readAsJson<T>(): Promise<T> {
const stringToParse = await this.readAsString();
return <T>JSON.parse(stringToParse);
}
private async readAll(): Promise<Record<string, any>> {
// do a read-all
const allData: INodeBuffer[] = [];
let count = 0;
const stream = this.getStream();
// populate the array with any existing buffers
while (count < stream.length) {
const chunk = stream.read(stream.length);
allData.push(chunk);
count += (chunk as INodeBuffer).length;
}
if (count < this.length) {
const readToEnd = new Promise<boolean>((resolve): void => {
const callback =
(cs: ContentStream) =>
(chunk: any): void => {
allData.push(chunk);
count += (chunk as INodeBuffer).length;
if (count === cs.length) {
resolve(true);
}
};
stream.subscribe(callback(this));
});
await readToEnd;
}
return { bufferArray: allData, size: count };
}
}