swc/crates/swc_bundler/tests/.cache/untrusted/adcd1c88e039856b981ac7a167115689616a9c51.ts
2021-11-09 20:42:49 +09:00

111 lines
2.8 KiB
TypeScript

// Loaded from https://raw.githubusercontent.com/nats-io/nats.deno/v1.0.0-rc4/nats-base-client/queued_iterator.ts
/*
* Copyright 2020-2021 The NATS Authors
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import { Deferred, deferred } from "./util.ts";
import { ErrorCode, NatsError } from "./error.ts";
export interface Dispatcher<T> {
push(v: T): void;
}
export class QueuedIterator<T> implements Dispatcher<T> {
inflight: number;
processed: number;
received: number; // this is updated by the protocol
protected noIterator: boolean;
protected done: boolean;
private signal: Deferred<void>;
private yields: T[];
yieldedCb?: (data: T) => void;
private err?: Error;
constructor() {
this.inflight = 0;
this.processed = 0;
this.received = 0;
this.noIterator = false;
this.done = false;
this.signal = deferred<void>();
this.yields = [];
}
[Symbol.asyncIterator]() {
return this.iterate();
}
push(v: T) {
if (this.done) {
return;
}
this.yields.push(v);
this.signal.resolve();
}
async *iterate(): AsyncIterableIterator<T> {
if (this.noIterator) {
throw new NatsError("unsupported iterator", ErrorCode.API_ERROR);
}
while (true) {
if (this.yields.length === 0) {
await this.signal;
}
if (this.err) {
throw this.err;
}
const yields = this.yields;
this.inflight = yields.length;
this.yields = [];
for (let i = 0; i < yields.length; i++) {
this.processed++;
yield yields[i];
if (this.yieldedCb) {
this.yieldedCb(yields[i]);
}
this.inflight--;
}
// yielding could have paused and microtask
// could have added messages. Prevent allocations
// if possible
if (this.done) {
break;
} else if (this.yields.length === 0) {
yields.length = 0;
this.yields = yields;
this.signal = deferred();
}
}
}
stop(err?: Error): void {
this.err = err;
this.done = true;
this.signal.resolve();
}
getProcessed(): number {
return this.noIterator ? this.received : this.processed;
}
getPending(): number {
return this.yields.length + this.inflight;
}
getReceived(): number {
return this.received;
}
}