@angular-devkit/core
Version:
Angular DevKit - Core Utility Library
288 lines • 27.9 kB
JavaScript
/**
* @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
*/
Object.defineProperty(exports, "__esModule", { value: true });
exports.getSystemPath = exports.asPosixPath = exports.asWindowsPath = exports.path = exports.noCacheNormalize = exports.normalize = exports.resetNormalizeCache = exports.fragment = exports.resolve = exports.relative = exports.isAbsolute = exports.join = exports.dirname = exports.basename = exports.extname = exports.split = exports.NormalizedRoot = exports.NormalizedSep = exports.PathCannotBeFragmentException = exports.PathMustBeAbsoluteException = exports.InvalidPathException = void 0;
const exception_1 = require("../exception");
class InvalidPathException extends exception_1.BaseException {
constructor(path) {
super(`Path ${JSON.stringify(path)} is invalid.`);
}
}
exports.InvalidPathException = InvalidPathException;
class PathMustBeAbsoluteException extends exception_1.BaseException {
constructor(path) {
super(`Path ${JSON.stringify(path)} must be absolute.`);
}
}
exports.PathMustBeAbsoluteException = PathMustBeAbsoluteException;
class PathCannotBeFragmentException extends exception_1.BaseException {
constructor(path) {
super(`Path ${JSON.stringify(path)} cannot be made a fragment.`);
}
}
exports.PathCannotBeFragmentException = PathCannotBeFragmentException;
/**
* The Separator for normalized path.
* @type {Path}
*/
exports.NormalizedSep = '/';
/**
* The root of a normalized path.
* @type {Path}
*/
exports.NormalizedRoot = exports.NormalizedSep;
/**
* Split a path into multiple path fragments. Each fragments except the last one will end with
* a path separator.
* @param {Path} path The path to split.
* @returns {Path[]} An array of path fragments.
*/
function split(path) {
const fragments = path.split(exports.NormalizedSep).map((x) => fragment(x));
if (fragments[fragments.length - 1].length === 0) {
fragments.pop();
}
return fragments;
}
exports.split = split;
/**
*
*/
function extname(path) {
const base = basename(path);
const i = base.lastIndexOf('.');
if (i < 1) {
return '';
}
else {
return base.slice(i);
}
}
exports.extname = extname;
/**
* Return the basename of the path, as a Path. See path.basename
*/
function basename(path) {
const i = path.lastIndexOf(exports.NormalizedSep);
if (i == -1) {
return fragment(path);
}
else {
return fragment(path.slice(path.lastIndexOf(exports.NormalizedSep) + 1));
}
}
exports.basename = basename;
/**
* Return the dirname of the path, as a Path. See path.dirname
*/
function dirname(path) {
const index = path.lastIndexOf(exports.NormalizedSep);
if (index === -1) {
return '';
}
const endIndex = index === 0 ? 1 : index; // case of file under root: '/file'
return normalize(path.slice(0, endIndex));
}
exports.dirname = dirname;
/**
* Join multiple paths together, and normalize the result. Accepts strings that will be
* normalized as well (but the original must be a path).
*/
function join(p1, ...others) {
if (others.length > 0) {
return normalize((p1 ? p1 + exports.NormalizedSep : '') + others.join(exports.NormalizedSep));
}
else {
return p1;
}
}
exports.join = join;
/**
* Returns true if a path is absolute.
*/
function isAbsolute(p) {
return p.startsWith(exports.NormalizedSep);
}
exports.isAbsolute = isAbsolute;
/**
* Returns a path such that `join(from, relative(from, to)) == to`.
* Both paths must be absolute, otherwise it does not make much sense.
*/
function relative(from, to) {
if (!isAbsolute(from)) {
throw new PathMustBeAbsoluteException(from);
}
if (!isAbsolute(to)) {
throw new PathMustBeAbsoluteException(to);
}
let p;
if (from == to) {
p = '';
}
else {
const splitFrom = split(from);
const splitTo = split(to);
while (splitFrom.length > 0 && splitTo.length > 0 && splitFrom[0] == splitTo[0]) {
splitFrom.shift();
splitTo.shift();
}
if (splitFrom.length == 0) {
p = splitTo.join(exports.NormalizedSep);
}
else {
p = splitFrom
.map(() => '..')
.concat(splitTo)
.join(exports.NormalizedSep);
}
}
return normalize(p);
}
exports.relative = relative;
/**
* Returns a Path that is the resolution of p2, from p1. If p2 is absolute, it will return p2,
* otherwise will join both p1 and p2.
*/
function resolve(p1, p2) {
if (isAbsolute(p2)) {
return p2;
}
else {
return join(p1, p2);
}
}
exports.resolve = resolve;
function fragment(path) {
if (path.indexOf(exports.NormalizedSep) != -1) {
throw new PathCannotBeFragmentException(path);
}
return path;
}
exports.fragment = fragment;
/**
* normalize() cache to reduce computation. For now this grows and we never flush it, but in the
* future we might want to add a few cache flush to prevent this from growing too large.
*/
let normalizedCache = new Map();
/**
* Reset the cache. This is only useful for testing.
* @private
*/
function resetNormalizeCache() {
normalizedCache = new Map();
}
exports.resetNormalizeCache = resetNormalizeCache;
/**
* Normalize a string into a Path. This is the only mean to get a Path type from a string that
* represents a system path. This method cache the results as real world paths tend to be
* duplicated often.
* Normalization includes:
* - Windows backslashes `\\` are replaced with `/`.
* - Windows drivers are replaced with `/X/`, where X is the drive letter.
* - Absolute paths starts with `/`.
* - Multiple `/` are replaced by a single one.
* - Path segments `.` are removed.
* - Path segments `..` are resolved.
* - If a path is absolute, having a `..` at the start is invalid (and will throw).
* @param path The path to be normalized.
*/
function normalize(path) {
let maybePath = normalizedCache.get(path);
if (!maybePath) {
maybePath = noCacheNormalize(path);
normalizedCache.set(path, maybePath);
}
return maybePath;
}
exports.normalize = normalize;
/**
* The no cache version of the normalize() function. Used for benchmarking and testing.
*/
function noCacheNormalize(path) {
if (path == '' || path == '.') {
return '';
}
else if (path == exports.NormalizedRoot) {
return exports.NormalizedRoot;
}
// Match absolute windows path.
const original = path;
if (path.match(/^[A-Z]:[/\\]/i)) {
path = '\\' + path[0] + '\\' + path.slice(3);
}
// We convert Windows paths as well here.
const p = path.split(/[/\\]/g);
let relative = false;
let i = 1;
// Special case the first one.
if (p[0] != '') {
p.unshift('.');
relative = true;
}
while (i < p.length) {
if (p[i] == '.') {
p.splice(i, 1);
}
else if (p[i] == '..') {
if (i < 2 && !relative) {
throw new InvalidPathException(original);
}
else if (i >= 2 && p[i - 1] != '..') {
p.splice(i - 1, 2);
i--;
}
else {
i++;
}
}
else if (p[i] == '') {
p.splice(i, 1);
}
else {
i++;
}
}
if (p.length == 1) {
return p[0] == '' ? exports.NormalizedSep : '';
}
else {
if (p[0] == '.') {
p.shift();
}
return p.join(exports.NormalizedSep);
}
}
exports.noCacheNormalize = noCacheNormalize;
const path = (strings, ...values) => {
return normalize(String.raw(strings, ...values));
};
exports.path = path;
function asWindowsPath(path) {
const drive = path.match(/^\/(\w)(?:\/(.*))?$/);
if (drive) {
const subPath = drive[2] ? drive[2].replace(/\//g, '\\') : '';
return `${drive[1]}:\\${subPath}`;
}
return path.replace(/\//g, '\\');
}
exports.asWindowsPath = asWindowsPath;
function asPosixPath(path) {
return path;
}
exports.asPosixPath = asPosixPath;
function getSystemPath(path) {
if (process.platform.startsWith('win32')) {
return asWindowsPath(path);
}
else {
return asPosixPath(path);
}
}
exports.getSystemPath = getSystemPath;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"path.js","sourceRoot":"","sources":["../../../../../../../../packages/angular_devkit/core/src/virtual-fs/path.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport { BaseException } from '../exception';\nimport { TemplateTag } from '../utils/literals';\n\nexport class InvalidPathException extends BaseException {\n  constructor(path: string) {\n    super(`Path ${JSON.stringify(path)} is invalid.`);\n  }\n}\nexport class PathMustBeAbsoluteException extends BaseException {\n  constructor(path: string) {\n    super(`Path ${JSON.stringify(path)} must be absolute.`);\n  }\n}\nexport class PathCannotBeFragmentException extends BaseException {\n  constructor(path: string) {\n    super(`Path ${JSON.stringify(path)} cannot be made a fragment.`);\n  }\n}\n\n/**\n * A Path recognized by most methods in the DevKit.\n */\nexport type Path = string & {\n  __PRIVATE_DEVKIT_PATH: void;\n};\n\n/**\n * A Path fragment (file or directory name) recognized by most methods in the DevKit.\n */\nexport type PathFragment = Path & {\n  __PRIVATE_DEVKIT_PATH_FRAGMENT: void;\n};\n\n/**\n * The Separator for normalized path.\n * @type {Path}\n */\nexport const NormalizedSep = '/' as Path;\n\n/**\n * The root of a normalized path.\n * @type {Path}\n */\nexport const NormalizedRoot = NormalizedSep;\n\n/**\n * Split a path into multiple path fragments. Each fragments except the last one will end with\n * a path separator.\n * @param {Path} path The path to split.\n * @returns {Path[]} An array of path fragments.\n */\nexport function split(path: Path): PathFragment[] {\n  const fragments = path.split(NormalizedSep).map((x) => fragment(x));\n  if (fragments[fragments.length - 1].length === 0) {\n    fragments.pop();\n  }\n\n  return fragments;\n}\n\n/**\n *\n */\nexport function extname(path: Path): string {\n  const base = basename(path);\n  const i = base.lastIndexOf('.');\n  if (i < 1) {\n    return '';\n  } else {\n    return base.slice(i);\n  }\n}\n\n/**\n * Return the basename of the path, as a Path. See path.basename\n */\nexport function basename(path: Path): PathFragment {\n  const i = path.lastIndexOf(NormalizedSep);\n  if (i == -1) {\n    return fragment(path);\n  } else {\n    return fragment(path.slice(path.lastIndexOf(NormalizedSep) + 1));\n  }\n}\n\n/**\n * Return the dirname of the path, as a Path. See path.dirname\n */\nexport function dirname(path: Path): Path {\n  const index = path.lastIndexOf(NormalizedSep);\n  if (index === -1) {\n    return '' as Path;\n  }\n\n  const endIndex = index === 0 ? 1 : index; // case of file under root: '/file'\n\n  return normalize(path.slice(0, endIndex));\n}\n\n/**\n * Join multiple paths together, and normalize the result. Accepts strings that will be\n * normalized as well (but the original must be a path).\n */\nexport function join(p1: Path, ...others: string[]): Path {\n  if (others.length > 0) {\n    return normalize((p1 ? p1 + NormalizedSep : '') + others.join(NormalizedSep));\n  } else {\n    return p1;\n  }\n}\n\n/**\n * Returns true if a path is absolute.\n */\nexport function isAbsolute(p: Path) {\n  return p.startsWith(NormalizedSep);\n}\n\n/**\n * Returns a path such that `join(from, relative(from, to)) == to`.\n * Both paths must be absolute, otherwise it does not make much sense.\n */\nexport function relative(from: Path, to: Path): Path {\n  if (!isAbsolute(from)) {\n    throw new PathMustBeAbsoluteException(from);\n  }\n  if (!isAbsolute(to)) {\n    throw new PathMustBeAbsoluteException(to);\n  }\n\n  let p: string;\n\n  if (from == to) {\n    p = '';\n  } else {\n    const splitFrom = split(from);\n    const splitTo = split(to);\n\n    while (splitFrom.length > 0 && splitTo.length > 0 && splitFrom[0] == splitTo[0]) {\n      splitFrom.shift();\n      splitTo.shift();\n    }\n\n    if (splitFrom.length == 0) {\n      p = splitTo.join(NormalizedSep);\n    } else {\n      p = splitFrom\n        .map(() => '..')\n        .concat(splitTo)\n        .join(NormalizedSep);\n    }\n  }\n\n  return normalize(p);\n}\n\n/**\n * Returns a Path that is the resolution of p2, from p1. If p2 is absolute, it will return p2,\n * otherwise will join both p1 and p2.\n */\nexport function resolve(p1: Path, p2: Path) {\n  if (isAbsolute(p2)) {\n    return p2;\n  } else {\n    return join(p1, p2);\n  }\n}\n\nexport function fragment(path: string): PathFragment {\n  if (path.indexOf(NormalizedSep) != -1) {\n    throw new PathCannotBeFragmentException(path);\n  }\n\n  return path as PathFragment;\n}\n\n/**\n * normalize() cache to reduce computation. For now this grows and we never flush it, but in the\n * future we might want to add a few cache flush to prevent this from growing too large.\n */\nlet normalizedCache = new Map<string, Path>();\n\n/**\n * Reset the cache. This is only useful for testing.\n * @private\n */\nexport function resetNormalizeCache() {\n  normalizedCache = new Map<string, Path>();\n}\n\n/**\n * Normalize a string into a Path. This is the only mean to get a Path type from a string that\n * represents a system path. This method cache the results as real world paths tend to be\n * duplicated often.\n * Normalization includes:\n *   - Windows backslashes `\\\\` are replaced with `/`.\n *   - Windows drivers are replaced with `/X/`, where X is the drive letter.\n *   - Absolute paths starts with `/`.\n *   - Multiple `/` are replaced by a single one.\n *   - Path segments `.` are removed.\n *   - Path segments `..` are resolved.\n *   - If a path is absolute, having a `..` at the start is invalid (and will throw).\n * @param path The path to be normalized.\n */\nexport function normalize(path: string): Path {\n  let maybePath = normalizedCache.get(path);\n  if (!maybePath) {\n    maybePath = noCacheNormalize(path);\n    normalizedCache.set(path, maybePath);\n  }\n\n  return maybePath;\n}\n\n/**\n * The no cache version of the normalize() function. Used for benchmarking and testing.\n */\nexport function noCacheNormalize(path: string): Path {\n  if (path == '' || path == '.') {\n    return '' as Path;\n  } else if (path == NormalizedRoot) {\n    return NormalizedRoot;\n  }\n\n  // Match absolute windows path.\n  const original = path;\n  if (path.match(/^[A-Z]:[/\\\\]/i)) {\n    path = '\\\\' + path[0] + '\\\\' + path.slice(3);\n  }\n\n  // We convert Windows paths as well here.\n  const p = path.split(/[/\\\\]/g);\n  let relative = false;\n  let i = 1;\n\n  // Special case the first one.\n  if (p[0] != '') {\n    p.unshift('.');\n    relative = true;\n  }\n\n  while (i < p.length) {\n    if (p[i] == '.') {\n      p.splice(i, 1);\n    } else if (p[i] == '..') {\n      if (i < 2 && !relative) {\n        throw new InvalidPathException(original);\n      } else if (i >= 2 && p[i - 1] != '..') {\n        p.splice(i - 1, 2);\n        i--;\n      } else {\n        i++;\n      }\n    } else if (p[i] == '') {\n      p.splice(i, 1);\n    } else {\n      i++;\n    }\n  }\n\n  if (p.length == 1) {\n    return p[0] == '' ? NormalizedSep : ('' as Path);\n  } else {\n    if (p[0] == '.') {\n      p.shift();\n    }\n\n    return p.join(NormalizedSep) as Path;\n  }\n}\n\nexport const path: TemplateTag<Path> = (strings, ...values) => {\n  return normalize(String.raw(strings, ...values));\n};\n\n// Platform-specific paths.\nexport type WindowsPath = string & {\n  __PRIVATE_DEVKIT_WINDOWS_PATH: void;\n};\nexport type PosixPath = string & {\n  __PRIVATE_DEVKIT_POSIX_PATH: void;\n};\n\nexport function asWindowsPath(path: Path): WindowsPath {\n  const drive = path.match(/^\\/(\\w)(?:\\/(.*))?$/);\n  if (drive) {\n    const subPath = drive[2] ? drive[2].replace(/\\//g, '\\\\') : '';\n\n    return `${drive[1]}:\\\\${subPath}` as WindowsPath;\n  }\n\n  return path.replace(/\\//g, '\\\\') as WindowsPath;\n}\n\nexport function asPosixPath(path: Path): PosixPath {\n  return path as string as PosixPath;\n}\n\nexport function getSystemPath(path: Path): string {\n  if (process.platform.startsWith('win32')) {\n    return asWindowsPath(path);\n  } else {\n    return asPosixPath(path);\n  }\n}\n"]}
;