@parcel/workers 2.0.0-nightly.150 → 2.0.0-nightly.1500

Sign up to get free protection for your applications and to get access to all the features.
Files changed (44) hide show
  1. package/index.d.ts +23 -0
  2. package/lib/Handle.js +16 -58
  3. package/lib/Worker.js +103 -62
  4. package/lib/WorkerFarm.js +272 -192
  5. package/lib/backend.js +4 -6
  6. package/lib/bus.js +11 -13
  7. package/lib/child.js +140 -116
  8. package/lib/childState.js +2 -4
  9. package/lib/core-worker.browser.js +4 -0
  10. package/lib/core-worker.js +4 -0
  11. package/lib/cpuCount.js +36 -25
  12. package/lib/index.js +35 -32
  13. package/lib/process/ProcessChild.js +18 -24
  14. package/lib/process/ProcessWorker.js +27 -38
  15. package/lib/threads/ThreadsChild.js +26 -28
  16. package/lib/threads/ThreadsWorker.js +25 -31
  17. package/lib/web/WebChild.js +44 -0
  18. package/lib/web/WebWorker.js +85 -0
  19. package/package.json +19 -8
  20. package/src/Handle.js +10 -39
  21. package/src/Worker.js +95 -22
  22. package/src/WorkerFarm.js +267 -62
  23. package/src/backend.js +5 -0
  24. package/src/bus.js +3 -2
  25. package/src/child.js +95 -26
  26. package/src/core-worker.browser.js +3 -0
  27. package/src/core-worker.js +2 -0
  28. package/src/cpuCount.js +23 -10
  29. package/src/index.js +8 -2
  30. package/src/process/ProcessChild.js +2 -1
  31. package/src/process/ProcessWorker.js +1 -1
  32. package/src/threads/ThreadsWorker.js +2 -2
  33. package/src/types.js +1 -1
  34. package/src/web/WebChild.js +50 -0
  35. package/src/web/WebWorker.js +85 -0
  36. package/test/cpuCount.test.js +1 -1
  37. package/test/integration/workerfarm/console.js +1 -1
  38. package/test/integration/workerfarm/logging.js +1 -1
  39. package/test/integration/workerfarm/reverse-handle.js +2 -2
  40. package/test/workerfarm.js +5 -5
  41. package/lib/Profiler.js +0 -70
  42. package/lib/Trace.js +0 -126
  43. package/src/Profiler.js +0 -93
  44. package/src/Trace.js +0 -121
package/src/WorkerFarm.js CHANGED
@@ -11,6 +11,8 @@ import type {
11
11
  } from './types';
12
12
  import type {HandleFunction} from './Handle';
13
13
 
14
+ import * as coreWorker from './core-worker';
15
+ import * as bus from './bus';
14
16
  import invariant from 'assert';
15
17
  import nullthrows from 'nullthrows';
16
18
  import EventEmitter from 'events';
@@ -20,20 +22,20 @@ import {
20
22
  restoreDeserializedObject,
21
23
  serialize,
22
24
  } from '@parcel/core';
23
- import ThrowableDiagnostic, {anyToDiagnostic} from '@parcel/diagnostic';
25
+ import ThrowableDiagnostic, {anyToDiagnostic, md} from '@parcel/diagnostic';
24
26
  import Worker, {type WorkerCall} from './Worker';
25
27
  import cpuCount from './cpuCount';
26
28
  import Handle from './Handle';
27
29
  import {child} from './childState';
28
30
  import {detectBackend} from './backend';
29
- import Profiler from './Profiler';
30
- import Trace from './Trace';
31
+ import {SamplingProfiler, Trace} from '@parcel/profiler';
31
32
  import fs from 'fs';
32
33
  import logger from '@parcel/logger';
33
34
 
34
- let profileId = 1;
35
35
  let referenceId = 1;
36
36
 
