fix(ngcc): correctly detect external files from nested `node_modules/` (#36559)
Previously, when we needed to detect whether a file is external to a package, we only checked whether the relative path to the file from the package's root started with `..`. This would detect external imports when the packages were siblings (e.g. peer dependencies or hoisted to the top of `node_modules/` by the package manager), but would fail to detect imports from packages located in nested `node_modules/` as external. For example, importing `node_modules/foo/node_modules/bar` from a file in `node_modules/foo/` would be considered internal to the `foo` package. This could result in processing/analyzing more files than necessary. More importantly it could lead to errors due to trying to analyze non-Angular packages that were direct dependencies of Angular packages. This commit fixes it by also verifying that the relative path to a file does not start with `node_modules/`. Jira issue: [FW-2068](https://angular-team.atlassian.net/browse/FW-2068) Fixes #36526 PR Close #36559
This commit is contained in:
parent
fee316161d
commit
6ab43d7335
|
@ -11,7 +11,8 @@ import {absoluteFromSourceFile, AbsoluteFsPath, relative} from '../../../src/ngt
|
||||||
import {DependencyTracker} from '../../../src/ngtsc/incremental/api';
|
import {DependencyTracker} from '../../../src/ngtsc/incremental/api';
|
||||||
|
|
||||||
export function isWithinPackage(packagePath: AbsoluteFsPath, sourceFile: ts.SourceFile): boolean {
|
export function isWithinPackage(packagePath: AbsoluteFsPath, sourceFile: ts.SourceFile): boolean {
|
||||||
return !relative(packagePath, absoluteFromSourceFile(sourceFile)).startsWith('..');
|
const relativePath = relative(packagePath, absoluteFromSourceFile(sourceFile));
|
||||||
|
return !relativePath.startsWith('..') && !relativePath.startsWith('node_modules/');
|
||||||
}
|
}
|
||||||
|
|
||||||
class NoopDependencyTracker implements DependencyTracker {
|
class NoopDependencyTracker implements DependencyTracker {
|
||||||
|
|
|
@ -313,17 +313,31 @@ runInEachFileSystem(() => {
|
||||||
contents: `
|
contents: `
|
||||||
import {Component, NgModule} from '@angular/core';
|
import {Component, NgModule} from '@angular/core';
|
||||||
import {ImportedComponent} from 'other/component';
|
import {ImportedComponent} from 'other/component';
|
||||||
|
import {NestedDependencyComponent} from 'nested/component';
|
||||||
|
|
||||||
export class LocalComponent {}
|
export class LocalComponent {}
|
||||||
LocalComponent.decorators = [{type: Component}];
|
LocalComponent.decorators = [{type: Component}];
|
||||||
|
|
||||||
export class MyModule {}
|
export class MyModule {}
|
||||||
MyModule.decorators = [{type: NgModule, args: [{
|
MyModule.decorators = [{type: NgModule, args: [{
|
||||||
declarations: [ImportedComponent, LocalComponent],
|
declarations: [ImportedComponent, NestedDependencyComponent, LocalComponent],
|
||||||
exports: [ImportedComponent, LocalComponent],
|
exports: [ImportedComponent, NestedDependencyComponent, LocalComponent],
|
||||||
},] }];
|
},] }];
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
|
// Do not define a `.d.ts` file to ensure that the `.js` file will be part of the TS
|
||||||
|
// program.
|
||||||
|
{
|
||||||
|
name: _('/node_modules/test-package/node_modules/nested/component.js'),
|
||||||
|
contents: `
|
||||||
|
import {Component} from '@angular/core';
|
||||||
|
export class NestedDependencyComponent {}
|
||||||
|
NestedDependencyComponent.decorators = [{type: Component}];
|
||||||
|
`,
|
||||||
|
isRoot: false,
|
||||||
|
},
|
||||||
|
// Do not define a `.d.ts` file to ensure that the `.js` file will be part of the TS
|
||||||
|
// program.
|
||||||
{
|
{
|
||||||
name: _('/node_modules/other/component.js'),
|
name: _('/node_modules/other/component.js'),
|
||||||
contents: `
|
contents: `
|
||||||
|
@ -333,12 +347,6 @@ runInEachFileSystem(() => {
|
||||||
`,
|
`,
|
||||||
isRoot: false,
|
isRoot: false,
|
||||||
},
|
},
|
||||||
{
|
|
||||||
name: _('/node_modules/other/component.d.ts'),
|
|
||||||
contents: `
|
|
||||||
import {Component} from '@angular/core';
|
|
||||||
export class ImportedComponent {}`
|
|
||||||
},
|
|
||||||
];
|
];
|
||||||
|
|
||||||
const analyzer = setUpAnalyzer(EXTERNAL_COMPONENT_PROGRAM);
|
const analyzer = setUpAnalyzer(EXTERNAL_COMPONENT_PROGRAM);
|
||||||
|
@ -349,6 +357,12 @@ runInEachFileSystem(() => {
|
||||||
const file = program.getSourceFile(_('/node_modules/other/component.js'))!;
|
const file = program.getSourceFile(_('/node_modules/other/component.js'))!;
|
||||||
expect(result.has(file)).toBe(false);
|
expect(result.has(file)).toBe(false);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should ignore classes from a file imported from a nested `node_modules/`', () => {
|
||||||
|
const file = program.getSourceFile(
|
||||||
|
_('/node_modules/test-package/node_modules/nested/component.js'))!;
|
||||||
|
expect(result.has(file)).toBe(false);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('diagnostic handling', () => {
|
describe('diagnostic handling', () => {
|
||||||
|
|
|
@ -92,8 +92,6 @@ runInEachFileSystem(() => {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
describe('getAllDecorators', () => {
|
describe('getAllDecorators', () => {
|
||||||
it('should include injected decorators', () => {
|
it('should include injected decorators', () => {
|
||||||
const directiveHandler = new DetectDecoratorHandler('Directive', HandlerPrecedence.WEAK);
|
const directiveHandler = new DetectDecoratorHandler('Directive', HandlerPrecedence.WEAK);
|
||||||
|
@ -143,7 +141,7 @@ runInEachFileSystem(() => {
|
||||||
expect(host.isInScope(internalClass)).toBe(true);
|
expect(host.isInScope(internalClass)).toBe(true);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should be false for nodes outside the entry-point', () => {
|
it('should be false for nodes outside the entry-point (in sibling package)', () => {
|
||||||
loadTestFiles([
|
loadTestFiles([
|
||||||
{name: _('/node_modules/external/index.js'), contents: `export class ExternalClass {}`},
|
{name: _('/node_modules/external/index.js'), contents: `export class ExternalClass {}`},
|
||||||
{
|
{
|
||||||
|
@ -163,6 +161,30 @@ runInEachFileSystem(() => {
|
||||||
|
|
||||||
expect(host.isInScope(externalClass)).toBe(false);
|
expect(host.isInScope(externalClass)).toBe(false);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should be false for nodes outside the entry-point (in nested `node_modules/`)', () => {
|
||||||
|
loadTestFiles([
|
||||||
|
{
|
||||||
|
name: _('/node_modules/test/index.js'),
|
||||||
|
contents: `
|
||||||
|
export {NestedDependencyClass} from 'nested';
|
||||||
|
export class InternalClass {}
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: _('/node_modules/test/node_modules/nested/index.js'),
|
||||||
|
contents: `export class NestedDependencyClass {}`,
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
const entryPoint =
|
||||||
|
makeTestEntryPointBundle('test', 'esm2015', false, [_('/node_modules/test/index.js')]);
|
||||||
|
const {host} = createMigrationHost({entryPoint, handlers: []});
|
||||||
|
const nestedDepClass = getDeclaration(
|
||||||
|
entryPoint.src.program, _('/node_modules/test/node_modules/nested/index.js'),
|
||||||
|
'NestedDependencyClass', isNamedClassDeclaration);
|
||||||
|
|
||||||
|
expect(host.isInScope(nestedDepClass)).toBe(false);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -24,45 +24,53 @@ runInEachFileSystem(() => {
|
||||||
{
|
{
|
||||||
name: _('/node_modules/test/entrypoint.js'),
|
name: _('/node_modules/test/entrypoint.js'),
|
||||||
contents: `
|
contents: `
|
||||||
import {a} from './a';
|
import {a} from './a';
|
||||||
import {b} from './b';
|
import {b} from './b';
|
||||||
import {x} from '../other/x';
|
import {x} from '../other/x';
|
||||||
`
|
import {e} from 'nested/e';
|
||||||
|
`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: _('/node_modules/test/a.js'),
|
name: _('/node_modules/test/a.js'),
|
||||||
contents: `
|
contents: `
|
||||||
import {c} from './c';
|
import {c} from './c';
|
||||||
export const a = 1;
|
export const a = 1;
|
||||||
`
|
`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: _('/node_modules/test/b.js'),
|
name: _('/node_modules/test/b.js'),
|
||||||
contents: `
|
contents: `
|
||||||
export const b = 42;
|
export const b = 42;
|
||||||
var factoryB = factory__PRE_R3__;
|
var factoryB = factory__PRE_R3__;
|
||||||
`
|
`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: _('/node_modules/test/c.js'),
|
name: _('/node_modules/test/c.js'),
|
||||||
contents: `
|
contents: `
|
||||||
export const c = 'So long, and thanks for all the fish!';
|
export const c = 'So long, and thanks for all the fish!';
|
||||||
var factoryC = factory__PRE_R3__;
|
var factoryC = factory__PRE_R3__;
|
||||||
var factoryD = factory__PRE_R3__;
|
var factoryD = factory__PRE_R3__;
|
||||||
`
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: _('/node_modules/test/node_modules/nested/e.js'),
|
||||||
|
contents: `
|
||||||
|
export const e = 1337;
|
||||||
|
var factoryE = factory__PRE_R3__;
|
||||||
|
`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: _('/node_modules/other/x.js'),
|
name: _('/node_modules/other/x.js'),
|
||||||
contents: `
|
contents: `
|
||||||
export const x = 3.142;
|
export const x = 3.142;
|
||||||
var factoryX = factory__PRE_R3__;
|
var factoryX = factory__PRE_R3__;
|
||||||
`
|
`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: _('/node_modules/other/x.d.ts'),
|
name: _('/node_modules/other/x.d.ts'),
|
||||||
contents: `
|
contents: `
|
||||||
export const x: number;
|
export const x: number;
|
||||||
`
|
`,
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
});
|
});
|
||||||
|
@ -111,6 +119,19 @@ runInEachFileSystem(() => {
|
||||||
const x = getSourceFileOrError(program, _('/node_modules/other/x.js'));
|
const x = getSourceFileOrError(program, _('/node_modules/other/x.js'));
|
||||||
expect(analysis.has(x)).toBe(false);
|
expect(analysis.has(x)).toBe(false);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should ignore files that are inside the package\'s `node_modules/`', () => {
|
||||||
|
loadTestFiles(TEST_PROGRAM);
|
||||||
|
const bundle = makeTestEntryPointBundle(
|
||||||
|
'test', 'esm2015', false, [_('/node_modules/test/entrypoint.js')]);
|
||||||
|
const program = bundle.src.program;
|
||||||
|
const host = new Esm2015ReflectionHost(new MockLogger(), false, bundle.src);
|
||||||
|
const analyzer = new SwitchMarkerAnalyzer(host, bundle.entryPoint.package);
|
||||||
|
const analysis = analyzer.analyzeProgram(program);
|
||||||
|
|
||||||
|
const x = getSourceFileOrError(program, _('/node_modules/test/node_modules/nested/e.js'));
|
||||||
|
expect(analysis.has(x)).toBe(false);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -12,20 +12,36 @@ import {isWithinPackage} from '../../src/analysis/util';
|
||||||
|
|
||||||
runInEachFileSystem(() => {
|
runInEachFileSystem(() => {
|
||||||
describe('isWithinPackage', () => {
|
describe('isWithinPackage', () => {
|
||||||
|
let _: typeof absoluteFrom;
|
||||||
|
|
||||||
|
beforeEach(() => _ = absoluteFrom);
|
||||||
|
|
||||||
it('should return true if the source-file is contained in the package', () => {
|
it('should return true if the source-file is contained in the package', () => {
|
||||||
const _ = absoluteFrom;
|
const packagePath = _('/node_modules/test');
|
||||||
const file =
|
const file =
|
||||||
ts.createSourceFile(_('/node_modules/test/src/index.js'), '', ts.ScriptTarget.ES2015);
|
ts.createSourceFile(_('/node_modules/test/src/index.js'), '', ts.ScriptTarget.ES2015);
|
||||||
const packagePath = _('/node_modules/test');
|
|
||||||
expect(isWithinPackage(packagePath, file)).toBe(true);
|
expect(isWithinPackage(packagePath, file)).toBe(true);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return false if the source-file is not contained in the package', () => {
|
it('should return false if the source-file is not contained in the package', () => {
|
||||||
const _ = absoluteFrom;
|
const packagePath = _('/node_modules/test');
|
||||||
const file =
|
const file =
|
||||||
ts.createSourceFile(_('/node_modules/other/src/index.js'), '', ts.ScriptTarget.ES2015);
|
ts.createSourceFile(_('/node_modules/other/src/index.js'), '', ts.ScriptTarget.ES2015);
|
||||||
const packagePath = _('/node_modules/test');
|
|
||||||
expect(isWithinPackage(packagePath, file)).toBe(false);
|
expect(isWithinPackage(packagePath, file)).toBe(false);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should return false if the source-file is inside the package\'s `node_modules/`', () => {
|
||||||
|
const packagePath = _('/node_modules/test');
|
||||||
|
|
||||||
|
// An external file inside the package's `node_modules/`.
|
||||||
|
const file1 = ts.createSourceFile(
|
||||||
|
_('/node_modules/test/node_modules/other/src/index.js'), '', ts.ScriptTarget.ES2015);
|
||||||
|
expect(isWithinPackage(packagePath, file1)).toBe(false);
|
||||||
|
|
||||||
|
// An internal file starting with `node_modules`.
|
||||||
|
const file2 = ts.createSourceFile(
|
||||||
|
_('/node_modules/test/node_modules_optimizer.js'), '', ts.ScriptTarget.ES2015);
|
||||||
|
expect(isWithinPackage(packagePath, file2)).toBe(true);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
Loading…
Reference in New Issue