UNPKG

aws-crt

Version:

NodeJS/browser bindings to the aws-c-* libraries

238 lines (196 loc) 7.64 kB
/* * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. * SPDX-License-Identifier: Apache-2.0. */ /** * @packageDocumentation */ import * as event from "./event"; /** * Converts payload to Buffer or string regardless of the supplied type * @param payload The payload to convert * @internal */ export function normalize_payload(payload: any): Buffer | string { if (payload instanceof Buffer) { // pass Buffer through return payload; } if (typeof payload === 'string') { // pass string through return payload; } if (ArrayBuffer.isView(payload)) { // return Buffer with view upon the same bytes (no copy) const view = payload as ArrayBufferView; return Buffer.from(view.buffer, view.byteOffset, view.byteLength); } if (payload instanceof ArrayBuffer) { // return Buffer with view upon the same bytes (no copy) return Buffer.from(payload); } if (typeof payload === 'object') { // Convert Object to JSON string return JSON.stringify(payload); } if (!payload) { return ""; } throw new TypeError("payload parameter must be a string, object, or DataView."); } /** * Converts payload to Buffer only, regardless of the supplied type * @param payload The payload to convert * @internal */ export function normalize_payload_to_buffer(payload: any): Buffer { let normalized = normalize_payload(payload); if (typeof normalized === 'string') { // pass string through return Buffer.from(normalized); } return normalized; } /** @internal */ export const DEFAULT_KEEP_ALIVE : number = 1200; function isValidTopicInternal(topic: string, isFilter: boolean) : boolean { if (topic.length === 0 || topic.length > 65535) { return false; } let sawHash : boolean = false; for (let segment of topic.split('/')) { if (sawHash) { return false; } if (segment.length === 0) { continue; } if (segment.includes("+")) { if (!isFilter) { return false; } if (segment.length > 1) { return false; } } if (segment.includes("#")) { if (!isFilter) { return false; } if (segment.length > 1) { return false; } sawHash = true; } } return true; } export function isValidTopicFilter(topicFilter: any) : boolean { if (typeof(topicFilter) !== 'string') { return false; } let topicFilterAsString = topicFilter as string; return isValidTopicInternal(topicFilterAsString, true); } export function isValidTopic(topic: any) : boolean { if (typeof(topic) !== 'string') { return false; } let topicAsString = topic as string; return isValidTopicInternal(topicAsString, false); } export type PublishAcknowledgementFunctor = () => void; /** * Wrapper class containing a one-use singleton handle that can be used to trigger sending the acknowledgement (Puback in * QoS 1, Pubrec in QoS 2) packet for an incoming publish. */ export class PublishAcknowledgementHandleWrapper { private ackHandle : PublishAcknowledgementHandle | null; constructor(handle : PublishAcknowledgementHandle | null) { this.ackHandle = handle; } /** * Attempt to take the acknowledgement handle held by the wrapper. This will only succeed for the first caller; * after the initial call, null will be returned. By taking the handle, the caller assumes responsibility * for sending the acknowledgement packet associated with the incoming publish packet. Failing to trigger the * acknowledgement will cause the broker to potentially re-send the publish. */ acquireHandle() : PublishAcknowledgementHandle | null { let handle = this.ackHandle; this.ackHandle = null; return handle; } } function movePublishAcknowledgementHandleWrapper(wrapper: PublishAcknowledgementHandleWrapper | undefined, compositionFunctor?: PublishAcknowledgementFunctor) : PublishAcknowledgementHandleWrapper | undefined { if (wrapper) { let handle = wrapper.acquireHandle(); if (compositionFunctor && handle) { let interiorHandle = handle; handle = new PublishAcknowledgementHandle(() => { interiorHandle.invokeAcknowledgement(); compositionFunctor(); }); } return new PublishAcknowledgementHandleWrapper(handle); } return undefined; } /** @internal */ export function emitAcknowledgeableEvent<T>(emitter: event.BufferedEventEmitter, ackEvent: string, ackEventPayload: T, wrapperFieldName: string, ackHandleWrapper?: PublishAcknowledgementHandleWrapper, compositionFunctor?: PublishAcknowledgementFunctor) : void { ackHandleWrapper = movePublishAcknowledgementHandleWrapper(ackHandleWrapper, compositionFunctor); if (ackHandleWrapper) { (ackEventPayload as any)[wrapperFieldName] = ackHandleWrapper; emitter.emitWithCallback(ackEvent, () => { if (ackHandleWrapper) { let handle = ackHandleWrapper.acquireHandle(); if (handle) { // Even if corked, all listeners have had a chance to react to the event // and acquire the acknowledgement handle if they wanted to. If no one did so, then we do it ourselves. handle.invokeAcknowledgement(); } } }, ackEventPayload); } else { emitter.emit(ackEvent, ackEventPayload); } } /** @internal */ export function queueAcknowledgeableEvent<T>(emitter: event.BufferedEventEmitter, ackEvent: string, ackEventPayload: T, wrapperFieldName: string, ackHandleWrapper?: PublishAcknowledgementHandleWrapper, compositionFunctor?: PublishAcknowledgementFunctor) : void { let wrapper : PublishAcknowledgementHandleWrapper | undefined = movePublishAcknowledgementHandleWrapper(ackHandleWrapper, compositionFunctor); queueMicrotask(() => { if (wrapper) { (ackEventPayload as any)[wrapperFieldName] = wrapper; emitter.emitWithCallback(ackEvent, () => { if (wrapper) { let handle = wrapper.acquireHandle(); if (handle) { // Even if corked, all listeners have had a chance to react to the event // and acquire the acknowledgement handle if they wanted to. If no one did so, then we do it ourselves. handle.invokeAcknowledgement(); } } }, ackEventPayload); } else { emitter.emit(ackEvent, ackEventPayload); } }); } /** * Object that allows the holder to trigger the acknowledgement for an associated publish packet. */ export class PublishAcknowledgementHandle { private acknowledgementFunction? : PublishAcknowledgementFunctor; constructor(acknowledgementFunction : PublishAcknowledgementFunctor) { this.acknowledgementFunction = acknowledgementFunction; } /** * trigger the acknowledgement for an associated Publish packet */ invokeAcknowledgement() : void { let acknowledgementFunction = this.acknowledgementFunction; this.acknowledgementFunction = undefined; if (acknowledgementFunction) { acknowledgementFunction(); } } }