1 | ;
|
2 |
|
3 | Object.defineProperty(exports, "__esModule", {
|
4 | value: true
|
5 | });
|
6 | exports.default = void 0;
|
7 |
|
8 | var _events = _interopRequireDefault(require("events"));
|
9 |
|
10 | var _autoBind = _interopRequireDefault(require("auto-bind"));
|
11 |
|
12 | var _AsyncQueue = _interopRequireWildcard(require("./AsyncQueue"));
|
13 |
|
14 | var _asyncSyncUtils = require("./asyncSyncUtils");
|
15 |
|
16 | var _errorUtils = require("./errorUtils");
|
17 |
|
18 | function _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 |
|
20 | function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
21 |
|
22 | function _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 | */
|
31 | class 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 |
|
276 | exports.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 |