astx
Version:
super powerful structural search and replace for JavaScript and TypeScript to automate your refactoring
172 lines (150 loc) • 14.2 kB
JavaScript
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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"}