Update the license headers throughout the repository to reference Google LLC rather than Google Inc, for the required license headers. PR Close #37205
		
			
				
	
	
		
			214 lines
		
	
	
		
			7.6 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			214 lines
		
	
	
		
			7.6 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| /**
 | |
|  * @license
 | |
|  * Copyright Google LLC 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 {AbsoluteFsPath} from '../../../../src/ngtsc/file_system';
 | |
| import {CreateCompileFn} from '../../../src/execution/api';
 | |
| import {startWorker} from '../../../src/execution/cluster/worker';
 | |
| import {Task, TaskCompletedCallback, TaskProcessingOutcome} from '../../../src/execution/tasks/api';
 | |
| import {FileToWrite} from '../../../src/rendering/utils';
 | |
| import {MockLogger} from '../../helpers/mock_logger';
 | |
| import {mockProperty, spyProperty} from '../../helpers/spy_utils';
 | |
| 
 | |
| 
 | |
| describe('startWorker()', () => {
 | |
|   const runAsClusterMaster = mockProperty(cluster, 'isMaster');
 | |
|   const mockProcessSend = mockProperty(process, 'send');
 | |
|   let processSendSpy: jasmine.Spy;
 | |
|   let compileFnSpy: jasmine.Spy;
 | |
|   let createCompileFnSpy: jasmine.Spy;
 | |
|   let mockLogger: MockLogger;
 | |
| 
 | |
|   beforeEach(() => {
 | |
|     compileFnSpy = jasmine.createSpy('compileFn');
 | |
|     createCompileFnSpy = jasmine.createSpy('createCompileFn').and.returnValue(compileFnSpy);
 | |
| 
 | |
|     processSendSpy = jasmine.createSpy('process.send');
 | |
|     mockProcessSend(processSendSpy);
 | |
| 
 | |
|     mockLogger = new MockLogger();
 | |
|   });
 | |
| 
 | |
|   describe('(on cluster master)', () => {
 | |
|     beforeEach(() => runAsClusterMaster(true));
 | |
| 
 | |
|     it('should throw an error', async () => {
 | |
|       await expectAsync(startWorker(mockLogger, createCompileFnSpy))
 | |
|           .toBeRejectedWithError('Tried to run cluster worker on the master process.');
 | |
|       expect(createCompileFnSpy).not.toHaveBeenCalled();
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   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');
 | |
| 
 | |
|     beforeEach(() => {
 | |
|       runAsClusterMaster(false);
 | |
|       mockClusterWorker(Object.assign(new EventEmitter(), {id: 42}) as cluster.Worker);
 | |
|     });
 | |
| 
 | |
|     it('should create the `compileFn()`', () => {
 | |
|       startWorker(mockLogger, createCompileFnSpy);
 | |
|       expect(createCompileFnSpy).toHaveBeenCalledWith(jasmine.any(Function), jasmine.any(Function));
 | |
|     });
 | |
| 
 | |
|     it('should set up `compileFn()` to send `transformed-files` messages to master', () => {
 | |
|       startWorker(mockLogger, createCompileFnSpy);
 | |
| 
 | |
|       const mockTransformedFiles: FileToWrite[] = [
 | |
|         {path: '/foo' as AbsoluteFsPath, contents: 'FOO'},
 | |
|         {path: '/bar' as AbsoluteFsPath, contents: 'BAR'},
 | |
|       ];
 | |
|       const beforeWritingFiles: Parameters<CreateCompileFn>[0] =
 | |
|           createCompileFnSpy.calls.argsFor(0)[0];
 | |
| 
 | |
|       beforeWritingFiles(mockTransformedFiles);
 | |
| 
 | |
|       expect(processSendSpy).toHaveBeenCalledTimes(1);
 | |
|       expect(processSendSpy)
 | |
|           .toHaveBeenCalledWith(
 | |
|               {type: 'transformed-files', files: ['/foo', '/bar']}, jasmine.any(Function));
 | |
|     });
 | |
| 
 | |
|     it('should set up `compileFn()` to send `task-completed` messages to master', () => {
 | |
|       startWorker(mockLogger, createCompileFnSpy);
 | |
|       const onTaskCompleted: TaskCompletedCallback = createCompileFnSpy.calls.argsFor(0)[1];
 | |
| 
 | |
|       onTaskCompleted(null as any, TaskProcessingOutcome.Processed, null);
 | |
|       expect(processSendSpy).toHaveBeenCalledTimes(1);
 | |
|       expect(processSendSpy)
 | |
|           .toHaveBeenCalledWith(
 | |
|               {type: 'task-completed', outcome: TaskProcessingOutcome.Processed, message: null},
 | |
|               jasmine.any(Function));
 | |
| 
 | |
|       processSendSpy.calls.reset();
 | |
| 
 | |
|       onTaskCompleted(null as any, TaskProcessingOutcome.Failed, 'error message');
 | |
|       expect(processSendSpy).toHaveBeenCalledTimes(1);
 | |
|       expect(processSendSpy)
 | |
|           .toHaveBeenCalledWith(
 | |
|               {
 | |
|                 type: 'task-completed',
 | |
|                 outcome: TaskProcessingOutcome.Failed,
 | |
|                 message: 'error message',
 | |
|               },
 | |
|               jasmine.any(Function));
 | |
|     });
 | |
| 
 | |
|     it('should return a promise (that is never resolved)', done => {
 | |
|       const promise = startWorker(mockLogger, createCompileFnSpy);
 | |
| 
 | |
|       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 = {
 | |
|         entryPoint: {name: 'foo'},
 | |
|         formatProperty: 'es2015',
 | |
|         processDts: true,
 | |
|       } as unknown as Task;
 | |
| 
 | |
|       startWorker(mockLogger, createCompileFnSpy);
 | |
|       cluster.worker.emit('message', {type: 'process-task', task: mockTask});
 | |
| 
 | |
|       expect(compileFnSpy).toHaveBeenCalledWith(mockTask);
 | |
|       expect(processSendSpy).not.toHaveBeenCalled();
 | |
| 
 | |
|       expect(mockLogger.logs.debug[0]).toEqual([
 | |
|         '[Worker #42] Processing task: {entryPoint: foo, formatProperty: es2015, processDts: true}',
 | |
|       ]);
 | |
|     });
 | |
| 
 | |
|     it('should send errors during task processing back to the master process', () => {
 | |
|       const mockTask = {
 | |
|         entryPoint: {name: 'foo'},
 | |
|         formatProperty: 'es2015',
 | |
|         processDts: true,
 | |
|       } as unknown as Task;
 | |
| 
 | |
|       let err: string|Error;
 | |
|       compileFnSpy.and.callFake(() => {
 | |
|         throw err;
 | |
|       });
 | |
| 
 | |
|       startWorker(mockLogger, createCompileFnSpy);
 | |
| 
 | |
|       err = 'Error string.';
 | |
|       cluster.worker.emit('message', {type: 'process-task', task: mockTask});
 | |
|       expect(processSendSpy)
 | |
|           .toHaveBeenCalledWith({type: 'error', error: err}, jasmine.any(Function));
 | |
| 
 | |
|       err = new Error('Error object.');
 | |
|       cluster.worker.emit('message', {type: 'process-task', task: mockTask});
 | |
|       expect(processSendSpy)
 | |
|           .toHaveBeenCalledWith({type: 'error', error: err.stack}, jasmine.any(Function));
 | |
|     });
 | |
| 
 | |
|     it('should exit on `ENOMEM` errors during task processing', () => {
 | |
|       const processExitSpy = jasmine.createSpy('process.exit');
 | |
|       const {
 | |
|         setMockValue: mockProcessExit,
 | |
|         installSpies: installProcessExitSpies,
 | |
|         uninstallSpies: uninstallProcessExitSpies,
 | |
|       } = spyProperty(process, 'exit');
 | |
| 
 | |
|       try {
 | |
|         installProcessExitSpies();
 | |
|         mockProcessExit(processExitSpy as unknown as typeof process.exit);
 | |
| 
 | |
|         const mockTask = {
 | |
|           entryPoint: {name: 'foo'},
 | |
|           formatProperty: 'es2015',
 | |
|           processDts: true,
 | |
|         } as unknown as Task;
 | |
| 
 | |
|         const noMemError = Object.assign(new Error('ENOMEM: not enough memory'), {code: 'ENOMEM'});
 | |
|         compileFnSpy.and.throwError(noMemError);
 | |
| 
 | |
|         startWorker(mockLogger, createCompileFnSpy);
 | |
|         cluster.worker.emit('message', {type: 'process-task', task: mockTask});
 | |
| 
 | |
|         expect(mockLogger.logs.warn).toEqual([[`[Worker #42] ${noMemError.stack}`]]);
 | |
|         expect(processExitSpy).toHaveBeenCalledWith(1);
 | |
|         expect(processSendSpy).not.toHaveBeenCalled();
 | |
|       } finally {
 | |
|         uninstallProcessExitSpies();
 | |
|       }
 | |
|     });
 | |
| 
 | |
|     it('should throw, when an unknown message type is received', () => {
 | |
|       startWorker(mockLogger, createCompileFnSpy);
 | |
|       cluster.worker.emit('message', {type: 'unknown', foo: 'bar'});
 | |
| 
 | |
|       expect(compileFnSpy).not.toHaveBeenCalled();
 | |
|       expect(processSendSpy)
 | |
|           .toHaveBeenCalledWith(
 | |
|               {
 | |
|                 type: 'error',
 | |
|                 error: jasmine.stringMatching(
 | |
|                     'Error: \\[Worker #42\\] Invalid message received: {"type":"unknown","foo":"bar"}'),
 | |
|               },
 | |
|               jasmine.any(Function));
 | |
|     });
 | |
|   });
 | |
| });
 |