UNPKG

jsii-docgen

Version:

generates api docs for jsii modules

343 lines 43.7 kB
"use strict"; var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { Object.defineProperty(o, "default", { enumerable: true, value: v }); }) : function(o, v) { o["default"] = v; }); var __importStar = (this && this.__importStar) || (function () { var ownKeys = function(o) { ownKeys = Object.getOwnPropertyNames || function (o) { var ar = []; for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k; return ar; }; return ownKeys(o); }; return function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]); __setModuleDefault(result, mod); return result; }; })(); var __classPrivateFieldSet = (this && this.__classPrivateFieldSet) || function (receiver, state, value, kind, f) { if (kind === "m") throw new TypeError("Private method is not writable"); if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a setter"); if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot write private member to an object whose class did not declare it"); return (kind === "a" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value; }; var __classPrivateFieldGet = (this && this.__classPrivateFieldGet) || function (receiver, state, kind, f) { if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a getter"); if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot read private member from an object whose class did not declare it"); return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver); }; var _Npm_npmCommand; Object.defineProperty(exports, "__esModule", { value: true }); exports.OptionalPeerDepsFilter = exports.Npm = void 0; const child_process_1 = require("child_process"); const fs_1 = require("fs"); const os = __importStar(require("os")); const path_1 = require("path"); const semver_1 = require("semver"); const documentation_1 = require("./documentation"); const errors_1 = require("../../errors"); class Npm { constructor(workingDirectory, logger = console.log, npmCommand) { this.workingDirectory = workingDirectory; this.logger = logger; _Npm_npmCommand.set(this, void 0); __classPrivateFieldSet(this, _Npm_npmCommand, npmCommand, "f"); } /** * Installs the designated package into this repository's working directory. * * @param target the name or path to the package that needs to be installed. * @param force whether to pass `--force` to `npm install`. * * @returns the name of the package that was installed. */ async install(target, force = false) { const commonFlags = [ ...force ? [ // force install, ignoring recommended protections such as platform checks. This is okay // because we are not actually executing the code being installed in this context. '--force', ] : [], // this is critical from a security perspective to prevent // code execution as part of the install command using npm hooks. (e.g postInstall) '--ignore-scripts', // save time by not running audit '--no-audit', // ensures npm does not insert anything in $PATH '--no-bin-links', // don't write or update a package-lock.json file '--no-package-lock', // always produce JSON output '--json', ]; try { assertSuccess(await this.runCommand(await this.npmCommandPath(), [ 'install', JSON.stringify(target), ...commonFlags, // ensures we are installing devDependencies, too. '--include=dev', '--include=peer', '--include=optional', // Make sure we get a `package.json` so we can figure out the actual package name. '--save', ], chunksToObject, { cwd: this.workingDirectory, shell: true, })); const { dependencies } = JSON.parse(await fs_1.promises.readFile((0, path_1.join)(this.workingDirectory, 'package.json'), 'utf-8')); const names = Object.keys(dependencies !== null && dependencies !== void 0 ? dependencies : {}); const name = names.length === 1 ? names[0] : (0, documentation_1.extractPackageName)(target); const optionalPeerDeps = await this.listOptionalPeerDeps(name); if (optionalPeerDeps.length > 0) { assertSuccess(await this.runCommand(await this.npmCommandPath(), [ 'install', ...optionalPeerDeps, ...commonFlags, // Save as optional in the root package.json (courtesy) '--save-optional', ], chunksToObject, { cwd: this.workingDirectory, shell: true, })); } return name; } catch (e) { if (!force && (e instanceof errors_1.NpmError) && e.npmErrorCode === 'EBADPLATFORM') { console.warn('npm install failed with EBADPLATFORM, retrying with --force'); return this.install(target, true); } return Promise.reject(e); } } async listOptionalPeerDeps(target) { var _a; const result = new Array(); const packageJson = JSON.parse(await fs_1.promises.readFile((0, path_1.join)(this.workingDirectory, 'node_modules', target, 'package.json'), 'utf-8')); for (const [name, { optional }] of Object.entries((_a = packageJson.peerDependenciesMeta) !== null && _a !== void 0 ? _a : {})) { if (!optional) { continue; } const version = packageJson.peerDependencies[name]; if (version == null) { continue; } result.push(JSON.stringify(`${name}@${version}`)); } return result; } /** * Obtains the path to the npm command that should be run. This always returns * the path to an npm >= 7, which "correctly" handles peerDependencies. If the * npm version that's available in $PATH satisfies this predicate, this will * simply return `npm`. */ async npmCommandPath() { if (__classPrivateFieldGet(this, _Npm_npmCommand, "f")) { return __classPrivateFieldGet(this, _Npm_npmCommand, "f"); } // Get the platform specific npm command const npm = npmPlatformAwareCommand(); try { // If the npm in $PATH is >= v7, we can use that directly. The // `npm version --json` command returns a JSON object containing the // versions of several components (npm, node, v8, etc...). We are only // interested in the `npm` key here. const { exitCode, stdout } = await this.runCommand(npm, ['version', '--json'], chunksToObject); if (exitCode === 0 && (0, semver_1.major)(stdout.npm) >= 7) { return __classPrivateFieldSet(this, _Npm_npmCommand, npm, "f"); } } catch (e) { this.logger('Could not determine version of npm in $PATH:', e); } // npm@8 is needed so that we also install peerDependencies - they are needed to construct // the full type system. this.logger('The npm in $PATH is not >= v7. Installing npm@8 locally...'); const result = await this.runCommand(npm, ['install', 'npm@8', '--no-package-lock', '--no-save', '--json'], chunksToObject, { cwd: this.workingDirectory, shell: true, }); assertSuccess(result); __classPrivateFieldSet(this, _Npm_npmCommand, (0, path_1.join)(this.workingDirectory, 'node_modules', '.bin', npm), "f"); this.logger(`Done installing npm@8 at ${__classPrivateFieldGet(this, _Npm_npmCommand, "f")}`); return __classPrivateFieldGet(this, _Npm_npmCommand, "f"); } /** * Runs the supplied command with the provided arguments, captures the data * pushed to STDOUT, and "parses" it using `outputTransform` to produce a * result. * * You must consult the `exitCode` of the return value to determine whether * the command was successful or not. Use the `assertSuccess` function to * throw/reject in case the execution was not successful. * * @param command the command to invoke. * @param args arguments to provide to the command. * @param outputTransform the function that will parse STDOUT data. * @param options additional `spawn` options, if necessary. */ async runCommand(command, args, outputTransform, options) { return new Promise((ok, ko) => { // On Windows, spawning a program ending in .cmd or .bat needs to run in a shell // https://nodejs.org/en/blog/vulnerability/april-2024-security-releases-2 const shell = onWindows() && (command.endsWith('.cmd') || command.endsWith('.bat')); const child = (0, child_process_1.spawn)(command, args, { shell, ...options, stdio: ['inherit', 'pipe', 'pipe'] }); const stdout = new Array(); child.stdout.on('data', (chunk) => { stdout.push(Buffer.from(chunk)); }); child.stderr.on('data', (chunk) => { stdout.push(Buffer.from(chunk)); }); child.once('error', ko); child.once('close', (exitCode, signal) => { try { ok({ command: `${command} ${args.join(' ')}`, exitCode, signal, stdout: outputTransform(stdout), }); } catch (error) { ko(error); } }); }); } } exports.Npm = Npm; _Npm_npmCommand = new WeakMap(); /** * A filter to apply when selecting optional peer dependencies, based on how * their version target is specified. */ var OptionalPeerDepsFilter; (function (OptionalPeerDepsFilter) { /** * Ignore all optional peer dependencies when installing. */ OptionalPeerDepsFilter[OptionalPeerDepsFilter["None"] = 0] = "None"; /** * Install only optional peer dependencies specified as a version range, and * ignore those specified as a URL or local path. */ OptionalPeerDepsFilter[OptionalPeerDepsFilter["VersionRange"] = 1] = "VersionRange"; /** * Install all optional peer dependencies regardless of how they are * specified. This requires URL and local-path dependencies to be reachable. */ OptionalPeerDepsFilter[OptionalPeerDepsFilter["All"] = 2] = "All"; })(OptionalPeerDepsFilter || (exports.OptionalPeerDepsFilter = OptionalPeerDepsFilter = {})); /** * Asserts the provided CommandResult corresponds to a command that exited with * code `0`. If that is not the case, this will throw an appropriate error, * either `NpmError` or `NoSpaceLeftOnDevice`. */ function assertSuccess(result) { var _a; const { command, exitCode, signal, stdout } = result; if (exitCode === 0) { return; } if (signal != null) { throw new errors_1.NpmError(`Command "${command}" was killed by ${signal}`, stdout); } if (exitCode === 228 || ((_a = stdout.error) === null || _a === void 0 ? void 0 : _a.code) === 'ENOSPC') { throw new errors_1.NoSpaceLeftOnDevice(`Command "${command}" failed due to insufficient available disk space`); } const { code, detail, summary } = stdout.error; const message = [ `Command "${command}" exited with code ${exitCode}`, summary ? `: ${summary}` : '', detail ? `\n${detail}` : '', // If we have an error, but neither detail nor summary, then we probably // have an actual Error object, so we'll stringify that here... stdout.error && !detail && !summary ? `: ${stdout.error}` : '', ].join(''); if (typeof summary === 'string' && summary.includes('must provide string spec')) { // happens when package.json dependencies don't have a spec. // for example: https://github.com/markusl/cdk-codepipeline-bitbucket-build-result-reporter/blob/v0.0.7/package.json throw new errors_1.UnInstallablePackageError(summary); } // happens when a package has been deleted from npm // for example: sns-app-jsii-component if (!code && !detail && typeof summary === 'string' && summary.includes('Cannot convert undefined or null to object')) { throw new errors_1.UnInstallablePackageError(summary); } switch (code) { case 'E404': // package (or dependency) can't be found on NPM. This can happen if the package depends on a deprecated package (for example). case 'EOVERRIDE': // Package contains some version overrides that conflict. case 'ERESOLVE': // dependency resolution problem requires a manual intervention (most likely...) case 'ENOVERSIONS': // package has been removed from npm throw new errors_1.UnInstallablePackageError(message); default: throw new errors_1.NpmError(message, stdout, code); } } /** * Concatenates the provided chunks into a single Buffer, converts it to a * string using the designated encoding, then JSON-parses it. If any part of * this process results in an error, returns an object that contains the error * and the raw chunks. */ function chunksToObject(chunks, encoding = 'utf-8') { const raw = Buffer.concat(chunks).toString(encoding); try { // npm will sometimes print non json log lines even though --json was requested. // observed these log lines always start with 'npm', so we filter those out. // for example: "npm notice New patch version of npm available! 8.1.0 -> 8.1.3" // for example: "npm ERR! must provide string spec" const onlyJson = raw.split(/[\r\n]+/) // split on any newlines, because npm returns inconsistent newline characters on Windows .filter(l => !l.startsWith('npm')) // Suppress debugger messages, if present... .filter(l => l !== 'Debugger attached.') .filter(l => l !== 'Waiting for the debugger to disconnect...') // Re-join... .join(os.EOL); return JSON.parse(onlyJson); } catch (error) { return { error, raw }; } } /** * Helper to detect if we are running on Windows. */ function onWindows() { return process.platform === 'win32'; } /** * Get the npm binary path depending on the platform. * @returns "npm.cmd" on Windows, otherwise "npm" */ function npmPlatformAwareCommand() { if (onWindows()) { return 'npm.cmd'; } return 'npm'; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"_npm.js","sourceRoot":"","sources":["../../../src/docgen/view/_npm.ts"],"names":[],"mappings":"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","sourcesContent":["import { spawn, SpawnOptionsWithoutStdio } from 'child_process';\nimport { promises as fs } from 'fs';\nimport * as os from 'os';\nimport { join } from 'path';\nimport { major } from 'semver';\nimport { extractPackageName } from './documentation';\nimport { NoSpaceLeftOnDevice, UnInstallablePackageError, NpmError } from '../../errors';\n\nexport class Npm {\n  #npmCommand: string | undefined;\n\n  public constructor(\n    private readonly workingDirectory: string,\n    private readonly logger = console.log,\n    npmCommand?: string,\n  ) {\n    this.#npmCommand = npmCommand;\n  }\n\n  /**\n   * Installs the designated package into this repository's working directory.\n   *\n   * @param target the name or path to the package that needs to be installed.\n   * @param force whether to pass `--force` to `npm install`.\n   *\n   * @returns the name of the package that was installed.\n   */\n  public async install(target: string, force = false): Promise<string> {\n    const commonFlags = [\n      ...force\n        ? [\n          // force install, ignoring recommended protections such as platform checks. This is okay\n          // because we are not actually executing the code being installed in this context.\n          '--force',\n        ]\n        : [],\n      // this is critical from a security perspective to prevent\n      // code execution as part of the install command using npm hooks. (e.g postInstall)\n      '--ignore-scripts',\n      // save time by not running audit\n      '--no-audit',\n      // ensures npm does not insert anything in $PATH\n      '--no-bin-links',\n      // don't write or update a package-lock.json file\n      '--no-package-lock',\n      // always produce JSON output\n      '--json',\n    ];\n\n    try {\n      assertSuccess(await this.runCommand(\n        await this.npmCommandPath(),\n        [\n          'install',\n          JSON.stringify(target),\n          ...commonFlags,\n          // ensures we are installing devDependencies, too.\n          '--include=dev',\n          '--include=peer',\n          '--include=optional',\n          // Make sure we get a `package.json` so we can figure out the actual package name.\n          '--save',\n        ],\n        chunksToObject,\n        {\n          cwd: this.workingDirectory,\n          shell: true,\n        },\n      ));\n\n      const { dependencies } = JSON.parse(await fs.readFile(join(this.workingDirectory, 'package.json'), 'utf-8'));\n      const names = Object.keys(dependencies ?? {});\n      const name = names.length === 1\n        ? names[0]\n        : extractPackageName(target);\n\n      const optionalPeerDeps = await this.listOptionalPeerDeps(name);\n      if (optionalPeerDeps.length > 0) {\n        assertSuccess(await this.runCommand(\n          await this.npmCommandPath(),\n          [\n            'install',\n            ...optionalPeerDeps,\n            ...commonFlags,\n            // Save as optional in the root package.json (courtesy)\n            '--save-optional',\n          ],\n          chunksToObject,\n          {\n            cwd: this.workingDirectory,\n            shell: true,\n          },\n        ));\n      }\n\n      return name;\n    } catch (e) {\n      if (!force && (e instanceof NpmError) && e.npmErrorCode === 'EBADPLATFORM') {\n        console.warn('npm install failed with EBADPLATFORM, retrying with --force');\n        return this.install(target, true);\n      }\n      return Promise.reject(e);\n    }\n  }\n\n  private async listOptionalPeerDeps(target: string): Promise<readonly string[]> {\n    const result = new Array<string>();\n\n    const packageJson = JSON.parse(await fs.readFile(join(this.workingDirectory, 'node_modules', target, 'package.json'), 'utf-8'));\n    for (const [name, { optional }] of Object.entries(packageJson.peerDependenciesMeta ?? {}) as [string, { optional: boolean }][]) {\n      if (!optional) {\n        continue;\n      }\n      const version = packageJson.peerDependencies[name];\n      if (version == null) {\n        continue;\n      }\n      result.push(JSON.stringify(`${name}@${version}`));\n    }\n\n    return result;\n  }\n\n  /**\n   * Obtains the path to the npm command that should be run. This always returns\n   * the path to an npm >= 7, which \"correctly\" handles peerDependencies. If the\n   * npm version that's available in $PATH satisfies this predicate, this will\n   * simply return `npm`.\n   */\n  private async npmCommandPath(): Promise<string> {\n    if (this.#npmCommand) {\n      return this.#npmCommand;\n    }\n\n    // Get the platform specific npm command\n    const npm = npmPlatformAwareCommand();\n\n    try {\n      // If the npm in $PATH is >= v7, we can use that directly. The\n      // `npm version --json` command returns a JSON object containing the\n      // versions of several components (npm, node, v8, etc...). We are only\n      // interested in the `npm` key here.\n      const { exitCode, stdout } = await this.runCommand(\n        npm, ['version', '--json'],\n        chunksToObject,\n      );\n      if (exitCode === 0 && major((stdout as any).npm) >= 7) {\n        return this.#npmCommand = npm;\n      }\n    } catch (e) {\n      this.logger('Could not determine version of npm in $PATH:', e);\n    }\n\n    // npm@8 is needed so that we also install peerDependencies - they are needed to construct\n    // the full type system.\n    this.logger('The npm in $PATH is not >= v7. Installing npm@8 locally...');\n    const result = await this.runCommand(\n      npm,\n      ['install', 'npm@8', '--no-package-lock', '--no-save', '--json'],\n      chunksToObject,\n      {\n        cwd: this.workingDirectory,\n        shell: true,\n      },\n    );\n    assertSuccess(result);\n\n    this.#npmCommand = join(this.workingDirectory, 'node_modules', '.bin', npm);\n    this.logger(`Done installing npm@8 at ${this.#npmCommand}`);\n    return this.#npmCommand;\n  }\n\n  /**\n   * Runs the supplied command with the provided arguments, captures the data\n   * pushed to STDOUT, and \"parses\" it using `outputTransform` to produce a\n   * result.\n   *\n   * You must consult the `exitCode` of the return value to determine whether\n   * the command was successful or not. Use the `assertSuccess` function to\n   * throw/reject in case the execution was not successful.\n   *\n   * @param command         the command to invoke.\n   * @param args            arguments to provide to the command.\n   * @param outputTransform the function that will parse STDOUT data.\n   * @param options         additional `spawn` options, if necessary.\n   */\n  private async runCommand<T = Buffer>(\n    command: string,\n    args: readonly string[],\n    outputTransform: (stderr: readonly Buffer[]) => T,\n    options?: SpawnOptionsWithoutStdio,\n  ): Promise<CommandResult<T>> {\n    return new Promise<CommandResult<T>>((ok, ko) => {\n      // On Windows, spawning a program ending in .cmd or .bat needs to run in a shell\n      // https://nodejs.org/en/blog/vulnerability/april-2024-security-releases-2\n      const shell = onWindows() && (command.endsWith('.cmd') || command.endsWith('.bat'));\n      const child = spawn(command, args, { shell, ...options, stdio: ['inherit', 'pipe', 'pipe'] });\n      const stdout = new Array<Buffer>();\n      child.stdout.on('data', (chunk) => {\n        stdout.push(Buffer.from(chunk));\n      });\n      child.stderr.on('data', (chunk) => {\n        stdout.push(Buffer.from(chunk));\n      });\n\n      child.once('error', ko);\n      child.once('close', (exitCode, signal) => {\n        try {\n          ok({\n            command: `${command} ${args.join(' ')}`,\n            exitCode,\n            signal,\n            stdout: outputTransform(stdout),\n          });\n        } catch (error) {\n          ko(error);\n        }\n      });\n    });\n  }\n}\n\n/**\n * A filter to apply when selecting optional peer dependencies, based on how\n * their version target is specified.\n */\nexport enum OptionalPeerDepsFilter {\n  /**\n   * Ignore all optional peer dependencies when installing.\n   */\n  None,\n\n  /**\n   * Install only optional peer dependencies specified as a version range, and\n   * ignore those specified as a URL or local path.\n   */\n  VersionRange,\n\n  /**\n   * Install all optional peer dependencies regardless of how they are\n   * specified. This requires URL and local-path dependencies to be reachable.\n   */\n  All,\n}\n\ninterface CommandResult<T> {\n  readonly command: string;\n  readonly exitCode: number | null;\n  readonly signal: NodeJS.Signals | null;\n  readonly stdout: T;\n}\ninterface SuccessfulCommandResult<T> extends CommandResult<T> {\n  readonly exitCode: 0;\n  readonly signal: null;\n}\n\n/**\n * Asserts the provided CommandResult corresponds to a command that exited with\n * code `0`. If that is not the case, this will throw an appropriate error,\n * either `NpmError` or `NoSpaceLeftOnDevice`.\n */\nfunction assertSuccess(result: CommandResult<ResponseObject>): asserts result is SuccessfulCommandResult<ResponseObject> {\n  const { command, exitCode, signal, stdout } = result;\n  if (exitCode === 0) {\n    return;\n  }\n  if (signal != null) {\n    throw new NpmError(`Command \"${command}\" was killed by ${signal}`, stdout);\n  }\n  if (exitCode === 228 || stdout.error?.code === 'ENOSPC') {\n    throw new NoSpaceLeftOnDevice(`Command \"${command}\" failed due to insufficient available disk space`);\n  }\n  const { code, detail, summary } = stdout.error;\n  const message = [\n    `Command \"${command}\" exited with code ${exitCode}`,\n    summary ? `: ${summary}` : '',\n    detail ? `\\n${detail}` : '',\n    // If we have an error, but neither detail nor summary, then we probably\n    // have an actual Error object, so we'll stringify that here...\n    stdout.error && !detail && !summary ? `: ${stdout.error}` : '',\n  ].join('');\n\n  if (typeof summary === 'string' && summary.includes('must provide string spec')) {\n    // happens when package.json dependencies don't have a spec.\n    // for example: https://github.com/markusl/cdk-codepipeline-bitbucket-build-result-reporter/blob/v0.0.7/package.json\n    throw new UnInstallablePackageError(summary);\n  }\n\n  // happens when a package has been deleted from npm\n  // for example: sns-app-jsii-component\n  if (!code && !detail && typeof summary === 'string' && summary.includes('Cannot convert undefined or null to object')) {\n    throw new UnInstallablePackageError(summary);\n  }\n\n  switch (code) {\n    case 'E404': // package (or dependency) can't be found on NPM. This can happen if the package depends on a deprecated package (for example).\n    case 'EOVERRIDE': // Package contains some version overrides that conflict.\n    case 'ERESOLVE': // dependency resolution problem requires a manual intervention (most likely...)\n    case 'ENOVERSIONS': // package has been removed from npm\n      throw new UnInstallablePackageError(message);\n    default:\n      throw new NpmError(message, stdout, code);\n  }\n}\n\n/**\n * Concatenates the provided chunks into a single Buffer, converts it to a\n * string using the designated encoding, then JSON-parses it. If any part of\n * this process results in an error, returns an object that contains the error\n * and the raw chunks.\n */\nfunction chunksToObject(chunks: readonly Buffer[], encoding: BufferEncoding = 'utf-8'): ResponseObject {\n  const raw = Buffer.concat(chunks).toString(encoding);\n  try {\n    // npm will sometimes print non json log lines even though --json was requested.\n    // observed these log lines always start with 'npm', so we filter those out.\n    // for example: \"npm notice New patch version of npm available! 8.1.0 -> 8.1.3\"\n    // for example: \"npm ERR! must provide string spec\"\n    const onlyJson = raw.split(/[\\r\\n]+/) // split on any newlines, because npm returns inconsistent newline characters on Windows\n      .filter(l => !l.startsWith('npm'))\n      // Suppress debugger messages, if present...\n      .filter(l => l !== 'Debugger attached.')\n      .filter(l => l !== 'Waiting for the debugger to disconnect...')\n      // Re-join...\n      .join(os.EOL);\n    return JSON.parse(onlyJson);\n  } catch (error) {\n    return { error, raw };\n  }\n}\n\ntype ResponseObject =\n  // The error when we failed to parse the output as JSON\n  | { readonly error: any; readonly raw: string }\n  // The error objects npm returns when operating in --json mode\n  | { readonly error: { readonly code: string; readonly summary: string; readonly detail: string } }\n  // The successful objects are treated as opaque blobs here\n  | { readonly error: undefined; readonly [key: string]: unknown };\n\n/**\n * Helper to detect if we are running on Windows.\n */\nfunction onWindows() {\n  return process.platform === 'win32';\n}\n\n/**\n * Get the npm binary path depending on the platform.\n * @returns \"npm.cmd\" on Windows, otherwise \"npm\"\n */\nfunction npmPlatformAwareCommand() {\n  if (onWindows()) {\n    return 'npm.cmd';\n  }\n\n  return 'npm';\n}\n"]}