fix(ngcc): correctly identify relative Windows-style import paths (#36372)
Previously, `isRelativePath()` assumed paths are *nix-style. This caused Windows-style paths (such as `C:\foo\some-package\some-file.js`) to not be recognized as "relative" imports. This commit fixes this by using the OS-agnostic `isRooted()` helper and also accounting for both styles of path delimiters: `/` and `\` PR Close #36372
This commit is contained in:
parent
81195a238b
commit
aecf9de738
|
@ -7,7 +7,7 @@
|
||||||
*/
|
*/
|
||||||
import * as ts from 'typescript';
|
import * as ts from 'typescript';
|
||||||
|
|
||||||
import {absoluteFrom, AbsoluteFsPath, FileSystem} from '../../src/ngtsc/file_system';
|
import {absoluteFrom, AbsoluteFsPath, FileSystem, isRooted} from '../../src/ngtsc/file_system';
|
||||||
import {KnownDeclaration} from '../../src/ngtsc/reflection';
|
import {KnownDeclaration} from '../../src/ngtsc/reflection';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -85,10 +85,11 @@ export type PathMappings = {
|
||||||
/**
|
/**
|
||||||
* Test whether a path is "relative".
|
* Test whether a path is "relative".
|
||||||
*
|
*
|
||||||
* Relative paths start with `/`, `./` or `../`; or are simply `.` or `..`.
|
* Relative paths start with `/`, `./` or `../` (or the Windows equivalents); or are simply `.` or
|
||||||
|
* `..`.
|
||||||
*/
|
*/
|
||||||
export function isRelativePath(path: string): boolean {
|
export function isRelativePath(path: string): boolean {
|
||||||
return /^\/|^\.\.?($|\/)/.test(path);
|
return isRooted(path) || /^\.\.?(\/|\\|$)/.test(path);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -67,16 +67,29 @@ runInEachFileSystem(() => {
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('resolveModule()', () => {
|
describe('resolveModuleImport()', () => {
|
||||||
describe('with relative paths', () => {
|
describe('with relative paths', () => {
|
||||||
it('should resolve sibling, child and aunt modules', () => {
|
it('should resolve sibling, child and aunt modules', () => {
|
||||||
const resolver = new ModuleResolver(getFileSystem());
|
const resolver = new ModuleResolver(getFileSystem());
|
||||||
|
|
||||||
|
// With relative file paths.
|
||||||
expect(resolver.resolveModuleImport('./x', _('/libs/local-package/index.js')))
|
expect(resolver.resolveModuleImport('./x', _('/libs/local-package/index.js')))
|
||||||
.toEqual(new ResolvedRelativeModule(_('/libs/local-package/x.js')));
|
.toEqual(new ResolvedRelativeModule(_('/libs/local-package/x.js')));
|
||||||
expect(resolver.resolveModuleImport('./sub-folder', _('/libs/local-package/index.js')))
|
expect(resolver.resolveModuleImport('./sub-folder', _('/libs/local-package/index.js')))
|
||||||
.toEqual(new ResolvedRelativeModule(_('/libs/local-package/sub-folder/index.js')));
|
.toEqual(new ResolvedRelativeModule(_('/libs/local-package/sub-folder/index.js')));
|
||||||
expect(resolver.resolveModuleImport('../x', _('/libs/local-package/sub-folder/index.js')))
|
expect(resolver.resolveModuleImport('../x', _('/libs/local-package/sub-folder/index.js')))
|
||||||
.toEqual(new ResolvedRelativeModule(_('/libs/local-package/x.js')));
|
.toEqual(new ResolvedRelativeModule(_('/libs/local-package/x.js')));
|
||||||
|
|
||||||
|
// With absolute file paths.
|
||||||
|
expect(resolver.resolveModuleImport(
|
||||||
|
_('/libs/local-package/x'), _('/libs/local-package/index.js')))
|
||||||
|
.toEqual(new ResolvedRelativeModule(_('/libs/local-package/x.js')));
|
||||||
|
expect(resolver.resolveModuleImport(
|
||||||
|
_('/libs/local-package/sub-folder'), _('/libs/local-package/index.js')))
|
||||||
|
.toEqual(new ResolvedRelativeModule(_('/libs/local-package/sub-folder/index.js')));
|
||||||
|
expect(resolver.resolveModuleImport(
|
||||||
|
_('/libs/local-package/x'), _('/libs/local-package/sub-folder/index.js')))
|
||||||
|
.toEqual(new ResolvedRelativeModule(_('/libs/local-package/x.js')));
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return `null` if the resolved module relative module does not exist', () => {
|
it('should return `null` if the resolved module relative module does not exist', () => {
|
||||||
|
|
|
@ -7,6 +7,8 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import * as ts from 'typescript';
|
import * as ts from 'typescript';
|
||||||
|
import {absoluteFrom as _abs} from '../../src/ngtsc/file_system';
|
||||||
|
import {runInEachFileSystem} from '../../src/ngtsc/file_system/testing';
|
||||||
import {KnownDeclaration} from '../../src/ngtsc/reflection';
|
import {KnownDeclaration} from '../../src/ngtsc/reflection';
|
||||||
import {FactoryMap, getTsHelperFnFromDeclaration, getTsHelperFnFromIdentifier, isRelativePath, stripExtension} from '../src/utils';
|
import {FactoryMap, getTsHelperFnFromDeclaration, getTsHelperFnFromIdentifier, isRelativePath, stripExtension} from '../src/utils';
|
||||||
|
|
||||||
|
@ -167,30 +169,36 @@ describe('getTsHelperFnFromIdentifier()', () => {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('isRelativePath()', () => {
|
runInEachFileSystem(() => {
|
||||||
it('should return true for relative paths', () => {
|
describe('isRelativePath()', () => {
|
||||||
expect(isRelativePath('.')).toBe(true);
|
it('should return true for relative paths', () => {
|
||||||
expect(isRelativePath('..')).toBe(true);
|
expect(isRelativePath('.')).toBe(true);
|
||||||
expect(isRelativePath('./')).toBe(true);
|
expect(isRelativePath('..')).toBe(true);
|
||||||
expect(isRelativePath('../')).toBe(true);
|
expect(isRelativePath('./')).toBe(true);
|
||||||
expect(isRelativePath('./abc/xyz')).toBe(true);
|
expect(isRelativePath('.\\')).toBe(true);
|
||||||
expect(isRelativePath('../abc/xyz')).toBe(true);
|
expect(isRelativePath('../')).toBe(true);
|
||||||
});
|
expect(isRelativePath('..\\')).toBe(true);
|
||||||
|
expect(isRelativePath('./abc/xyz')).toBe(true);
|
||||||
|
expect(isRelativePath('.\\abc\\xyz')).toBe(true);
|
||||||
|
expect(isRelativePath('../abc/xyz')).toBe(true);
|
||||||
|
expect(isRelativePath('..\\abc\\xyz')).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
it('should return true for absolute paths', () => {
|
it('should return true for absolute paths', () => {
|
||||||
expect(isRelativePath('/')).toBe(true);
|
expect(isRelativePath(_abs('/'))).toBe(true);
|
||||||
expect(isRelativePath('/abc/xyz')).toBe(true);
|
expect(isRelativePath(_abs('/abc/xyz'))).toBe(true);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return false for other paths', () => {
|
it('should return false for other paths', () => {
|
||||||
expect(isRelativePath('abc')).toBe(false);
|
expect(isRelativePath('abc')).toBe(false);
|
||||||
expect(isRelativePath('abc/xyz')).toBe(false);
|
expect(isRelativePath('abc/xyz')).toBe(false);
|
||||||
expect(isRelativePath('.abc')).toBe(false);
|
expect(isRelativePath('.abc')).toBe(false);
|
||||||
expect(isRelativePath('..abc')).toBe(false);
|
expect(isRelativePath('..abc')).toBe(false);
|
||||||
expect(isRelativePath('@abc')).toBe(false);
|
expect(isRelativePath('@abc')).toBe(false);
|
||||||
expect(isRelativePath('.abc/xyz')).toBe(false);
|
expect(isRelativePath('.abc/xyz')).toBe(false);
|
||||||
expect(isRelativePath('..abc/xyz')).toBe(false);
|
expect(isRelativePath('..abc/xyz')).toBe(false);
|
||||||
expect(isRelativePath('@abc/xyz')).toBe(false);
|
expect(isRelativePath('@abc/xyz')).toBe(false);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -7,7 +7,7 @@
|
||||||
*/
|
*/
|
||||||
export {CachedFileSystem} from './src/cached_file_system';
|
export {CachedFileSystem} from './src/cached_file_system';
|
||||||
export {NgtscCompilerHost} from './src/compiler_host';
|
export {NgtscCompilerHost} from './src/compiler_host';
|
||||||
export {absoluteFrom, absoluteFromSourceFile, basename, dirname, getFileSystem, isRoot, join, relative, relativeFrom, resolve, setFileSystem} from './src/helpers';
|
export {absoluteFrom, absoluteFromSourceFile, basename, dirname, getFileSystem, isRoot, isRooted, join, relative, relativeFrom, resolve, setFileSystem} from './src/helpers';
|
||||||
export {LogicalFileSystem, LogicalProjectPath} from './src/logical';
|
export {LogicalFileSystem, LogicalProjectPath} from './src/logical';
|
||||||
export {NodeJSFileSystem} from './src/node_js_file_system';
|
export {NodeJSFileSystem} from './src/node_js_file_system';
|
||||||
export {AbsoluteFsPath, FileStats, FileSystem, PathSegment, PathString} from './src/types';
|
export {AbsoluteFsPath, FileStats, FileSystem, PathSegment, PathString} from './src/types';
|
||||||
|
|
|
@ -37,8 +37,8 @@ export function absoluteFromSourceFile(sf: ts.SourceFile): AbsoluteFsPath {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Convert the path `path` to a `PathSegment`, throwing an error if it's not a relative path.
|
* Convert the path `path` to a `PathSegment`, throwing an error if it's not a relative path.
|
||||||
*/
|
*/
|
||||||
export function relativeFrom(path: string): PathSegment {
|
export function relativeFrom(path: string): PathSegment {
|
||||||
const normalized = normalizeSeparators(path);
|
const normalized = normalizeSeparators(path);
|
||||||
if (fs.isRooted(normalized)) {
|
if (fs.isRooted(normalized)) {
|
||||||
|
@ -73,6 +73,13 @@ export function isRoot(path: AbsoluteFsPath): boolean {
|
||||||
return fs.isRoot(path);
|
return fs.isRoot(path);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Static access to `isRooted`.
|
||||||
|
*/
|
||||||
|
export function isRooted(path: string): boolean {
|
||||||
|
return fs.isRooted(path);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Static access to `relative`.
|
* Static access to `relative`.
|
||||||
*/
|
*/
|
||||||
|
|
Loading…
Reference in New Issue