2016-11-30 16:59:53 -05: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
|
|
|
|
*/
|
|
|
|
|
2017-02-17 15:55:55 -05:00
|
|
|
import {ɵReflectionCapabilities, ɵreflector} from '@angular/core';
|
2016-11-30 16:59:53 -05:00
|
|
|
import {makeTempDir} from '@angular/tsc-wrapped/test/test_support';
|
|
|
|
import * as fs from 'fs';
|
|
|
|
import * as path from 'path';
|
|
|
|
|
|
|
|
import {main} from '../src/main';
|
|
|
|
|
|
|
|
|
|
|
|
describe('compiler-cli', () => {
|
|
|
|
let basePath: string;
|
|
|
|
let write: (fileName: string, content: string) => void;
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
basePath = makeTempDir();
|
|
|
|
write = (fileName: string, content: string) => {
|
|
|
|
fs.writeFileSync(path.join(basePath, fileName), content, {encoding: 'utf-8'});
|
|
|
|
};
|
|
|
|
write('tsconfig.json', `{
|
|
|
|
"compilerOptions": {
|
|
|
|
"experimentalDecorators": true,
|
|
|
|
"types": [],
|
|
|
|
"outDir": "built",
|
|
|
|
"declaration": true,
|
2016-12-15 12:12:40 -05:00
|
|
|
"module": "es2015",
|
|
|
|
"moduleResolution": "node"
|
2016-11-30 16:59:53 -05:00
|
|
|
},
|
|
|
|
"angularCompilerOptions": {
|
|
|
|
"annotateForClosureCompiler": true
|
|
|
|
},
|
|
|
|
"files": ["test.ts"]
|
|
|
|
}`);
|
2016-12-15 12:12:40 -05:00
|
|
|
const nodeModulesPath = path.resolve(basePath, 'node_modules');
|
|
|
|
fs.mkdirSync(nodeModulesPath);
|
|
|
|
fs.symlinkSync(path.resolve(__dirname, '..', '..'), path.resolve(nodeModulesPath, '@angular'));
|
2016-11-30 16:59:53 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
// Restore reflector since AoT compiler will update it with a new static reflector
|
2017-02-17 15:55:55 -05:00
|
|
|
afterEach(() => { ɵreflector.updateCapabilities(new ɵReflectionCapabilities()); });
|
2016-11-30 16:59:53 -05:00
|
|
|
|
|
|
|
it('should compile without errors', (done) => {
|
|
|
|
write('test.ts', 'export const A = 1;');
|
|
|
|
|
2016-11-30 19:37:28 -05:00
|
|
|
const mockConsole = {error: (s: string) => {}};
|
2016-11-30 16:59:53 -05:00
|
|
|
|
|
|
|
spyOn(mockConsole, 'error');
|
|
|
|
|
|
|
|
main({p: basePath}, mockConsole.error)
|
|
|
|
.then((exitCode) => {
|
|
|
|
expect(mockConsole.error).not.toHaveBeenCalled();
|
|
|
|
expect(exitCode).toEqual(0);
|
|
|
|
done();
|
|
|
|
})
|
|
|
|
.catch(e => done.fail(e));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not print the stack trace if user input file does not exist', (done) => {
|
2016-11-30 19:37:28 -05:00
|
|
|
const mockConsole = {error: (s: string) => {}};
|
2016-11-30 16:59:53 -05:00
|
|
|
|
|
|
|
spyOn(mockConsole, 'error');
|
|
|
|
|
|
|
|
main({p: basePath}, mockConsole.error)
|
|
|
|
.then((exitCode) => {
|
2016-11-30 19:45:40 -05:00
|
|
|
expect(mockConsole.error)
|
|
|
|
.toHaveBeenCalledWith(
|
|
|
|
`Error File '` + path.join(basePath, 'test.ts') + `' not found.`);
|
2016-11-30 16:59:53 -05:00
|
|
|
expect(mockConsole.error).not.toHaveBeenCalledWith('Compilation failed');
|
2016-11-30 19:37:28 -05:00
|
|
|
expect(exitCode).toEqual(1);
|
2016-11-30 16:59:53 -05:00
|
|
|
done();
|
|
|
|
})
|
|
|
|
.catch(e => done.fail(e));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not print the stack trace if user input file is malformed', (done) => {
|
2016-11-30 19:37:28 -05:00
|
|
|
write('test.ts', 'foo;');
|
2016-11-30 16:59:53 -05:00
|
|
|
|
2016-11-30 19:37:28 -05:00
|
|
|
const mockConsole = {error: (s: string) => {}};
|
2016-11-30 16:59:53 -05:00
|
|
|
|
|
|
|
spyOn(mockConsole, 'error');
|
|
|
|
|
|
|
|
main({p: basePath}, mockConsole.error)
|
|
|
|
.then((exitCode) => {
|
2016-11-30 19:45:40 -05:00
|
|
|
expect(mockConsole.error)
|
|
|
|
.toHaveBeenCalledWith(
|
|
|
|
'Error at ' + path.join(basePath, 'test.ts') + `:1:1: Cannot find name 'foo'.`);
|
2016-11-30 16:59:53 -05:00
|
|
|
expect(mockConsole.error).not.toHaveBeenCalledWith('Compilation failed');
|
2016-11-30 19:37:28 -05:00
|
|
|
expect(exitCode).toEqual(1);
|
|
|
|
done();
|
|
|
|
})
|
|
|
|
.catch(e => done.fail(e));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not print the stack trace if cannot find the imported module', (done) => {
|
2016-11-30 19:45:40 -05:00
|
|
|
write('test.ts', `import {MyClass} from './not-exist-deps';`);
|
2016-11-30 19:37:28 -05:00
|
|
|
|
|
|
|
const mockConsole = {error: (s: string) => {}};
|
|
|
|
|
|
|
|
spyOn(mockConsole, 'error');
|
|
|
|
|
|
|
|
main({p: basePath}, mockConsole.error)
|
|
|
|
.then((exitCode) => {
|
2016-11-30 19:45:40 -05:00
|
|
|
expect(mockConsole.error)
|
|
|
|
.toHaveBeenCalledWith(
|
|
|
|
'Error at ' + path.join(basePath, 'test.ts') +
|
|
|
|
`:1:23: Cannot find module './not-exist-deps'.`);
|
2016-11-30 19:37:28 -05:00
|
|
|
expect(mockConsole.error).not.toHaveBeenCalledWith('Compilation failed');
|
|
|
|
expect(exitCode).toEqual(1);
|
|
|
|
done();
|
|
|
|
})
|
|
|
|
.catch(e => done.fail(e));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not print the stack trace if cannot import', (done) => {
|
|
|
|
write('empty-deps.ts', 'export const A = 1;');
|
2016-11-30 19:45:40 -05:00
|
|
|
write('test.ts', `import {MyClass} from './empty-deps';`);
|
2016-11-30 19:37:28 -05:00
|
|
|
|
|
|
|
const mockConsole = {error: (s: string) => {}};
|
|
|
|
|
|
|
|
spyOn(mockConsole, 'error');
|
|
|
|
|
|
|
|
main({p: basePath}, mockConsole.error)
|
|
|
|
.then((exitCode) => {
|
2016-11-30 19:45:40 -05:00
|
|
|
expect(mockConsole.error)
|
|
|
|
.toHaveBeenCalledWith(
|
|
|
|
'Error at ' + path.join(basePath, 'test.ts') + `:1:9: Module '"` +
|
|
|
|
path.join(basePath, 'empty-deps') + `"' has no exported member 'MyClass'.`);
|
2016-11-30 19:37:28 -05:00
|
|
|
expect(mockConsole.error).not.toHaveBeenCalledWith('Compilation failed');
|
|
|
|
expect(exitCode).toEqual(1);
|
|
|
|
done();
|
|
|
|
})
|
|
|
|
.catch(e => done.fail(e));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not print the stack trace if type mismatches', (done) => {
|
|
|
|
write('empty-deps.ts', 'export const A = "abc";');
|
|
|
|
write('test.ts', `
|
|
|
|
import {A} from './empty-deps';
|
|
|
|
A();
|
|
|
|
`);
|
|
|
|
|
|
|
|
const mockConsole = {error: (s: string) => {}};
|
|
|
|
|
|
|
|
spyOn(mockConsole, 'error');
|
|
|
|
|
|
|
|
main({p: basePath}, mockConsole.error)
|
|
|
|
.then((exitCode) => {
|
2016-11-30 19:45:40 -05:00
|
|
|
expect(mockConsole.error)
|
|
|
|
.toHaveBeenCalledWith(
|
|
|
|
'Error at ' + path.join(basePath, 'test.ts') +
|
2017-01-24 12:05:34 -05:00
|
|
|
':3:7: Cannot invoke an expression whose type lacks a call signature. ' +
|
|
|
|
'Type \'String\' has no compatible call signatures.');
|
2016-11-30 19:37:28 -05:00
|
|
|
expect(mockConsole.error).not.toHaveBeenCalledWith('Compilation failed');
|
|
|
|
expect(exitCode).toEqual(1);
|
2016-11-30 16:59:53 -05:00
|
|
|
done();
|
|
|
|
})
|
|
|
|
.catch(e => done.fail(e));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should print the stack trace on compiler internal errors', (done) => {
|
|
|
|
write('test.ts', 'export const A = 1;');
|
|
|
|
|
2016-11-30 19:37:28 -05:00
|
|
|
const mockConsole = {error: (s: string) => {}};
|
2016-11-30 16:59:53 -05:00
|
|
|
|
|
|
|
spyOn(mockConsole, 'error');
|
|
|
|
|
|
|
|
main({p: 'not-exist'}, mockConsole.error)
|
|
|
|
.then((exitCode) => {
|
|
|
|
expect(mockConsole.error).toHaveBeenCalled();
|
|
|
|
expect(mockConsole.error).toHaveBeenCalledWith('Compilation failed');
|
|
|
|
expect(exitCode).toEqual(1);
|
|
|
|
done();
|
|
|
|
})
|
|
|
|
.catch(e => done.fail(e));
|
|
|
|
});
|
2016-11-30 19:45:40 -05:00
|
|
|
});
|