projen
Version:
CDK for software projects
298 lines • 36 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.codeArtifactRegex = void 0;
exports.renderBundleName = renderBundleName;
exports.extractCodeArtifactDetails = extractCodeArtifactDetails;
exports.minVersion = minVersion;
exports.tryResolveModule = tryResolveModule;
exports.tryResolveModuleManifestPath = tryResolveModuleManifestPath;
exports.tryResolveManifestPathFromDefaultExport = tryResolveManifestPathFromDefaultExport;
exports.tryResolveManifestPathFromPath = tryResolveManifestPathFromPath;
exports.tryResolveManifestPathFromSearch = tryResolveManifestPathFromSearch;
exports.tryResolveModuleManifest = tryResolveModuleManifest;
exports.tryResolveDependencyVersion = tryResolveDependencyVersion;
exports.hasDependencyVersion = hasDependencyVersion;
exports.installedVersionProbablyMatches = installedVersionProbablyMatches;
const fs_1 = require("fs");
const path_1 = require("path");
const semver = require("semver");
const node_package_1 = require("./node-package");
const util_1 = require("../util");
function renderBundleName(entrypoint) {
const parts = (0, path_1.join)(entrypoint).split(path_1.sep);
if (parts[0] === "src") {
parts.shift(); // just remove 'src' if its the first element for ergonomics
}
const p = parts.join(path_1.posix.sep);
const dir = (0, path_1.dirname)(p);
const base = (0, path_1.basename)(p, (0, path_1.extname)(p));
return path_1.posix.join(dir, base);
}
/**
* Regex for AWS CodeArtifact registry
*/
exports.codeArtifactRegex = /^https:\/\/(?<registry>(?<domain>[^\.]+)-(?<accountId>\d{12})\.d\.codeartifact\.(?<region>[^\.]+).*\.amazonaws\.com\/.*\/(?<repository>[^\/.]+)\/)/;
/**
* gets AWS details from the Code Artifact registry URL
* throws exception if not matching expected pattern
* @param registryUrl Code Artifact registry URL
* @returns object containing the (domain, accountId, region, repository)
*/
function extractCodeArtifactDetails(registryUrl) {
const match = registryUrl.match(exports.codeArtifactRegex);
if (match?.groups) {
const { domain, accountId, region, repository, registry } = match.groups;
return { domain, accountId, region, repository, registry };
}
throw new Error("Could not get CodeArtifact details from npm Registry");
}
function minVersion(version) {
if (semver.validRange(version)) {
return semver.minVersion(version)?.version;
}
else {
return version;
}
}
/**
* Attempt to resolve location of the given `moduleId`.
* @param moduleId Module ID to lookup.
* @param options Passed through to `require.resolve`.
*/
function tryResolveModule(moduleId, options) {
try {
return require.resolve(moduleId, options);
}
catch {
return undefined;
}
}
/**
* Attempt to resolve a module's manifest (package.json) path via `require.resolve` lookup.
*
* @remarks
* If the target package has `exports` that differ from the default
* (i.e, it defines the `exports` field in its manifest) and does not
* explicitly include an entry for `package.json`, this strategy will fail.
* See {@link tryResolveManifestPathFromDefaultExport} as an alternative.
*
* @param moduleId Module ID to lookup.
* @param options Passed through to `require.resolve`.
*/
function tryResolveModuleManifestPath(moduleId, options) {
// cannot just `require('dependency/package.json')` here because
// `options.paths` may not overlap with this node proc's resolution paths.
const manifestId = `${moduleId}/package.json`;
return tryResolveModule(manifestId, options);
}
/**
* Attempt to resolve a module's manifest (package.json) path by looking for the nearest
* `package.json` file that is an ancestor to the module's default export location.
*
* @param moduleId Module ID to lookup.
* @param options Passed through to `require.resolve`.
*/
function tryResolveManifestPathFromDefaultExport(moduleId, options) {
const defaultExportPath = tryResolveModule(moduleId, options);
if (!defaultExportPath) {
return undefined;
}
const moduleDir = (0, util_1.findUp)("package.json", defaultExportPath);
if (!moduleDir) {
return undefined;
}
return (0, path_1.join)(moduleDir, "package.json");
}
/**
* Attempt to resolve a module's manifest (package.json) path by checking for its existence under `node_modules` relative to `basePath`.
*
* @remarks
* This strategy can be helpful in the scenario that a module defines
* custom exports without `package.json` and no default export (i.e, some type definition packages).
*
* @param moduleId Module ID to lookup.
* @param basePath Root path to search from.
*/
function tryResolveManifestPathFromPath(moduleId, basePath) {
const base = basePath.includes("node_modules")
? basePath
: (0, path_1.join)(basePath, "node_modules");
const filePath = (0, path_1.resolve)(base, ...moduleId.split("/"), "package.json");
if ((0, fs_1.existsSync)(filePath)) {
return filePath;
}
return undefined;
}
/**
* Attempt to resolve a module's manifest (package.json) path by searching for it in the optionally provided paths array
* as well as the current node processes' default resolution paths.
* @param moduleId Module ID to search for.
* @param options Search options.
*/
function tryResolveManifestPathFromSearch(moduleId, options) {
const searchPaths = [
...(options?.paths ?? []),
...(require.resolve.paths(moduleId) ?? []),
];
for (const path of searchPaths) {
const result = tryResolveManifestPathFromPath(moduleId, path);
// early return on first result.
if (result) {
return result;
}
}
return undefined;
}
/**
* Attempt to resolve a module's manifest (package.json) using multiple strategies.
* @param moduleId Module to resolve manifest path for.
* @param options Resolution options.
*/
function tryResolveModuleManifest(moduleId, options) {
const strategies = [
tryResolveModuleManifestPath,
tryResolveManifestPathFromDefaultExport,
tryResolveManifestPathFromSearch,
];
for (const strategy of strategies) {
const result = strategy(moduleId, options);
// early return on first result.
if (result) {
try {
const manifest = JSON.parse((0, fs_1.readFileSync)(result, "utf8"));
// verify name matches target module.
if (manifest.name === moduleId) {
return manifest;
}
}
catch {
// continue to next strategy.
}
}
}
return undefined;
}
/**
* Attempt to resolve the installed version of a given dependency.
* @param dependencyName Name of dependency.
* @param options Optional options passed through to `require.resolve`.
*/
function tryResolveDependencyVersion(dependencyName, options) {
const manifest = tryResolveModuleManifest(dependencyName, options);
if (!manifest) {
return undefined;
}
return manifest?.version;
}
/**
* Whether the given dependency version is installed
*
* This can be used to test for the presence of certain versions of devDependencies,
* and do something dependency-specific in certain Components. For example, test for
* a version of Jest and generate different configs based on the Jest version.
*
* NOTE: The implementation of this function currently is currently
* approximate: to do it correctly, we would need a separate implementation
* for every package manager, to query its installed version (either that, or we
* would code to query `package-lock.json`, `yarn.lock`, etc...).
*
* Instead, we will look at `package.json`, and assume that the versions
* picked by the package manager match ~that. This will work well enough for
* major version checks, but may fail for point versions.
*
* What we SHOULD do is: `actualVersion ∈ checkRange`.
*
* What we do instead is a slightly more sophisticated version of
* `requestedRange ∩ checkRange != ∅`. This will always give a correct result if
* `requestedRange ⊆ checkRange`, but may give false positives when `checkRange
* ⊆ requestedRange`.
*
* May return `undefined` if the question cannot be answered. These include the
* following cases:
*
* - The dependency is requested via local file dependencies (`file://...`)
* - The dependency uses an other type of URL, such as a GitHub URL
* - The dependency is not found in the `package.json`, such as when
* bootstrapping from an external projen package, and the `package.json`
* file only has that one external package as a dependency
*
* Otherwise it will return `true` if the installed version is probably in the
* requested range, and `false` if it is probably not.
*
* This API may eventually be added to the public projen API, but only after
* we implement exact version checking.
*
* @param dependencyName The name of the dependency
* @param checkRange A particular version, or range of versions.
*/
function hasDependencyVersion(project, dependencyName, checkRange) {
const file = node_package_1.NodePackage.of(project)?.file;
if (!file) {
return undefined;
}
if (!(0, fs_1.existsSync)(file.absolutePath)) {
return undefined;
}
const pj = JSON.parse((0, fs_1.readFileSync)(file.absolutePath, "utf-8"));
// Technicaly, we should be intersecting all ranges to come up with the most narrow dependency
// range, but `semver` doesn't allow doing that and we don't want to add a dependency on `semver-intersect`.
//
// Let's take the first dependency declaration we find, and assume that people
// set up their `package.json` correctly.
let requestedRange;
for (const key of ["dependencies", "devDependencies", "peerDependencies"]) {
const deps = pj[key] ?? {};
let requestedVersion = deps[dependencyName];
if (requestedVersion) {
// If this is not a valid range, it could be 'file:dep.tgz', or a GitHub URL. No way to know what
// version we're getting, bail out.
if (!semver.validRange(requestedVersion)) {
return undefined;
}
requestedRange = requestedVersion;
break;
}
}
// If the dependency is not found in the `package.json`, we can't answer the question (yet).
// This means that the dependency hasn't been added yet, which means we know (upstream) what we're going to request,
// or we're going to ask for '*' and we'll get the latest version.
if (!requestedRange) {
return undefined;
}
return installedVersionProbablyMatches(requestedRange, checkRange);
}
/**
* Whether the given requestedRange *probably* leads to the installation of a version that matches checkRange
*
* We assume that NPM always installs the most recent version of a package that
* is allowed by the requestedRange.
*/
function installedVersionProbablyMatches(requestedRange, checkRange) {
const options = {
includePrerelease: true,
loose: true,
};
// No questions asked: always true
if (semver.subset(requestedRange, checkRange, options)) {
return true;
}
// Also no questions asked: always false
if (!semver.intersects(requestedRange, checkRange, options)) {
return false;
}
// Now we're in tricky territory. We intersect, but aren't a full subset.
// We are in one of the following 2 situations, which we will tie-break by
// assuming NPM will install the most recent matching version in 'requested'.
//
// requested | check | result
// -----------|----------|-----------
// >= 2 | >= 3 | probably true (chance of FP)
// <= 2 | <= 1 | probably false (change of FN)
//
// `semver` doesn't make it easy to distinguish these cases (we can't request
// the `maxVersion` that satisfies a range). Instead what we do is
// get the `minVersion` of each range, and if they compare equal we assume
// we're in the bottom case with `<=` checks, and return `false`.
return !semver.eq(semver.minVersion(requestedRange, options) ?? "1.2.3", semver.minVersion(checkRange, options) ?? "1.2.3");
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"util.js","sourceRoot":"","sources":["../../src/javascript/util.ts"],"names":[],"mappings":"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","sourcesContent":["import { existsSync, readFileSync } from \"fs\";\nimport { basename, dirname, extname, join, sep, resolve, posix } from \"path\";\nimport * as semver from \"semver\";\nimport { NodePackage } from \"./node-package\";\nimport { Project } from \"../project\";\nimport { findUp } from \"../util\";\n\n/**\n * Basic interface for `package.json`.\n */\ninterface PackageManifest {\n  [key: string]: any;\n\n  /**\n   * Package name.\n   */\n  name: string;\n  /**\n   * Package version.\n   */\n  version: string;\n}\n\nexport function renderBundleName(entrypoint: string) {\n  const parts = join(entrypoint).split(sep);\n  if (parts[0] === \"src\") {\n    parts.shift(); // just remove 'src' if its the first element for ergonomics\n  }\n\n  const p = parts.join(posix.sep);\n  const dir = dirname(p);\n  const base = basename(p, extname(p));\n  return posix.join(dir, base);\n}\n\n/**\n * Regex for AWS CodeArtifact registry\n */\nexport const codeArtifactRegex =\n  /^https:\\/\\/(?<registry>(?<domain>[^\\.]+)-(?<accountId>\\d{12})\\.d\\.codeartifact\\.(?<region>[^\\.]+).*\\.amazonaws\\.com\\/.*\\/(?<repository>[^\\/.]+)\\/)/;\n\n/**\n * gets AWS details from the Code Artifact registry URL\n * throws exception if not matching expected pattern\n * @param registryUrl Code Artifact registry URL\n * @returns object containing the (domain, accountId, region, repository)\n */\nexport function extractCodeArtifactDetails(registryUrl: string) {\n  const match = registryUrl.match(codeArtifactRegex);\n  if (match?.groups) {\n    const { domain, accountId, region, repository, registry } = match.groups;\n    return { domain, accountId, region, repository, registry };\n  }\n  throw new Error(\"Could not get CodeArtifact details from npm Registry\");\n}\n\nexport function minVersion(version: string): string | undefined {\n  if (semver.validRange(version)) {\n    return semver.minVersion(version)?.version;\n  } else {\n    return version;\n  }\n}\n\n/**\n * Attempt to resolve location of the given `moduleId`.\n * @param moduleId Module ID to lookup.\n * @param options Passed through to `require.resolve`.\n */\nexport function tryResolveModule(\n  moduleId: string,\n  options?: { paths: string[] }\n): string | undefined {\n  try {\n    return require.resolve(moduleId, options);\n  } catch {\n    return undefined;\n  }\n}\n\n/**\n * Attempt to resolve a module's manifest (package.json) path via `require.resolve` lookup.\n *\n * @remarks\n * If the target package has `exports` that differ from the default\n * (i.e, it defines the `exports` field in its manifest) and does not\n * explicitly include an entry for `package.json`, this strategy will fail.\n * See {@link tryResolveManifestPathFromDefaultExport} as an alternative.\n *\n * @param moduleId Module ID to lookup.\n * @param options Passed through to `require.resolve`.\n */\nexport function tryResolveModuleManifestPath(\n  moduleId: string,\n  options?: { paths: string[] }\n): string | undefined {\n  // cannot just `require('dependency/package.json')` here because\n  // `options.paths` may not overlap with this node proc's resolution paths.\n  const manifestId = `${moduleId}/package.json`;\n  return tryResolveModule(manifestId, options);\n}\n\n/**\n * Attempt to resolve a module's manifest (package.json) path by looking for the nearest\n * `package.json` file that is an ancestor to the module's default export location.\n *\n * @param moduleId Module ID to lookup.\n * @param options Passed through to `require.resolve`.\n */\nexport function tryResolveManifestPathFromDefaultExport(\n  moduleId: string,\n  options?: { paths: string[] }\n): string | undefined {\n  const defaultExportPath = tryResolveModule(moduleId, options);\n  if (!defaultExportPath) {\n    return undefined;\n  }\n  const moduleDir = findUp(\"package.json\", defaultExportPath);\n  if (!moduleDir) {\n    return undefined;\n  }\n  return join(moduleDir, \"package.json\");\n}\n\n/**\n * Attempt to resolve a module's manifest (package.json) path by checking for its existence under `node_modules` relative to `basePath`.\n *\n * @remarks\n * This strategy can be helpful in the scenario that a module defines\n * custom exports without `package.json` and no default export (i.e, some type definition packages).\n *\n * @param moduleId Module ID to lookup.\n * @param basePath Root path to search from.\n */\nexport function tryResolveManifestPathFromPath(\n  moduleId: string,\n  basePath: string\n) {\n  const base = basePath.includes(\"node_modules\")\n    ? basePath\n    : join(basePath, \"node_modules\");\n  const filePath = resolve(base, ...moduleId.split(\"/\"), \"package.json\");\n  if (existsSync(filePath)) {\n    return filePath;\n  }\n  return undefined;\n}\n\n/**\n * Attempt to resolve a module's manifest (package.json) path by searching for it in the optionally provided paths array\n * as well as the current node processes' default resolution paths.\n * @param moduleId Module ID to search for.\n * @param options Search options.\n */\nexport function tryResolveManifestPathFromSearch(\n  moduleId: string,\n  options?: { paths: string[] }\n): string | undefined {\n  const searchPaths = [\n    ...(options?.paths ?? []),\n    ...(require.resolve.paths(moduleId) ?? []),\n  ];\n  for (const path of searchPaths) {\n    const result = tryResolveManifestPathFromPath(moduleId, path);\n    // early return on first result.\n    if (result) {\n      return result;\n    }\n  }\n  return undefined;\n}\n\n/**\n * Attempt to resolve a module's manifest (package.json) using multiple strategies.\n * @param moduleId Module to resolve manifest path for.\n * @param options Resolution options.\n */\nexport function tryResolveModuleManifest(\n  moduleId: string,\n  options?: { paths: string[] }\n): PackageManifest | undefined {\n  const strategies = [\n    tryResolveModuleManifestPath,\n    tryResolveManifestPathFromDefaultExport,\n    tryResolveManifestPathFromSearch,\n  ];\n  for (const strategy of strategies) {\n    const result = strategy(moduleId, options);\n    // early return on first result.\n    if (result) {\n      try {\n        const manifest = JSON.parse(\n          readFileSync(result, \"utf8\")\n        ) as PackageManifest;\n        // verify name matches target module.\n        if (manifest.name === moduleId) {\n          return manifest;\n        }\n      } catch {\n        // continue to next strategy.\n      }\n    }\n  }\n  return undefined;\n}\n\n/**\n * Attempt to resolve the installed version of a given dependency.\n * @param dependencyName Name of dependency.\n * @param options Optional options passed through to `require.resolve`.\n */\nexport function tryResolveDependencyVersion(\n  dependencyName: string,\n  options?: { paths: string[] }\n): string | undefined {\n  const manifest = tryResolveModuleManifest(dependencyName, options);\n  if (!manifest) {\n    return undefined;\n  }\n  return manifest?.version;\n}\n\n/**\n * Whether the given dependency version is installed\n *\n * This can be used to test for the presence of certain versions of devDependencies,\n * and do something dependency-specific in certain Components. For example, test for\n * a version of Jest and generate different configs based on the Jest version.\n *\n * NOTE: The implementation of this function currently is currently\n * approximate: to do it correctly, we would need a separate implementation\n * for every package manager, to query its installed version (either that, or we\n * would code to query `package-lock.json`, `yarn.lock`, etc...).\n *\n * Instead, we will look at `package.json`, and assume that the versions\n * picked by the package manager match ~that. This will work well enough for\n * major version checks, but may fail for point versions.\n *\n * What we SHOULD do is: `actualVersion ∈ checkRange`.\n *\n * What we do instead is a slightly more sophisticated version of\n * `requestedRange ∩ checkRange != ∅`. This will always give a correct result if\n * `requestedRange ⊆ checkRange`, but may give false positives when `checkRange\n * ⊆ requestedRange`.\n *\n * May return `undefined` if the question cannot be answered. These include the\n * following cases:\n *\n *   - The dependency is requested via local file dependencies (`file://...`)\n *   - The dependency uses an other type of URL, such as a GitHub URL\n *   - The dependency is not found in the `package.json`, such as when\n *     bootstrapping from an external projen package, and the `package.json`\n *     file only has that one external package as a dependency\n *\n * Otherwise it will return `true` if the installed version is probably in the\n * requested range, and `false` if it is probably not.\n *\n * This API may eventually be added to the public projen API, but only after\n * we implement exact version checking.\n *\n * @param dependencyName The name of the dependency\n * @param checkRange A particular version, or range of versions.\n */\nexport function hasDependencyVersion(\n  project: Project,\n  dependencyName: string,\n  checkRange: string\n): boolean | undefined {\n  const file = NodePackage.of(project)?.file;\n  if (!file) {\n    return undefined;\n  }\n\n  if (!existsSync(file.absolutePath)) {\n    return undefined;\n  }\n\n  const pj = JSON.parse(readFileSync(file.absolutePath, \"utf-8\"));\n\n  // Technicaly, we should be intersecting all ranges to come up with the most narrow dependency\n  // range, but `semver` doesn't allow doing that and we don't want to add a dependency on `semver-intersect`.\n  //\n  // Let's take the first dependency declaration we find, and assume that people\n  // set up their `package.json` correctly.\n  let requestedRange: string | undefined;\n  for (const key of [\"dependencies\", \"devDependencies\", \"peerDependencies\"]) {\n    const deps = pj[key] ?? {};\n    let requestedVersion = deps[dependencyName];\n    if (requestedVersion) {\n      // If this is not a valid range, it could be 'file:dep.tgz', or a GitHub URL. No way to know what\n      // version we're getting, bail out.\n      if (!semver.validRange(requestedVersion)) {\n        return undefined;\n      }\n      requestedRange = requestedVersion;\n      break;\n    }\n  }\n\n  // If the dependency is not found in the `package.json`, we can't answer the question (yet).\n  // This means that the dependency hasn't been added yet, which means we know (upstream) what we're going to request,\n  // or we're going to ask for '*' and we'll get the latest version.\n  if (!requestedRange) {\n    return undefined;\n  }\n\n  return installedVersionProbablyMatches(requestedRange, checkRange);\n}\n\n/**\n * Whether the given requestedRange *probably* leads to the installation of a version that matches checkRange\n *\n * We assume that NPM always installs the most recent version of a package that\n * is allowed by the requestedRange.\n */\nexport function installedVersionProbablyMatches(\n  requestedRange: string,\n  checkRange: string\n): boolean {\n  const options = {\n    includePrerelease: true,\n    loose: true,\n  };\n\n  // No questions asked: always true\n  if (semver.subset(requestedRange, checkRange, options)) {\n    return true;\n  }\n\n  // Also no questions asked: always false\n  if (!semver.intersects(requestedRange, checkRange, options)) {\n    return false;\n  }\n\n  // Now we're in tricky territory. We intersect, but aren't a full subset.\n  // We are in one of the following 2 situations, which we will tie-break by\n  // assuming NPM will install the most recent matching version in 'requested'.\n  //\n  // requested  | check    | result\n  // -----------|----------|-----------\n  //   >= 2     |  >= 3    | probably true (chance of FP)\n  //   <= 2     |  <= 1    | probably false (change of FN)\n  //\n  // `semver` doesn't make it easy to distinguish these cases (we can't request\n  // the `maxVersion` that satisfies a range). Instead what we do is\n  // get the `minVersion` of each range, and if they compare equal we assume\n  // we're in the bottom case with `<=` checks, and return `false`.\n\n  return !semver.eq(\n    semver.minVersion(requestedRange, options) ?? \"1.2.3\",\n    semver.minVersion(checkRange, options) ?? \"1.2.3\"\n  );\n}\n"]}