UNPKG

22.1 kBJavaScriptView Raw
1"use strict";
2
3Object.defineProperty(exports, "__esModule", {
4 value: true
5});
6exports.default = void 0;
7
8var _events = _interopRequireDefault(require("events"));
9
10var _autoBind = _interopRequireDefault(require("auto-bind"));
11
12var _AsyncQueue = _interopRequireWildcard(require("./AsyncQueue"));
13
14var _asyncSyncUtils = require("./asyncSyncUtils");
15
16var _errorUtils = require("./errorUtils");
17
18function _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } else { var newObj = {}; if (obj != null) { for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) { var desc = Object.defineProperty && Object.getOwnPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : {}; if (desc.get || desc.set) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } } newObj.default = obj; return newObj; } }
19
20function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
21
22function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
23
24/**
25 * A thread-pool abstraction for ES6 async operations",
26 *
27 * @export
28 * @class ThreadPool
29 * @extends {EventEmitter}
30 */
31class ThreadPool extends _events.default {
32 static async run(options) {
33 options.errorHandler = options.errorHandler || (err => {
34 throw err;
35 });
36
37 const tp = new ThreadPool(options);
38
39 if (options.items) {
40 await tp.queueItems(options.items);
41 }
42
43 return await tp.runAllQueued();
44 }
45
46 static async all(items, task, threads) {
47 const tp = new ThreadPool({
48 task: task,
49 threads: threads,
50 items: items
51 });
52
53 if (items) {
54 await tp.queueItems(items);
55 }
56
57 return await tp.runAllQueued();
58 }
59 /**
60 * Creates an instance of ThreadPool
61 */
62
63
64 constructor(options) {
65 super();
66
67 _defineProperty(this, "queuedCount", 0);
68
69 _defineProperty(this, "startedCount", 0);
70
71 _defineProperty(this, "endedCount", 0);
72
73 _defineProperty(this, "_options", void 0);
74
75 _defineProperty(this, "_errorHandler", void 0);
76
77 _defineProperty(this, "_uncaughtErrors", []);
78
79 _defineProperty(this, "_closed", false);
80
81 _defineProperty(this, "_queuedTasks", void 0);
82
83 _defineProperty(this, "_threadsSemaphore", void 0);
84
85 _defineProperty(this, "_completeEvent", void 0);
86
87 _defineProperty(this, "_allTasksCompleteOrSomeFailedEvent", void 0);
88
89 _defineProperty(this, "_res", []);
90
91 this._options = options;
92 this._options.threads = this._options.threads || Infinity;
93
94 this._errorHandler = this._options.errorHandler || (err => {
95 throw err;
96 }); // eslint-disable-line no-console
97
98
99 this._queuedTasks = new _AsyncQueue.default({
100 name: options.name,
101 maxSize: options.queueMaxSize
102 });
103 this._threadsSemaphore = new _asyncSyncUtils.Semaphore(this._options.threads);
104 this._completeEvent = new _asyncSyncUtils.OneTimeBroadcastEvent(false);
105 this._allTasksCompleteOrSomeFailedEvent = new _asyncSyncUtils.OneTimeBroadcastEvent(false);
106 (0, _autoBind.default)(this);
107 }
108 /**
109 * Queues an itme from the ThreadPool
110 */
111
112
113 async queueItem(item) {
114 if (this._closed) throw new Error(`Trying to queue a job to a closed ThreadPool`);
115 const index = this.queuedCount++;
116 await this._queuedTasks.enqueue({
117 func: async () => this._res[index] = await this._options.task(item),
118 index: index
119 });
120 }
121 /**
122 * Queues an itme from the ThreadPool
123 */
124
125
126 async queueItems(queueItem) {
127 for (const item of queueItem) {
128 await this.queueItem(item);
129 }
130 }
131 /**
132 * Starts executing all queued tasks
133 *
134 * This function should be awaited - it will return after the ThreadPool has been closed and all it's tasks completed, or after a task threw an error.
135 */
136
137
138 async run() {
139 try {
140 while (true) {
141 // wait for an available task
142 // TODO: also wait for errors here?
143 let task;
144
145 try {
146 task = await this._queuedTasks.dequeue();
147 } catch (err) {
148 if (err instanceof _AsyncQueue.QueueClosedError) break;
149 throw err;
150 } // wait for an available 'thread'
151
152
153 await this._threadsSemaphore.enter();
154
155 if (this._uncaughtErrors.length) {
156 break;
157 }
158
159 this._allTasksCompleteOrSomeFailedEvent.reset(); // NOTE: no await
160
161
162 this._runTask(task);
163 } // wait for completion
164
165
166 if (this.startedCount > 0) {
167 await this._allTasksCompleteOrSomeFailedEvent.wait();
168 }
169
170 this._throwUncaughtErrors();
171 } finally {
172 this._completeEvent.signal();
173 }
174 }
175 /**
176 * Starts executing all queued tasks
177 *
178 * This function should not be awaited - it will return immediatly
179 */
180
181
182 startRun() {
183 (async () => {
184 try {
185 await this.run();
186 } catch (err) {} // eslint-disable-line no-empty
187
188 })();
189 }
190 /**
191 * Closes the ThreadPool for further task queueing, the ThreadPool's completion can be awaited afte'r it's called
192 */
193
194
195 close() {
196 this._closed = true;
197
198 this._queuedTasks.close(); // NOTE: NO AWAIT
199 // TODO: await completion
200
201 }
202 /**
203 * Closes the ThreadPool, runs it's tasks and awaits their completion
204 */
205
206
207 async runAllQueued() {
208 this.close();
209 await this.run();
210 return this._res;
211 }
212 /**
213 * Awaits the closing and completion of all ThreadPool tasks
214 */
215
216
217 async waitComplete() {
218 await this._completeEvent.wait();
219
220 this._throwUncaughtErrors();
221
222 return this._res;
223 }
224 /**
225 * Closes the ThreadPool and awaits the running and completion of all ThreadPool tasks
226 */
227
228
229 async closeAndWaitComplete() {
230 this.close();
231 return await this.waitComplete();
232 }
233 /****************** privates ******************/
234
235
236 _throwUncaughtErrors() {
237 if (this._uncaughtErrors.length > 0) {
238 throw new _errorUtils.ExtendedError(`Errors were thrown during execution of ThreadPool`, {
239 threadPoolName: this._options.name,
240 errorCount: this._uncaughtErrors.length,
241 errorMessages: this._uncaughtErrors.map(e => e.message),
242 uncaughtErrors: this._uncaughtErrors
243 });
244 }
245 }
246
247 async _runTask(task) {
248 try {
249 this.startedCount++;
250 await task.func();
251 } catch (err) {
252 try {
253 this._errorHandler(err);
254 } catch (err2) {
255 this._uncaughtErrors.push(err2);
256
257 await this._allTasksCompleteOrSomeFailedEvent.signal();
258 }
259 } finally {
260 this.endedCount++; // fire 'progress' event
261
262 this.emit('progress', {
263 endedCount: this.endedCount
264 });
265
266 this._threadsSemaphore.exit();
267
268 if (this._threadsSemaphore.takenCount === 0) {
269 this._allTasksCompleteOrSomeFailedEvent.signal();
270 }
271 }
272 }
273
274}
275
276exports.default = ThreadPool;
277//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/ThreadPool.js"],"names":["ThreadPool","EventEmitter","run","options","errorHandler","err","tp","items","queueItems","runAllQueued","all","task","threads","constructor","_options","Infinity","_errorHandler","_queuedTasks","AsyncQueue","name","maxSize","queueMaxSize","_threadsSemaphore","Semaphore","_completeEvent","OneTimeBroadcastEvent","_allTasksCompleteOrSomeFailedEvent","queueItem","item","_closed","Error","index","queuedCount","enqueue","func","_res","dequeue","QueueClosedError","enter","_uncaughtErrors","length","reset","_runTask","startedCount","wait","_throwUncaughtErrors","signal","startRun","close","waitComplete","closeAndWaitComplete","ExtendedError","threadPoolName","errorCount","errorMessages","map","e","message","uncaughtErrors","err2","push","endedCount","emit","exit","takenCount"],"mappings":"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","sourcesContent":["// @flow\n\nimport EventEmitter from 'events'\nimport autoBind from 'auto-bind'\n\nimport AsyncQueue, { QueueClosedError } from './AsyncQueue'\nimport { OneTimeBroadcastEvent, Semaphore } from './asyncSyncUtils'\nimport { ExtendedError } from './errorUtils'\n\n\ntype ThreadPoolOptions<T, R> = {\n  name?: ?string,\n  threads?: ?number,\n  items?: ?Array<T>,\n  task: (item: T) => Promise<R>,\n  errorHandler?: ?(err: Error) => void, \n  queueMaxSize?: ?number,\n}\n\ntype Task<R> = { \n  func: () => Promise<R>, \n  index: number,\n}\n\n/**\n * A thread-pool abstraction for ES6 async operations\",\n * \n * @export\n * @class ThreadPool\n * @extends {EventEmitter}\n */\nexport default class ThreadPool<T, R> extends EventEmitter {\n  \n  queuedCount: number = 0\n  startedCount: number = 0\n  endedCount: number = 0\n\n  _options: ThreadPoolOptions<T, R>\n  _errorHandler: (err: Error) => void\n  _uncaughtErrors: Array<Error> = []\n  _closed: boolean = false\n  _queuedTasks: AsyncQueue<{ func: () => Promise<R>, index: number }>\n  _threadsSemaphore: Semaphore\n  _completeEvent: OneTimeBroadcastEvent\n  _allTasksCompleteOrSomeFailedEvent: OneTimeBroadcastEvent\n  _res: Array<R> = []\n\n\n  static async run(options: ThreadPoolOptions<T, R>): Promise<Array<R>> {\n    options.errorHandler = options.errorHandler || ((err) => { throw err })\n    const tp = new ThreadPool(options)\n    if (options.items) {\n      await tp.queueItems(options.items)\n    }\n    return await tp.runAllQueued()\n  }\n\n  static async all(items: Array<T>, task: (item: T) => Promise<R>, threads: ?number): Promise<Array<R>> {\n    const tp = new ThreadPool({\n      task: task,\n      threads: threads,\n      items: items,\n    })\n    if (items) {\n      await tp.queueItems(items)\n    }\n    return await tp.runAllQueued()\n  }\n\n\n  /**\n   * Creates an instance of ThreadPool\n   */\n  constructor(options: ThreadPoolOptions<T, R>) {\n    super()\n\n    this._options = options\n    this._options.threads = this._options.threads || Infinity\n\n    this._errorHandler = this._options.errorHandler || ((err) => { throw err }) // eslint-disable-line no-console\n\n    this._queuedTasks = new AsyncQueue({ \n      name: options.name,\n      maxSize: options.queueMaxSize,\n    })\n    this._threadsSemaphore = new Semaphore(this._options.threads)\n    this._completeEvent = new OneTimeBroadcastEvent(false)\n    this._allTasksCompleteOrSomeFailedEvent = new OneTimeBroadcastEvent(false)\n\n    autoBind(this)\n  }\n\n  /**\n   * Queues an itme from the ThreadPool\n   */\n  async queueItem(item: T): Promise<void> {\n    if (this._closed)\n      throw new Error(`Trying to queue a job to a closed ThreadPool`)\n\n    const index = this.queuedCount++   \n    await this._queuedTasks.enqueue({ \n      func: async () => this._res[index] = await this._options.task(item), \n      index: index,\n    })\n  }\n\n  /**\n   * Queues an itme from the ThreadPool\n   */\n  async queueItems(queueItem: Array<T>): Promise<void> {\n    for (const item of queueItem) {\n      await this.queueItem(item)\n    }\n  }\n  \n  /**\n   * Starts executing all queued tasks\n   * \n   * This function should be awaited - it will return after the ThreadPool has been closed and all it's tasks completed, or after a task threw an error.\n   */\n  async run() {\n    try {\n      while (true) {\n\n        // wait for an available task\n        // TODO: also wait for errors here?\n        let task\n        try {\n          task = await this._queuedTasks.dequeue()\n        } catch (err) {\n          if (err instanceof QueueClosedError)\n            break\n          throw err\n        }\n  \n        // wait for an available 'thread'\n        await this._threadsSemaphore.enter()\n        if (this._uncaughtErrors.length) {\n          break\n        }\n  \n        this._allTasksCompleteOrSomeFailedEvent.reset()\n        \n        // NOTE: no await\n        this._runTask(task)\n      }\n  \n      // wait for completion\n      if (this.startedCount > 0) {\n        await this._allTasksCompleteOrSomeFailedEvent.wait()\n      }\n  \n      this._throwUncaughtErrors()\n    } finally {\n      this._completeEvent.signal()\n    }\n  }\n\n  /**\n   * Starts executing all queued tasks\n   * \n   * This function should not be awaited - it will return immediatly\n   */\n  startRun() {\n    (async () => {\n      try {\n        await this.run()\n      } catch (err) {} // eslint-disable-line no-empty\n    })()\n  }\n\n  /**\n   * Closes the ThreadPool for further task queueing, the ThreadPool's completion can be awaited afte'r it's called\n   */\n  close() {\n    this._closed = true\n    this._queuedTasks.close() // NOTE: NO AWAIT\n    // TODO: await completion\n  }\n\n  /**\n   * Closes the ThreadPool, runs it's tasks and awaits their completion\n   */\n  async runAllQueued(): Promise<Array<R>> {\n    this.close()\n    await this.run()\n    return this._res\n  }\n\n  /**\n   * Awaits the closing and completion of all ThreadPool tasks\n   */\n  async waitComplete(): Promise<Array<R>> {\n    await this._completeEvent.wait()\n    this._throwUncaughtErrors()\n    return this._res\n  }\n\n  /**\n   * Closes the ThreadPool and awaits the running and completion of all ThreadPool tasks\n   */\n  async closeAndWaitComplete(): Promise<Array<R>> {\n    this.close()\n    return await this.waitComplete()\n  }\n\n  /****************** privates ******************/\n\n  _throwUncaughtErrors() {   \n    if (this._uncaughtErrors.length > 0) {\n      throw new ExtendedError(`Errors were thrown during execution of ThreadPool`, {\n        threadPoolName: this._options.name,\n        errorCount: this._uncaughtErrors.length,\n        errorMessages: this._uncaughtErrors.map(e => e.message),\n        uncaughtErrors: this._uncaughtErrors,\n      })\n    }\n  }\n\n  async _runTask(task: Task<R>) {\n    try {\n      this.startedCount++\n      await task.func()\n    } catch (err) {\n      try {\n        this._errorHandler(err)\n      } catch (err2) {\n        this._uncaughtErrors.push(err2)\n        await this._allTasksCompleteOrSomeFailedEvent.signal()\n      }\n    } finally {\n      this.endedCount++          \n      // fire 'progress' event\n      this.emit('progress', { endedCount: this.endedCount })\n      this._threadsSemaphore.exit()\n      if (this._threadsSemaphore.takenCount === 0) {\n        this._allTasksCompleteOrSomeFailedEvent.signal()\n      }\n    }\n  }\n}\n"]}
\No newline at end of file