37
+ export opaque type SharedReference = number;
38
+
37
39
  export type FarmOptions = {|
38
40
  maxConcurrentWorkers: number,
39
41
  maxConcurrentCallsPerWorker: number,
@@ -42,23 +44,27 @@ export type FarmOptions = {|
42
44
  warmWorkers: boolean,
43
45
  workerPath?: FilePath,
44
46
  backend: BackendType,
45
- patchConsole?: boolean,
47
+ shouldPatchConsole?: boolean,
48
+ shouldTrace?: boolean,
46
49
  |};
47
50
 
48
- type WorkerModule = {|
51
+ type WorkerModule = {
49
52
  +[string]: (...args: Array<mixed>) => Promise<mixed>,
50
- |};
53
+ ...
54
+ };
51
55
 
52
56
  export type WorkerApi = {|
53
57
  callMaster(CallRequest, ?boolean): Promise<mixed>,
54
58
  createReverseHandle(fn: HandleFunction): Handle,
55
- getSharedReference(ref: number): mixed,
56
- resolveSharedReference(value: mixed): ?number,
59
+ getSharedReference(ref: SharedReference): mixed,
60
+ resolveSharedReference(value: mixed): ?SharedReference,
57
61
  callChild?: (childId: number, request: HandleCallRequest) => Promise<mixed>,
58
62
  |};
59
63
 
60
64
  export {Handle};
61
65
 
66
+ const DEFAULT_MAX_CONCURRENT_CALLS: number = 30;
67
+
62
68
  /**
63
69
  * workerPath should always be defined inside farmOptions
64
70
  */
@@ -67,20 +73,25 @@ export default class WorkerFarm extends EventEmitter {
67
73
  callQueue: Array<WorkerCall> = [];
68
74
  ending: boolean = false;
69
75
  localWorker: WorkerModule;
76
+ localWorkerInit: ?Promise<void>;
70
77
  options: FarmOptions;
71
78
  run: HandleFunction;
72
79
  warmWorkers: number = 0;
80
+ readyWorkers: number = 0;
73
81
  workers: Map<number, Worker> = new Map();
74
82
  handles: Map<number, Handle> = new Map();
75
- sharedReferences: Map<number, mixed> = new Map();
76
- sharedReferencesByValue: Map<mixed, number> = new Map();
77
- profiler: ?Profiler;
83
+ sharedReferences: Map<SharedReference, mixed> = new Map();
84
+ sharedReferencesByValue: Map<mixed, SharedReference> = new Map();
85
+ serializedSharedReferences: Map<SharedReference, ?ArrayBuffer> = new Map();
86
+ profiler: ?SamplingProfiler;
78
87
 
79
88
  constructor(farmOptions: $Shape<FarmOptions> = {}) {
80
89
  super();
81
90
  this.options = {
82
91
  maxConcurrentWorkers: WorkerFarm.getNumWorkers(),
83
- maxConcurrentCallsPerWorker: WorkerFarm.getConcurrentCallsPerWorker(),
92
+ maxConcurrentCallsPerWorker: WorkerFarm.getConcurrentCallsPerWorker(
93
+ farmOptions.shouldTrace ? 1 : DEFAULT_MAX_CONCURRENT_CALLS,
94
+ ),
84
95
  forcedKillTime: 500,
85
96
  warmWorkers: false,
86
97
  useLocalWorker: true, // TODO: setting this to false makes some tests fail, figure out why
@@ -92,14 +103,54 @@ export default class WorkerFarm extends EventEmitter {
92
103
  throw new Error('Please provide a worker path!');
93
104
  }
94
105
 
95
- // $FlowFixMe this must be dynamic
96
- this.localWorker = require(this.options.workerPath);
106
+ // $FlowFixMe
107
+ if (process.browser) {
108
+ if (this.options.workerPath === '@parcel/core/src/worker.js') {
109
+ this.localWorker = coreWorker;
110
+ } else {
111
+ throw new Error(
112
+ 'No dynamic require possible: ' + this.options.workerPath,
113
+ );
114
+ }
115
+ } else {
116
+ // $FlowFixMe this must be dynamic
117
+ this.localWorker = require(this.options.workerPath);
118
+ }
119
+
120
+ this.localWorkerInit =
121
+ this.localWorker.childInit != null ? this.localWorker.childInit() : null;
122
+
97
123
  this.run = this.createHandle('run');
98
124
 
125
+ // Worker thread stdout is by default piped into the process stdout, if there are enough worker
126
+ // threads to exceed the default listener limit, then anything else piping into stdout will trigger
127
+ // the `MaxListenersExceededWarning`, so we should ensure the max listeners is at least equal to the
128
+ // number of workers + 1 for the main thread.
129
+ //
130
+ // Note this can't be fixed easily where other things pipe into stdout - even after starting > 10 worker
131
+ // threads `process.stdout.getMaxListeners()` will still return 10, however adding another pipe into `stdout`
132
+ // will give the warning with `<worker count + 1>` as the number of listeners.
133
+ process.stdout?.setMaxListeners(
134
+ Math.max(
135
+ process.stdout.getMaxListeners(),
136
+ WorkerFarm.getNumWorkers() + 1,
137
+ ),
138
+ );
139
+
99
140
  this.startMaxWorkers();
100
141
  }
101
142
 
102
- workerApi = {
143
+ workerApi: {|
144
+ callChild: (childId: number, request: HandleCallRequest) => Promise<mixed>,
145
+ callMaster: (
146
+ request: CallRequest,
147
+ awaitResponse?: ?boolean,
148
+ ) => Promise<mixed>,
149
+ createReverseHandle: (fn: HandleFunction) => Handle,
150
+ getSharedReference: (ref: SharedReference) => mixed,
151
+ resolveSharedReference: (value: mixed) => void | SharedReference,
152
+ runHandle: (handle: Handle, args: Array<any>) => Promise<mixed>,
153
+ |} = {
103
154
  callMaster: async (
104
155
  request: CallRequest,
105
156
  awaitResponse: ?boolean = true,
@@ -122,7 +173,13 @@ export default class WorkerFarm extends EventEmitter {
122
173
  retries: 0,
123
174
  });
124
175
  }),
125
- getSharedReference: (ref: number) => this.sharedReferences.get(ref),
176
+ runHandle: (handle: Handle, args: Array<any>): Promise<mixed> =>
177
+ this.workerApi.callChild(nullthrows(handle.childId), {
178
+ handle: handle.id,
179
+ args,
180
+ }),
181
+ getSharedReference: (ref: SharedReference) =>
182
+ this.sharedReferences.get(ref),
126
183
  resolveSharedReference: (value: mixed) =>
127
184
  this.sharedReferencesByValue.get(value),
128
185
  };
@@ -155,30 +212,46 @@ export default class WorkerFarm extends EventEmitter {
155
212
  );
156
213
  }
157
214
 
158
- createHandle(method: string): HandleFunction {
159
- return (...args) => {
215
+ createHandle(method: string, useMainThread: boolean = false): HandleFunction {
216
+ if (!this.options.useLocalWorker) {
217
+ useMainThread = false;
218
+ }
219
+
220
+ return async (...args) => {
160
221
  // Child process workers are slow to start (~600ms).
161
222
  // While we're waiting, just run on the main thread.
162
223
  // This significantly speeds up startup time.
163
- if (this.shouldUseRemoteWorkers()) {
224
+ if (this.shouldUseRemoteWorkers() && !useMainThread) {
164
225
  return this.addCall(method, [...args, false]);
165
226
  } else {
166
227
  if (this.options.warmWorkers && this.shouldStartRemoteWorkers()) {
167
228
  this.warmupWorker(method, args);
168
229
  }
169
230
 
170
- let processedArgs = restoreDeserializedObject(
171
- prepareForSerialization([...args, false]),
172
- );
231
+ let processedArgs;
232
+ if (!useMainThread) {
233
+ processedArgs = restoreDeserializedObject(
234
+ prepareForSerialization([...args, false]),
235
+ );
236
+ } else {
237
+ processedArgs = args;
238
+ }
239
+
240
+ if (this.localWorkerInit != null) {
241
+ await this.localWorkerInit;
242
+ this.localWorkerInit = null;
243
+ }
173
244
  return this.localWorker[method](this.workerApi, ...processedArgs);
174
245
  }
175
246
  };
176
247
  }
177
248
 
178
- onError(error: ErrorWithCode, worker: Worker) {
249
+ onError(error: ErrorWithCode, worker: Worker): void | Promise<void> {
179
250
  // Handle ipc errors
180
251
  if (error.code === 'ERR_IPC_CHANNEL_CLOSED') {
181
252
  return this.stopWorker(worker);
253
+ } else {
254
+ logger.error(error, '@parcel/workers');
182
255
  }
183
256
  }
184
257
 
@@ -186,14 +259,22 @@ export default class WorkerFarm extends EventEmitter {
186
259
  let worker = new Worker({
187
260
  forcedKillTime: this.options.forcedKillTime,
188
261
  backend: this.options.backend,
189
- patchConsole: this.options.patchConsole,
262
+ shouldPatchConsole: this.options.shouldPatchConsole,
263
+ shouldTrace: this.options.shouldTrace,
264
+ sharedReferences: this.sharedReferences,
190
265
  });
191
266
 
192
267
  worker.fork(nullthrows(this.options.workerPath));
193
268
 
194
269
  worker.on('request', data => this.processRequest(data, worker));
195
270
 
196
- worker.on('ready', () => this.processQueue());
271
+ worker.on('ready', () => {
272
+ this.readyWorkers++;
273
+ if (this.readyWorkers === this.options.maxConcurrentWorkers) {
274
+ this.emit('ready');
275
+ }
276
+ this.processQueue();
277
+ });
197
278
  worker.on('response', () => this.processQueue());
198
279
 
199
280
  worker.on('error', err => this.onError(err, worker));
@@ -231,7 +312,11 @@ export default class WorkerFarm extends EventEmitter {
231
312
  this.startChild();
232
313
  }
233
314
 
234
- for (let worker of this.workers.values()) {
315
+ let workers = [...this.workers.values()].sort(
316
+ (a, b) => a.calls.size - b.calls.size,
317
+ );
318
+
319
+ for (let worker of workers) {
235
320
  if (!this.callQueue.length) {
236
321
  break;
237
322
  }
@@ -241,9 +326,22 @@ export default class WorkerFarm extends EventEmitter {
241
326
  }
242
327
 
243
328
  if (worker.calls.size < this.options.maxConcurrentCallsPerWorker) {
244
- worker.call(this.callQueue.shift());
329
+ this.callWorker(worker, this.callQueue.shift());
330
+ }
331
+ }
332
+ }
333
+
334
+ async callWorker(worker: Worker, call: WorkerCall): Promise<void> {
335
+ for (let ref of this.sharedReferences.keys()) {
336
+ if (!worker.sentSharedReferences.has(ref)) {
337
+ await worker.sendSharedReference(
338
+ ref,
339
+ this.getSerializedSharedReference(ref),
340
+ );
245
341
  }
246
342
  }
343
+
344
+ worker.call(call);
247
345
  }
248
346
 
249
347
  async processRequest(
@@ -255,10 +353,19 @@ export default class WorkerFarm extends EventEmitter {
255
353
  let {method, args, location, awaitResponse, idx, handle: handleId} = data;
256
354
  let mod;
257
355
  if (handleId != null) {
258
- mod = nullthrows(this.handles.get(handleId)).fn;
356
+ mod = nullthrows(this.handles.get(handleId)?.fn);
259
357
  } else if (location) {
260
- // $FlowFixMe this must be dynamic
261
- mod = require(location);
358
+ // $FlowFixMe
359
+ if (process.browser) {
360
+ if (location === '@parcel/workers/src/bus.js') {
361
+ mod = (bus: any);
362
+ } else {
363
+ throw new Error('No dynamic require possible: ' + location);
364
+ }
365
+ } else {
366
+ // $FlowFixMe this must be dynamic
367
+ mod = require(location);
368
+ }
262
369
  } else {
263
370
  throw new Error('Unknown request');
264
371
  }
@@ -286,7 +393,6 @@ export default class WorkerFarm extends EventEmitter {
286
393
  }
287
394
  } else {
288
395
  // ESModule default interop
289
- // $FlowFixMe
290
396
  if (mod.__esModule && !mod[method] && mod.default) {
291
397
  mod = mod.default;
292
398
  }
@@ -331,6 +437,10 @@ export default class WorkerFarm extends EventEmitter {
331
437
  async end(): Promise<void> {
332
438
  this.ending = true;
333
439
 
440
+ await Promise.all(
441
+ Array.from(this.workers.values()).map(worker => this.stopWorker(worker)),
442
+ );
443
+
334
444
  for (let handle of this.handles.values()) {
335
445
  handle.dispose();
336
446
  }
@@ -338,9 +448,6 @@ export default class WorkerFarm extends EventEmitter {
338
448
  this.sharedReferences = new Map();
339
449
  this.sharedReferencesByValue = new Map();
340
450
 
341
- await Promise.all(
342
- Array.from(this.workers.values()).map(worker => this.stopWorker(worker)),
343
- );
344
451
  this.ending = false;
345
452
  }
346
453
 
@@ -362,40 +469,37 @@ export default class WorkerFarm extends EventEmitter {
362
469
  );
363
470
  }
364
471
 
365
- createReverseHandle(fn: HandleFunction) {
366
- let handle = new Handle({fn, workerApi: this.workerApi});
472
+ createReverseHandle(fn: HandleFunction): Handle {
473
+ let handle = new Handle({fn});
367
474
  this.handles.set(handle.id, handle);
368
475
  return handle;
369
476
  }
370
477
 
371
- async createSharedReference(value: mixed) {
478
+ createSharedReference(
479
+ value: mixed,
480
+ isCacheable: boolean = true,
481
+ ): {|ref: SharedReference, dispose(): Promise<mixed>|} {
372
482
  let ref = referenceId++;
373
483
  this.sharedReferences.set(ref, value);
374
484
  this.sharedReferencesByValue.set(value, ref);
375
- let promises = [];
376
- for (let worker of this.workers.values()) {
377
- promises.push(
378
- new Promise((resolve, reject) => {
379
- worker.call({
380
- method: 'createSharedReference',
381
- args: [ref, value],
382
- resolve,
383
- reject,
384
- retries: 0,
385
- });
386
- }),
387
- );
485
+ if (!isCacheable) {
486
+ this.serializedSharedReferences.set(ref, null);
388
487
  }
389
488
 
390
- await Promise.all(promises);
391
-
392
489
  return {
393
490
  ref,
394
491
  dispose: () => {
395
492
  this.sharedReferences.delete(ref);
396
493
  this.sharedReferencesByValue.delete(value);
494
+ this.serializedSharedReferences.delete(ref);
495
+
397
496
  let promises = [];
398
497
  for (let worker of this.workers.values()) {
498
+ if (!worker.sentSharedReferences.has(ref)) {
499
+ continue;
500
+ }
501
+
502
+ worker.sentSharedReferences.delete(ref);
399
503
  promises.push(
400
504
  new Promise((resolve, reject) => {
401
505
  worker.call({
@@ -403,6 +507,7 @@ export default class WorkerFarm extends EventEmitter {
403
507
  args: [ref],
404
508
  resolve,
405
509
  reject,
510
+ skipReadyCheck: true,
406
511
  retries: 0,
407
512
  });
408
513
  }),
@@ -413,6 +518,24 @@ export default class WorkerFarm extends EventEmitter {
413
518
  };
414
519
  }
415
520
 
521
+ getSerializedSharedReference(ref: SharedReference): ArrayBuffer {
522
+ let cached = this.serializedSharedReferences.get(ref);
523
+ if (cached) {
524
+ return cached;
525
+ }
526
+
527
+ let value = this.sharedReferences.get(ref);
528
+ let buf = serialize(value).buffer;
529
+
530
+ // If the reference was created with the isCacheable option set to false,
531
+ // serializedSharedReferences will contain `null` as the value.
532
+ if (cached !== null) {
533
+ this.serializedSharedReferences.set(ref, buf);
534
+ }
535
+
536
+ return buf;
537
+ }
538
+
416
539
  async startProfile() {
417
540
  let promises = [];
418
541
  for (let worker of this.workers.values()) {
@@ -424,12 +547,13 @@ export default class WorkerFarm extends EventEmitter {
424
547
  resolve,
425
548
  reject,
426
549
  retries: 0,
550
+ skipReadyCheck: true,
427
551
  });
428
552
  }),
429
553
  );
430
554
  }
431
555
 
432
- this.profiler = new Profiler();
556
+ this.profiler = new SamplingProfiler();
433
557
 
434
558
  promises.push(this.profiler.startProfiling());
435
559
  await Promise.all(promises);
@@ -453,6 +577,7 @@ export default class WorkerFarm extends EventEmitter {
453
577
  resolve,
454
578
  reject,
455
579
  retries: 0,
580
+ skipReadyCheck: true,
456
581
  });
457
582
  }),
458
583
  );
@@ -460,7 +585,7 @@ export default class WorkerFarm extends EventEmitter {
460
585
 
461
586
  var profiles = await Promise.all(promises);
462
587
  let trace = new Trace();
463
- let filename = `profile-${profileId++}.trace`;
588
+ let filename = `profile-${getTimeId()}.trace`;
464
589
  let stream = trace.pipe(fs.createWriteStream(filename));
465
590
 
466
591
  for (let profile of profiles) {
@@ -474,21 +599,84 @@ export default class WorkerFarm extends EventEmitter {
474
599
 
475
600
  logger.info({
476
601
  origin: '@parcel/workers',
477
- message: `Wrote profile to ${filename}`,
602
+ message: md`Wrote profile to ${filename}`,
478
603
  });
479
604
  }
480
605
 
481
- static getNumWorkers() {
606
+ async callAllWorkers(method: string, args: Array<any>) {
607
+ let promises = [];
608
+ for (let worker of this.workers.values()) {
609
+ promises.push(
610
+ new Promise((resolve, reject) => {
611
+ worker.call({
612
+ method,
613
+ args,
614
+ resolve,
615
+ reject,
616
+ retries: 0,
617
+ });
618
+ }),
619
+ );
620
+ }
621
+
622
+ promises.push(this.localWorker[method](this.workerApi, ...args));
623
+ await Promise.all(promises);
624
+ }
625
+
626
+ async takeHeapSnapshot() {
627
+ let snapshotId = getTimeId();
628
+
629
+ try {
630
+ let snapshotPaths = await Promise.all(
631
+ [...this.workers.values()].map(
632
+ worker =>
633
+ new Promise((resolve, reject) => {
634
+ worker.call({
635
+ method: 'takeHeapSnapshot',
636
+ args: [snapshotId],
637
+ resolve,
638
+ reject,
639
+ retries: 0,
640
+ skipReadyCheck: true,
641
+ });
642
+ }),
643
+ ),
644
+ );
645
+
646
+ logger.info({
647
+ origin: '@parcel/workers',
648
+ message: md`Wrote heap snapshots to the following paths:\n${snapshotPaths.join(
649
+ '\n',
650
+ )}`,
651
+ });
652
+ } catch {
653
+ logger.error({
654
+ origin: '@parcel/workers',
655
+ message: 'Unable to take heap snapshots. Note: requires Node 11.13.0+',
656
+ });
657
+ }
658
+ }
659
+
660
+ static getNumWorkers(): number {
482
661
  return process.env.PARCEL_WORKERS
483
662
  ? parseInt(process.env.PARCEL_WORKERS, 10)
484
- : cpuCount();
663
+ : Math.min(4, Math.ceil(cpuCount() / 2));
485
664
  }
486
665
 
487
- static isWorker() {
666
+ static isWorker(): boolean {
488
667
  return !!child;
489
668
  }
490
669
 
491
- static getWorkerApi() {
670
+ static getWorkerApi(): {|
671
+ callMaster: (
672
+ request: CallRequest,
673
+ awaitResponse?: ?boolean,
674
+ ) => Promise<mixed>,
675
+ createReverseHandle: (fn: (...args: Array<any>) => mixed) => Handle,
676
+ getSharedReference: (ref: SharedReference) => mixed,
677
+ resolveSharedReference: (value: mixed) => void | SharedReference,
678
+ runHandle: (handle: Handle, args: Array<any>) => Promise<mixed>,
679
+ |} {
492
680
  invariant(
493
681
  child != null,
494
682
  'WorkerFarm.getWorkerApi can only be called within workers',
@@ -496,7 +684,24 @@ export default class WorkerFarm extends EventEmitter {
496
684
  return child.workerApi;
497
685
  }
498
686
 
499
- static getConcurrentCallsPerWorker() {
500
- return parseInt(process.env.PARCEL_MAX_CONCURRENT_CALLS, 10) || 5;
687
+ static getConcurrentCallsPerWorker(
688
+ defaultValue?: number = DEFAULT_MAX_CONCURRENT_CALLS,
689
+ ): number {
690
+ return (
691
+ parseInt(process.env.PARCEL_MAX_CONCURRENT_CALLS, 10) || defaultValue
692
+ );
501
693
  }
502
694
  }
695
+
696
+ function getTimeId() {
697
+ let now = new Date();
698
+ return (
699
+ String(now.getFullYear()) +
700
+ String(now.getMonth() + 1).padStart(2, '0') +
701
+ String(now.getDate()).padStart(2, '0') +
702
+ '-' +
703
+ String(now.getHours()).padStart(2, '0') +
704
+ String(now.getMinutes()).padStart(2, '0') +
705
+ String(now.getSeconds()).padStart(2, '0')
706
+ );
707
+ }
package/src/backend.js CHANGED
@@ -2,6 +2,9 @@
2
2
  import type {BackendType, WorkerImpl} from './types';
3
3
 
4
4
  export function detectBackend(): BackendType {
5
+ // $FlowFixMe
6
+ if (process.browser) return 'web';
7
+
5
8
  switch (process.env.PARCEL_WORKER_BACKEND) {
6
9
  case 'threads':
7
10
  case 'process':
@@ -22,6 +25,8 @@ export function getWorkerBackend(backend: BackendType): Class<WorkerImpl> {
22
25
  return require('./threads/ThreadsWorker').default;
23
26
  case 'process':
24
27
  return require('./process/ProcessWorker').default;
28
+ case 'web':
29
+ return require('./web/WebWorker').default;
25
30
  default:
26
31
  throw new Error(`Invalid backend: ${backend}`);
27
32
  }
package/src/bus.js CHANGED
@@ -7,7 +7,8 @@ class Bus extends EventEmitter {
7
7
  if (child) {
8
8
  child.workerApi.callMaster(
9
9
  {
10
- location: __filename,
10
+ // $FlowFixMe
11
+ location: process.browser ? '@parcel/workers/src/bus.js' : __filename,
11
12
  method: 'emit',
12
13
  args: [event, ...args],
13
14
  },
@@ -20,4 +21,4 @@ class Bus extends EventEmitter {
20
21
  }
21
22
  }
22
23
 
23
- export default new Bus();
24
+ export default (new Bus(): Bus);