2019-08-29 18:47:54 +03:00
|
|
|
/**
|
|
|
|
|
* @license
|
|
|
|
|
* Copyright Google Inc. All Rights Reserved.
|
|
|
|
|
*
|
|
|
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
|
|
|
* found in the LICENSE file at https://angular.io/license
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/// <reference types="node" />
|
|
|
|
|
|
|
|
|
|
import * as cluster from 'cluster';
|
|
|
|
|
|
2019-12-05 21:02:57 +02:00
|
|
|
import {Logger} from '../../logging/logger';
|
2019-08-29 18:47:54 +03:00
|
|
|
import {CompileFn, CreateCompileFn} from '../api';
|
2020-03-14 13:38:27 +00:00
|
|
|
import {stringifyTask} from '../tasks/utils';
|
2019-08-29 18:47:54 +03:00
|
|
|
|
|
|
|
|
import {MessageToWorker} from './api';
|
|
|
|
|
import {sendMessageToMaster} from './utils';
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* A cluster worker is responsible for processing one task (i.e. one format property for a specific
|
|
|
|
|
* entry-point) at a time and reporting results back to the cluster master.
|
|
|
|
|
*/
|
|
|
|
|
export class ClusterWorker {
|
|
|
|
|
private compile: CompileFn;
|
|
|
|
|
|
2019-12-05 21:02:57 +02:00
|
|
|
constructor(private logger: Logger, createCompileFn: CreateCompileFn) {
|
2019-08-29 18:47:54 +03:00
|
|
|
if (cluster.isMaster) {
|
|
|
|
|
throw new Error('Tried to instantiate `ClusterWorker` on the master process.');
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-14 13:20:51 +00:00
|
|
|
this.compile = createCompileFn(
|
|
|
|
|
(_task, outcome, message) =>
|
|
|
|
|
sendMessageToMaster({type: 'task-completed', outcome, message}));
|
2019-08-29 18:47:54 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
run(): Promise<void> {
|
|
|
|
|
// Listen for `ProcessTaskMessage`s and process tasks.
|
|
|
|
|
cluster.worker.on('message', (msg: MessageToWorker) => {
|
|
|
|
|
try {
|
|
|
|
|
switch (msg.type) {
|
|
|
|
|
case 'process-task':
|
2019-12-05 21:02:57 +02:00
|
|
|
this.logger.debug(
|
|
|
|
|
`[Worker #${cluster.worker.id}] Processing task: ${stringifyTask(msg.task)}`);
|
2019-08-29 18:47:54 +03:00
|
|
|
return this.compile(msg.task);
|
|
|
|
|
default:
|
|
|
|
|
throw new Error(
|
2019-12-05 21:02:57 +02:00
|
|
|
`[Worker #${cluster.worker.id}] Invalid message received: ${JSON.stringify(msg)}`);
|
2019-08-29 18:47:54 +03:00
|
|
|
}
|
|
|
|
|
} catch (err) {
|
|
|
|
|
sendMessageToMaster({
|
|
|
|
|
type: 'error',
|
|
|
|
|
error: (err instanceof Error) ? (err.stack || err.message) : err,
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// Return a promise that is never resolved.
|
|
|
|
|
return new Promise(() => undefined);
|
|
|
|
|
}
|
|
|
|
|
}
|