145 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
		
		
			
		
	
	
			145 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * @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';
							 | 
						||
| 
								 | 
							
								import {EventEmitter} from 'events';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								import {Task, TaskCompletedCallback, TaskProcessingOutcome} from '../../../src/execution/api';
							 | 
						||
| 
								 | 
							
								import {ClusterWorker} from '../../../src/execution/cluster/worker';
							 | 
						||
| 
								 | 
							
								import {mockProperty} from '../../helpers/spy_utils';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								describe('ClusterWorker', () => {
							 | 
						||
| 
								 | 
							
								  const runAsClusterMaster = mockProperty(cluster, 'isMaster');
							 | 
						||
| 
								 | 
							
								  const mockProcessSend = mockProperty(process, 'send');
							 | 
						||
| 
								 | 
							
								  let processSendSpy: jasmine.Spy;
							 | 
						||
| 
								 | 
							
								  let compileFnSpy: jasmine.Spy;
							 | 
						||
| 
								 | 
							
								  let createCompileFnSpy: jasmine.Spy;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  beforeEach(() => {
							 | 
						||
| 
								 | 
							
								    compileFnSpy = jasmine.createSpy('compileFn');
							 | 
						||
| 
								 | 
							
								    createCompileFnSpy = jasmine.createSpy('createCompileFn').and.returnValue(compileFnSpy);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    processSendSpy = jasmine.createSpy('process.send');
							 | 
						||
| 
								 | 
							
								    mockProcessSend(processSendSpy);
							 | 
						||
| 
								 | 
							
								  });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  describe('constructor()', () => {
							 | 
						||
| 
								 | 
							
								    describe('(on cluster master)', () => {
							 | 
						||
| 
								 | 
							
								      beforeEach(() => runAsClusterMaster(true));
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      it('should throw an error', () => {
							 | 
						||
| 
								 | 
							
								        expect(() => new ClusterWorker(createCompileFnSpy))
							 | 
						||
| 
								 | 
							
								            .toThrowError('Tried to instantiate `ClusterWorker` on the master process.');
							 | 
						||
| 
								 | 
							
								        expect(createCompileFnSpy).not.toHaveBeenCalled();
							 | 
						||
| 
								 | 
							
								      });
							 | 
						||
| 
								 | 
							
								    });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    describe('(on cluster worker)', () => {
							 | 
						||
| 
								 | 
							
								      beforeEach(() => runAsClusterMaster(false));
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      it('should create the `compileFn()`', () => {
							 | 
						||
| 
								 | 
							
								        new ClusterWorker(createCompileFnSpy);
							 | 
						||
| 
								 | 
							
								        expect(createCompileFnSpy).toHaveBeenCalledWith(jasmine.any(Function));
							 | 
						||
| 
								 | 
							
								      });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      it('should set up `compileFn()` to send a `task-completed` message to master', () => {
							 | 
						||
| 
								 | 
							
								        new ClusterWorker(createCompileFnSpy);
							 | 
						||
| 
								 | 
							
								        const onTaskCompleted: TaskCompletedCallback = createCompileFnSpy.calls.argsFor(0)[0];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        onTaskCompleted(null as any, TaskProcessingOutcome.AlreadyProcessed);
							 | 
						||
| 
								 | 
							
								        expect(processSendSpy).toHaveBeenCalledTimes(1);
							 | 
						||
| 
								 | 
							
								        expect(processSendSpy).toHaveBeenCalledWith({
							 | 
						||
| 
								 | 
							
								          type: 'task-completed',
							 | 
						||
| 
								 | 
							
								          outcome: TaskProcessingOutcome.AlreadyProcessed,
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        onTaskCompleted(null as any, TaskProcessingOutcome.Processed);
							 | 
						||
| 
								 | 
							
								        expect(processSendSpy).toHaveBeenCalledTimes(2);
							 | 
						||
| 
								 | 
							
								        expect(processSendSpy).toHaveBeenCalledWith({
							 | 
						||
| 
								 | 
							
								          type: 'task-completed',
							 | 
						||
| 
								 | 
							
								          outcome: TaskProcessingOutcome.Processed,
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								      });
							 | 
						||
| 
								 | 
							
								    });
							 | 
						||
| 
								 | 
							
								  });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  describe('run()', () => {
							 | 
						||
| 
								 | 
							
								    describe(
							 | 
						||
| 
								 | 
							
								        '(on cluster master)',
							 | 
						||
| 
								 | 
							
								        () => {/* No tests needed, becasue the constructor would have thrown. */});
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    describe('(on cluster worker)', () => {
							 | 
						||
| 
								 | 
							
								      // The `cluster.worker` property is normally `undefined` on the master process and set to the
							 | 
						||
| 
								 | 
							
								      // current `cluster.Worker` on worker processes.
							 | 
						||
| 
								 | 
							
								      const mockClusterWorker = mockProperty(cluster, 'worker');
							 | 
						||
| 
								 | 
							
								      let worker: ClusterWorker;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      beforeEach(() => {
							 | 
						||
| 
								 | 
							
								        runAsClusterMaster(false);
							 | 
						||
| 
								 | 
							
								        mockClusterWorker(Object.assign(new EventEmitter(), {id: 42}) as cluster.Worker);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        worker = new ClusterWorker(createCompileFnSpy);
							 | 
						||
| 
								 | 
							
								      });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      it('should return a promise (that is never resolved)', done => {
							 | 
						||
| 
								 | 
							
								        const promise = worker.run();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        expect(promise).toEqual(jasmine.any(Promise));
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        promise.then(
							 | 
						||
| 
								 | 
							
								            () => done.fail('Expected promise not to resolve'),
							 | 
						||
| 
								 | 
							
								            () => done.fail('Expected promise not to reject'));
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        // We can't wait forever to verify that the promise is not resolved, but at least verify
							 | 
						||
| 
								 | 
							
								        // that it is not resolved immediately.
							 | 
						||
| 
								 | 
							
								        setTimeout(done, 100);
							 | 
						||
| 
								 | 
							
								      });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      it('should handle `process-task` messages', () => {
							 | 
						||
| 
								 | 
							
								        const mockTask = { foo: 'bar' } as unknown as Task;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        worker.run();
							 | 
						||
| 
								 | 
							
								        cluster.worker.emit('message', {type: 'process-task', task: mockTask});
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        expect(compileFnSpy).toHaveBeenCalledWith(mockTask);
							 | 
						||
| 
								 | 
							
								        expect(processSendSpy).not.toHaveBeenCalled();
							 | 
						||
| 
								 | 
							
								      });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      it('should send errors during task processing back to the master process', () => {
							 | 
						||
| 
								 | 
							
								        let err: string|Error;
							 | 
						||
| 
								 | 
							
								        compileFnSpy.and.callFake(() => { throw err; });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        worker.run();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        err = 'Error string.';
							 | 
						||
| 
								 | 
							
								        cluster.worker.emit('message', {type: 'process-task', task: {} as Task});
							 | 
						||
| 
								 | 
							
								        expect(processSendSpy).toHaveBeenCalledWith({type: 'error', error: err});
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        err = new Error('Error object.');
							 | 
						||
| 
								 | 
							
								        cluster.worker.emit('message', {type: 'process-task', task: {} as Task});
							 | 
						||
| 
								 | 
							
								        expect(processSendSpy).toHaveBeenCalledWith({type: 'error', error: err.stack});
							 | 
						||
| 
								 | 
							
								      });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      it('should throw, when an unknown message type is received', () => {
							 | 
						||
| 
								 | 
							
								        worker.run();
							 | 
						||
| 
								 | 
							
								        cluster.worker.emit('message', {type: 'unknown', foo: 'bar'});
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        expect(compileFnSpy).not.toHaveBeenCalled();
							 | 
						||
| 
								 | 
							
								        expect(processSendSpy).toHaveBeenCalledWith({
							 | 
						||
| 
								 | 
							
								          type: 'error',
							 | 
						||
| 
								 | 
							
								          error: jasmine.stringMatching(
							 | 
						||
| 
								 | 
							
								              'Error: Invalid message received on worker #42: {"type":"unknown","foo":"bar"}'),
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								      });
							 | 
						||
| 
								 | 
							
								    });
							 | 
						||
| 
								 | 
							
								  });
							 | 
						||
| 
								 | 
							
								});
							 |