UNPKG

astx

Version:

super powerful structural search and replace for JavaScript and TypeScript to automate your refactoring

172 lines (150 loc) 14.2 kB
import RingBuffer from './RingBuffer.mjs' export default class PushPullIterable { queue pushQueue = [] pullQueue = [] producing = true consuming = true iterating = false consumeError produceError constructor(capacity) { if ( (capacity !== Infinity && !Number.isFinite(capacity)) || capacity < 0 || capacity % 1 ) { throw new Error(`invalid capacity: ${capacity}`) } this.queue = new RingBuffer(capacity) } [Symbol.asyncIterator]() { if (this.iterating) { throw new Error( `this iterable doesn't support creating more than one iterator` ) } this.iterating = true // eslint-disable-next-line @typescript-eslint/no-this-alias const self = this return { async next() { return self.pull() }, async return() { self.iteratorReturn() return { value: undefined, done: true, } }, async throw(error) { self.iteratorThrow({ error, }) return { value: undefined, done: true, } }, } } async push(value) { if (!this.producing) { throw new Error(`can't push after returning or throwing`) } if (!this.consuming) return false if (this.consumeError) throw this.consumeError const waitingPull = this.pullQueue.shift() if (waitingPull) { waitingPull.resolve({ value, done: false, }) return this.consuming } if (!this.queue.isFull) { this.queue.push(value) return this.consuming } return new Promise((resolve, reject) => { this.pushQueue.push({ resolve: (keepGoing) => { if (keepGoing && this.producing) this.queue.push(value) resolve(keepGoing) }, reject, }) }) } async pull() { if (!this.consuming) { throw new Error(`can't call next after returning or throwing`) } if (this.produceError) throw this.produceError if (this.queue.size) { var _this$pushQueue$shift const pulled = this.queue.pull() ;(_this$pushQueue$shift = this.pushQueue.shift()) === null || _this$pushQueue$shift === void 0 ? void 0 : _this$pushQueue$shift.resolve(true) return { value: pulled, done: false, } } if (!this.producing) { return { value: undefined, done: true, } } return new Promise((resolve, reject) => { this.pullQueue.push({ resolve, reject, }) }) } return() { if (!this.producing) return this.producing = false const { pullQueue } = this this.pullQueue = [] for (const pull of pullQueue) { pull.resolve({ value: undefined, done: true, }) } } // eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types throw(error) { if (!this.producing) return this.producing = false this.produceError = error const { pullQueue } = this this.pullQueue = [] for (const pull of pullQueue) { pull.reject(error) } } iteratorReturn() { if (!this.consuming) return this.consuming = false const { pushQueue } = this this.pushQueue = [] for (const push of pushQueue) { push.resolve(false) } } iteratorThrow(error) { if (!this.consuming) return this.consuming = false this.consumeError = error const { pushQueue } = this this.pushQueue = [] for (const push of pushQueue) { push.reject(error) } } } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["RingBuffer","PushPullIterable","queue","pushQueue","pullQueue","producing","consuming","iterating","consumeError","produceError","constructor","capacity","Infinity","Number","isFinite","Error","Symbol","asyncIterator","self","next","pull","return","iteratorReturn","value","undefined","done","throw","error","iteratorThrow","push","waitingPull","shift","resolve","isFull","Promise","reject","keepGoing","size","pulled"],"sources":["../../src/util/PushPullIterable.ts"],"sourcesContent":["import RingBuffer from './RingBuffer'\n\ntype ResolveReject<T> = {\n  resolve: (value: T) => void\n  reject: (error?: any) => void\n}\n\nexport default class PushPullIterable<T> implements AsyncIterable<T> {\n  private queue: RingBuffer<T>\n  private pushQueue: ResolveReject<boolean>[] = []\n  private pullQueue: ResolveReject<IteratorResult<T>>[] = []\n  private producing = true\n  private consuming = true\n  private iterating = false\n  private consumeError: any\n  private produceError: any\n\n  constructor(capacity: number) {\n    if (\n      (capacity !== Infinity && !Number.isFinite(capacity)) ||\n      capacity < 0 ||\n      capacity % 1\n    ) {\n      throw new Error(`invalid capacity: ${capacity}`)\n    }\n    this.queue = new RingBuffer(capacity)\n  }\n\n  [Symbol.asyncIterator](): AsyncIterator<T> {\n    if (this.iterating) {\n      throw new Error(\n        `this iterable doesn't support creating more than one iterator`\n      )\n    }\n    this.iterating = true\n    // eslint-disable-next-line @typescript-eslint/no-this-alias\n    const self = this\n    return {\n      async next(): Promise<IteratorResult<T>> {\n        return self.pull()\n      },\n      async return(): Promise<IteratorResult<T>> {\n        self.iteratorReturn()\n        return { value: undefined, done: true }\n      },\n      async throw(error?: any): Promise<IteratorResult<T>> {\n        self.iteratorThrow({ error })\n        return { value: undefined, done: true }\n      },\n    }\n  }\n\n  async push(value: T): Promise<boolean> {\n    if (!this.producing) {\n      throw new Error(`can't push after returning or throwing`)\n    }\n    if (!this.consuming) return false\n    if (this.consumeError) throw this.consumeError\n    const waitingPull = this.pullQueue.shift()\n    if (waitingPull) {\n      waitingPull.resolve({ value, done: false })\n      return this.consuming\n    }\n    if (!this.queue.isFull) {\n      this.queue.push(value)\n      return this.consuming\n    }\n    return new Promise<boolean>((resolve, reject) => {\n      this.pushQueue.push({\n        resolve: (keepGoing: boolean) => {\n          if (keepGoing && this.producing) this.queue.push(value)\n          resolve(keepGoing)\n        },\n        reject,\n      })\n    })\n  }\n\n  private async pull(): Promise<IteratorResult<T>> {\n    if (!this.consuming) {\n      throw new Error(`can't call next after returning or throwing`)\n    }\n    if (this.produceError) throw this.produceError\n    if (this.queue.size) {\n      const pulled: T = this.queue.pull() as any\n      this.pushQueue.shift()?.resolve(true)\n      return { value: pulled, done: false }\n    }\n    if (!this.producing) {\n      return { value: undefined, done: true }\n    }\n    return new Promise<IteratorResult<T>>((resolve, reject) => {\n      this.pullQueue.push({ resolve, reject })\n    })\n  }\n\n  return(): void {\n    if (!this.producing) return\n    this.producing = false\n    const { pullQueue } = this\n    this.pullQueue = []\n    for (const pull of pullQueue) {\n      pull.resolve({ value: undefined, done: true })\n    }\n  }\n\n  // eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types\n  throw(error?: any): void {\n    if (!this.producing) return\n    this.producing = false\n    this.produceError = error\n    const { pullQueue } = this\n    this.pullQueue = []\n    for (const pull of pullQueue) {\n      pull.reject(error)\n    }\n  }\n\n  private iteratorReturn(): void {\n    if (!this.consuming) return\n    this.consuming = false\n    const { pushQueue } = this\n    this.pushQueue = []\n    for (const push of pushQueue) {\n      push.resolve(false)\n    }\n  }\n\n  private iteratorThrow(error?: any): void {\n    if (!this.consuming) return\n    this.consuming = false\n    this.consumeError = error\n    const { pushQueue } = this\n    this.pushQueue = []\n    for (const push of pushQueue) {\n      push.reject(error)\n    }\n  }\n}\n"],"mappings":"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"}