2018-08-09 10:59:10 -04: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
* /
import * as ts from 'typescript' ;
2019-03-20 09:47:58 -04:00
2019-06-06 15:22:32 -04:00
import { absoluteFrom , getFileSystem , relativeFrom } from '../../../src/ngtsc/file_system' ;
import { runInEachFileSystem } from '../../../src/ngtsc/file_system/testing' ;
import { loadTestFiles } from '../../../test/helpers' ;
2019-12-19 17:43:12 -05:00
import { createDependencyInfo } from '../../src/dependencies/dependency_host' ;
2020-01-08 12:10:25 -05:00
import { DtsDependencyHost } from '../../src/dependencies/dts_dependency_host' ;
2019-07-11 07:34:45 -04:00
import { EsmDependencyHost , hasImportOrReexportStatements , isStringImportOrReexport } from '../../src/dependencies/esm_dependency_host' ;
2019-04-28 15:47:57 -04:00
import { ModuleResolver } from '../../src/dependencies/module_resolver' ;
2018-08-09 10:59:10 -04:00
2019-06-06 15:22:32 -04:00
runInEachFileSystem ( ( ) = > {
2019-03-20 09:47:58 -04:00
2019-06-06 15:22:32 -04:00
describe ( 'EsmDependencyHost' , ( ) = > {
let _ : typeof absoluteFrom ;
let host : EsmDependencyHost ;
beforeEach ( ( ) = > {
_ = absoluteFrom ;
setupMockFileSystem ( ) ;
const fs = getFileSystem ( ) ;
host = new EsmDependencyHost ( fs , new ModuleResolver ( fs ) ) ;
2018-08-09 10:59:10 -04:00
} ) ;
2019-06-06 15:22:32 -04:00
describe ( 'getDependencies()' , ( ) = > {
it ( 'should not generate a TS AST if the source does not contain any imports or re-exports' ,
( ) = > {
spyOn ( ts , 'createSourceFile' ) ;
2019-12-19 17:43:12 -05:00
host . collectDependencies (
_ ( '/no/imports/or/re-exports/index.js' ) , createDependencyInfo ( ) ) ;
2019-06-06 15:22:32 -04:00
expect ( ts . createSourceFile ) . not . toHaveBeenCalled ( ) ;
} ) ;
2018-08-09 10:59:10 -04:00
2019-06-06 15:22:32 -04:00
it ( 'should resolve all the external imports of the source file' , ( ) = > {
2019-12-19 17:43:12 -05:00
const { dependencies , missing , deepImports } = createDependencyInfo ( ) ;
host . collectDependencies (
_ ( '/external/imports/index.js' ) , { dependencies , missing , deepImports } ) ;
2019-06-06 15:22:32 -04:00
expect ( dependencies . size ) . toBe ( 2 ) ;
expect ( missing . size ) . toBe ( 0 ) ;
expect ( deepImports . size ) . toBe ( 0 ) ;
expect ( dependencies . has ( _ ( '/node_modules/lib-1' ) ) ) . toBe ( true ) ;
expect ( dependencies . has ( _ ( '/node_modules/lib-1/sub-1' ) ) ) . toBe ( true ) ;
} ) ;
2019-04-28 15:47:57 -04:00
2019-06-06 15:22:32 -04:00
it ( 'should resolve all the external re-exports of the source file' , ( ) = > {
2019-12-19 17:43:12 -05:00
const { dependencies , missing , deepImports } = createDependencyInfo ( ) ;
host . collectDependencies (
_ ( '/external/re-exports/index.js' ) , { dependencies , missing , deepImports } ) ;
2019-06-06 15:22:32 -04:00
expect ( dependencies . size ) . toBe ( 2 ) ;
expect ( missing . size ) . toBe ( 0 ) ;
expect ( deepImports . size ) . toBe ( 0 ) ;
expect ( dependencies . has ( _ ( '/node_modules/lib-1' ) ) ) . toBe ( true ) ;
expect ( dependencies . has ( _ ( '/node_modules/lib-1/sub-1' ) ) ) . toBe ( true ) ;
} ) ;
2018-11-09 14:13:51 -05:00
2019-06-06 15:22:32 -04:00
it ( 'should capture missing external imports' , ( ) = > {
2019-12-19 17:43:12 -05:00
const { dependencies , missing , deepImports } = createDependencyInfo ( ) ;
host . collectDependencies (
_ ( '/external/imports-missing/index.js' ) , { dependencies , missing , deepImports } ) ;
2018-08-09 10:59:10 -04:00
2019-06-06 15:22:32 -04:00
expect ( dependencies . size ) . toBe ( 1 ) ;
expect ( dependencies . has ( _ ( '/node_modules/lib-1' ) ) ) . toBe ( true ) ;
expect ( missing . size ) . toBe ( 1 ) ;
expect ( missing . has ( relativeFrom ( 'missing' ) ) ) . toBe ( true ) ;
expect ( deepImports . size ) . toBe ( 0 ) ;
} ) ;
2018-08-09 10:59:10 -04:00
2019-06-06 15:22:32 -04:00
it ( 'should not register deep imports as missing' , ( ) = > {
// This scenario verifies the behavior of the dependency analysis when an external import
// is found that does not map to an entry-point but still exists on disk, i.e. a deep
// import. Such deep imports are captured for diagnostics purposes.
2019-12-19 17:43:12 -05:00
const { dependencies , missing , deepImports } = createDependencyInfo ( ) ;
host . collectDependencies (
_ ( '/external/deep-import/index.js' ) , { dependencies , missing , deepImports } ) ;
2018-08-09 10:59:10 -04:00
2019-06-06 15:22:32 -04:00
expect ( dependencies . size ) . toBe ( 0 ) ;
expect ( missing . size ) . toBe ( 0 ) ;
expect ( deepImports . size ) . toBe ( 1 ) ;
expect ( deepImports . has ( _ ( '/node_modules/lib-1/deep/import' ) ) ) . toBe ( true ) ;
} ) ;
2019-04-28 15:47:57 -04:00
2019-06-06 15:22:32 -04:00
it ( 'should recurse into internal dependencies' , ( ) = > {
2019-12-19 17:43:12 -05:00
const { dependencies , missing , deepImports } = createDependencyInfo ( ) ;
host . collectDependencies (
_ ( '/internal/outer/index.js' ) , { dependencies , missing , deepImports } ) ;
2018-08-09 10:59:10 -04:00
2019-06-06 15:22:32 -04:00
expect ( dependencies . size ) . toBe ( 1 ) ;
expect ( dependencies . has ( _ ( '/node_modules/lib-1/sub-1' ) ) ) . toBe ( true ) ;
expect ( missing . size ) . toBe ( 0 ) ;
expect ( deepImports . size ) . toBe ( 0 ) ;
} ) ;
2018-08-09 10:59:10 -04:00
2019-06-06 15:22:32 -04:00
it ( 'should handle circular internal dependencies' , ( ) = > {
2019-12-19 17:43:12 -05:00
const { dependencies , missing , deepImports } = createDependencyInfo ( ) ;
host . collectDependencies (
_ ( '/internal/circular-a/index.js' ) , { dependencies , missing , deepImports } ) ;
2019-06-06 15:22:32 -04:00
expect ( dependencies . size ) . toBe ( 2 ) ;
expect ( dependencies . has ( _ ( '/node_modules/lib-1' ) ) ) . toBe ( true ) ;
expect ( dependencies . has ( _ ( '/node_modules/lib-1/sub-1' ) ) ) . toBe ( true ) ;
expect ( missing . size ) . toBe ( 0 ) ;
expect ( deepImports . size ) . toBe ( 0 ) ;
} ) ;
2018-08-09 10:59:10 -04:00
2019-06-06 15:22:32 -04:00
it ( 'should support `paths` alias mappings when resolving modules' , ( ) = > {
const fs = getFileSystem ( ) ;
host = new EsmDependencyHost ( fs , new ModuleResolver ( fs , {
baseUrl : '/dist' ,
paths : {
'@app/*' : [ '*' ] ,
'@lib/*/test' : [ 'lib/*/test' ] ,
}
} ) ) ;
2019-12-19 17:43:12 -05:00
const { dependencies , missing , deepImports } = createDependencyInfo ( ) ;
host . collectDependencies ( _ ( '/path-alias/index.js' ) , { dependencies , missing , deepImports } ) ;
2019-06-06 15:22:32 -04:00
expect ( dependencies . size ) . toBe ( 4 ) ;
expect ( dependencies . has ( _ ( '/dist/components' ) ) ) . toBe ( true ) ;
expect ( dependencies . has ( _ ( '/dist/shared' ) ) ) . toBe ( true ) ;
expect ( dependencies . has ( _ ( '/dist/lib/shared/test' ) ) ) . toBe ( true ) ;
expect ( dependencies . has ( _ ( '/node_modules/lib-1' ) ) ) . toBe ( true ) ;
expect ( missing . size ) . toBe ( 0 ) ;
expect ( deepImports . size ) . toBe ( 0 ) ;
} ) ;
2019-07-11 07:34:45 -04:00
it ( 'should handle entry-point paths with no extension' , ( ) = > {
2019-12-19 17:43:12 -05:00
const { dependencies , missing , deepImports } = createDependencyInfo ( ) ;
host . collectDependencies (
_ ( '/external/imports/index' ) , { dependencies , missing , deepImports } ) ;
2019-07-11 07:34:45 -04:00
expect ( dependencies . size ) . toBe ( 2 ) ;
expect ( missing . size ) . toBe ( 0 ) ;
expect ( deepImports . size ) . toBe ( 0 ) ;
expect ( dependencies . has ( _ ( '/node_modules/lib-1' ) ) ) . toBe ( true ) ;
expect ( dependencies . has ( _ ( '/node_modules/lib-1/sub-1' ) ) ) . toBe ( true ) ;
} ) ;
2019-12-19 17:43:12 -05:00
it ( 'should resolve modules based on the provided `moduleResolver`' , ( ) = > {
const fs = getFileSystem ( ) ;
loadTestFiles ( [
{
name : _ ( '/external/index.d.ts' ) ,
contents : ` import * from './internal-typings'; ` ,
} ,
{
name : _ ( '/external/internal-typings.d.ts' ) ,
contents : ` export {X} from 'lib-1'; \ nexport {Y} from 'lib-1/sub-1'; ` ,
}
] ) ;
// Default JS mode will not pick up `internal-typings.d.ts` dependency
const jsHost = new EsmDependencyHost ( fs , new ModuleResolver ( fs ) ) ;
const jsDeps = createDependencyInfo ( ) ;
jsHost . collectDependencies ( _ ( '/external/index.d.ts' ) , jsDeps ) ;
expect ( jsDeps . dependencies . size ) . toEqual ( 0 ) ;
expect ( jsDeps . deepImports . size ) . toEqual ( 0 ) ;
expect ( jsDeps . missing . size ) . toEqual ( 1 ) ;
expect ( jsDeps . missing . has ( relativeFrom ( './internal-typings' ) ) ) . toBeTruthy ( ) ;
// Typings mode will pick up `internal-typings.d.ts` dependency
2020-01-08 12:10:25 -05:00
const dtsHost = new DtsDependencyHost ( fs ) ;
2019-12-19 17:43:12 -05:00
const dtsDeps = createDependencyInfo ( ) ;
dtsHost . collectDependencies ( _ ( '/external/index.d.ts' ) , dtsDeps ) ;
expect ( dtsDeps . dependencies . size ) . toEqual ( 2 ) ;
expect ( dtsDeps . dependencies . has ( _ ( '/node_modules/lib-1' ) ) ) . toBeTruthy ( ) ;
expect ( dtsDeps . dependencies . has ( _ ( '/node_modules/lib-1/sub-1' ) ) ) . toBeTruthy ( ) ;
expect ( dtsDeps . deepImports . size ) . toEqual ( 0 ) ;
expect ( dtsDeps . missing . size ) . toEqual ( 0 ) ;
} ) ;
2018-08-09 10:59:10 -04:00
} ) ;
2019-06-06 15:22:32 -04:00
function setupMockFileSystem ( ) : void {
loadTestFiles ( [
{
name : _ ( '/no/imports/or/re-exports/index.js' ) ,
contents : '// some text but no import-like statements'
} ,
{ name : _ ( '/no/imports/or/re-exports/package.json' ) , contents : '{"esm2015": "./index.js"}' } ,
{ name : _ ( '/no/imports/or/re-exports/index.metadata.json' ) , contents : 'MOCK METADATA' } ,
{
name : _ ( '/external/imports/index.js' ) ,
contents : ` import {X} from 'lib-1'; \ nimport {Y} from 'lib-1/sub-1'; `
} ,
{ name : _ ( '/external/imports/package.json' ) , contents : '{"esm2015": "./index.js"}' } ,
{ name : _ ( '/external/imports/index.metadata.json' ) , contents : 'MOCK METADATA' } ,
{
name : _ ( '/external/re-exports/index.js' ) ,
contents : ` export {X} from 'lib-1'; \ nexport {Y} from 'lib-1/sub-1'; `
} ,
{ name : _ ( '/external/re-exports/package.json' ) , contents : '{"esm2015": "./index.js"}' } ,
{ name : _ ( '/external/re-exports/index.metadata.json' ) , contents : 'MOCK METADATA' } ,
{
name : _ ( '/external/imports-missing/index.js' ) ,
contents : ` import {X} from 'lib-1'; \ nimport {Y} from 'missing'; `
} ,
{ name : _ ( '/external/imports-missing/package.json' ) , contents : '{"esm2015": "./index.js"}' } ,
{ name : _ ( '/external/imports-missing/index.metadata.json' ) , contents : 'MOCK METADATA' } ,
{
name : _ ( '/external/deep-import/index.js' ) ,
contents : ` import {Y} from 'lib-1/deep/import'; `
} ,
{ name : _ ( '/external/deep-import/package.json' ) , contents : '{"esm2015": "./index.js"}' } ,
{ name : _ ( '/external/deep-import/index.metadata.json' ) , contents : 'MOCK METADATA' } ,
{ name : _ ( '/internal/outer/index.js' ) , contents : ` import {X} from '../inner'; ` } ,
{ name : _ ( '/internal/outer/package.json' ) , contents : '{"esm2015": "./index.js"}' } ,
{ name : _ ( '/internal/outer/index.metadata.json' ) , contents : 'MOCK METADATA' } ,
{
name : _ ( '/internal/inner/index.js' ) ,
contents : ` import {Y} from 'lib-1/sub-1'; export declare class X {} `
} ,
{
name : _ ( '/internal/circular-a/index.js' ) ,
contents :
` import {B} from '../circular-b'; import {X} from '../circular-b'; export {Y} from 'lib-1/sub-1'; `
} ,
{
name : _ ( '/internal/circular-b/index.js' ) ,
contents :
` import {A} from '../circular-a'; import {Y} from '../circular-a'; export {X} from 'lib-1'; `
} ,
{ name : _ ( '/internal/circular-a/package.json' ) , contents : '{"esm2015": "./index.js"}' } ,
{ name : _ ( '/internal/circular-a/index.metadata.json' ) , contents : 'MOCK METADATA' } ,
{ name : _ ( '/re-directed/index.js' ) , contents : ` import {Z} from 'lib-1/sub-2'; ` } ,
{ name : _ ( '/re-directed/package.json' ) , contents : '{"esm2015": "./index.js"}' } ,
{ name : _ ( '/re-directed/index.metadata.json' ) , contents : 'MOCK METADATA' } ,
{
name : _ ( '/path-alias/index.js' ) ,
contents :
` import {TestHelper} from '@app/components'; \ nimport {Service} from '@app/shared'; \ nimport {TestHelper} from '@lib/shared/test'; \ nimport {X} from 'lib-1'; `
} ,
{ name : _ ( '/path-alias/package.json' ) , contents : '{"esm2015": "./index.js"}' } ,
{ name : _ ( '/path-alias/index.metadata.json' ) , contents : 'MOCK METADATA' } ,
{ name : _ ( '/node_modules/lib-1/index.js' ) , contents : 'export declare class X {}' } ,
{ name : _ ( '/node_modules/lib-1/package.json' ) , contents : '{"esm2015": "./index.js"}' } ,
{ name : _ ( '/node_modules/lib-1/index.metadata.json' ) , contents : 'MOCK METADATA' } ,
{
name : _ ( '/node_modules/lib-1/deep/import/index.js' ) ,
contents : 'export declare class DeepImport {}'
} ,
{ name : _ ( '/node_modules/lib-1/sub-1/index.js' ) , contents : 'export declare class Y {}' } ,
{ name : _ ( '/node_modules/lib-1/sub-1/package.json' ) , contents : '{"esm2015": "./index.js"}' } ,
{ name : _ ( '/node_modules/lib-1/sub-1/index.metadata.json' ) , contents : 'MOCK METADATA' } ,
{ name : _ ( '/node_modules/lib-1/sub-2.js' ) , contents : ` export * from './sub-2/sub-2'; ` } ,
{ name : _ ( '/node_modules/lib-1/sub-2/sub-2.js' ) , contents : ` export declare class Z {}'; ` } ,
{ name : _ ( '/node_modules/lib-1/sub-2/package.json' ) , contents : '{"esm2015": "./sub-2.js"}' } ,
{ name : _ ( '/node_modules/lib-1/sub-2/sub-2.metadata.json' ) , contents : 'MOCK METADATA' } ,
{ name : _ ( '/dist/components/index.js' ) , contents : ` class MyComponent {}; ` } ,
{ name : _ ( '/dist/components/package.json' ) , contents : '{"esm2015": "./index.js"}' } ,
{ name : _ ( '/dist/components/index.metadata.json' ) , contents : 'MOCK METADATA' } ,
{
name : _ ( '/dist/shared/index.js' ) ,
contents : ` import {X} from 'lib-1'; \ nexport class Service {} `
} ,
{ name : _ ( '/dist/shared/package.json' ) , contents : '{"esm2015": "./index.js"}' } ,
{ name : _ ( '/dist/shared/index.metadata.json' ) , contents : 'MOCK METADATA' } ,
{ name : _ ( '/dist/lib/shared/test/index.js' ) , contents : ` export class TestHelper {} ` } ,
{ name : _ ( '/dist/lib/shared/test/package.json' ) , contents : '{"esm2015": "./index.js"}' } ,
{ name : _ ( '/dist/lib/shared/test/index.metadata.json' ) , contents : 'MOCK METADATA' } ,
] ) ;
2018-08-09 10:59:10 -04:00
}
2019-06-06 15:22:32 -04:00
describe ( 'isStringImportOrReexport' , ( ) = > {
it ( 'should return true if the statement is an import' , ( ) = > {
2019-07-11 07:34:45 -04:00
expect ( isStringImportOrReexport ( createStatement ( 'import {X} from "some/x";' ) ) ) . toBe ( true ) ;
expect ( isStringImportOrReexport ( createStatement ( 'import * as X from "some/x";' ) ) )
2019-06-06 15:22:32 -04:00
. toBe ( true ) ;
} ) ;
it ( 'should return true if the statement is a re-export' , ( ) = > {
2019-07-11 07:34:45 -04:00
expect ( isStringImportOrReexport ( createStatement ( 'export {X} from "some/x";' ) ) ) . toBe ( true ) ;
expect ( isStringImportOrReexport ( createStatement ( 'export * from "some/x";' ) ) ) . toBe ( true ) ;
2019-06-06 15:22:32 -04:00
} ) ;
it ( 'should return false if the statement is not an import or a re-export' , ( ) = > {
2019-07-11 07:34:45 -04:00
expect ( isStringImportOrReexport ( createStatement ( 'class X {}' ) ) ) . toBe ( false ) ;
expect ( isStringImportOrReexport ( createStatement ( 'export function foo() {}' ) ) ) . toBe ( false ) ;
expect ( isStringImportOrReexport ( createStatement ( 'export const X = 10;' ) ) ) . toBe ( false ) ;
2019-06-06 15:22:32 -04:00
} ) ;
function createStatement ( source : string ) {
return ts
. createSourceFile ( 'source.js' , source , ts . ScriptTarget . ES2015 , false , ts . ScriptKind . JS )
. statements [ 0 ] ;
}
2018-08-09 10:59:10 -04:00
} ) ;
2019-06-06 15:22:32 -04:00
describe ( 'hasImportOrReexportStatements' , ( ) = > {
it ( 'should return true if there is an import statement' , ( ) = > {
2019-07-11 07:34:45 -04:00
expect ( hasImportOrReexportStatements ( 'import {X} from "some/x";' ) ) . toBe ( true ) ;
expect ( hasImportOrReexportStatements ( 'import * as X from "some/x";' ) ) . toBe ( true ) ;
expect ( hasImportOrReexportStatements ( 'blah blah\n\n import {X} from "some/x";\nblah blah' ) )
2019-06-06 15:22:32 -04:00
. toBe ( true ) ;
2019-07-11 07:34:45 -04:00
expect ( hasImportOrReexportStatements ( '\t\timport {X} from "some/x";' ) ) . toBe ( true ) ;
2019-06-06 15:22:32 -04:00
} ) ;
it ( 'should return true if there is a re-export statement' , ( ) = > {
2019-07-11 07:34:45 -04:00
expect ( hasImportOrReexportStatements ( 'export {X} from "some/x";' ) ) . toBe ( true ) ;
expect ( hasImportOrReexportStatements ( 'blah blah\n\n export {X} from "some/x";\nblah blah' ) )
2019-06-06 15:22:32 -04:00
. toBe ( true ) ;
2019-07-11 07:34:45 -04:00
expect ( hasImportOrReexportStatements ( '\t\texport {X} from "some/x";' ) ) . toBe ( true ) ;
expect ( hasImportOrReexportStatements (
2019-06-06 15:22:32 -04:00
'blah blah\n\n export * from "@angular/core;\nblah blah' ) )
. toBe ( true ) ;
} ) ;
it ( 'should return false if there is no import nor re-export statement' , ( ) = > {
2019-07-11 07:34:45 -04:00
expect ( hasImportOrReexportStatements ( 'blah blah' ) ) . toBe ( false ) ;
expect ( hasImportOrReexportStatements ( 'export function moo() {}' ) ) . toBe ( false ) ;
expect ( hasImportOrReexportStatements ( 'Some text that happens to include the word import' ) )
2019-06-06 15:22:32 -04:00
. toBe ( false ) ;
} ) ;
2018-08-09 10:59:10 -04:00
} ) ;
} ) ;
2019-06-06 15:22:32 -04:00
} ) ;