"use strict"; var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.makePatch = void 0; const chalk_1 = __importDefault(require("chalk")); const path_1 = require("./path"); const spawnSafe_1 = require("./spawnSafe"); const filterFiles_1 = require("./filterFiles"); const fs_extra_1 = require("fs-extra"); const rimraf_1 = require("rimraf"); const fs_extra_2 = require("fs-extra"); const tmp_1 = require("tmp"); const patchFs_1 = require("./patchFs"); const PackageDetails_1 = require("./PackageDetails"); const resolveRelativeFileDependencies_1 = require("./resolveRelativeFileDependencies"); const getPackageResolution_1 = require("./getPackageResolution"); const parse_1 = require("./patch/parse"); const zlib_1 = require("zlib"); const getPackageVersion_1 = require("./getPackageVersion"); const createIssue_1 = require("./createIssue"); function printNoPackageFoundError(packageName, packageJsonPath) { console.error(`No such package ${packageName} File not found: ${packageJsonPath}`); } function makePatch({ packagePathSpecifier, appPath, packageManager, includePaths, excludePaths, patchDir, createIssue, }) { const packageDetails = PackageDetails_1.getPatchDetailsFromCliString(packagePathSpecifier); if (!packageDetails) { console.error("No such package", packagePathSpecifier); return; } const appPackageJson = require(path_1.join(appPath, "package.json")); const packagePath = path_1.join(appPath, packageDetails.path); const packageJsonPath = path_1.join(packagePath, "package.json"); if (!fs_extra_1.existsSync(packageJsonPath)) { printNoPackageFoundError(packagePathSpecifier, packageJsonPath); process.exit(1); } const tmpRepo = tmp_1.dirSync({ unsafeCleanup: true }); const tmpRepoPackagePath = path_1.join(tmpRepo.name, packageDetails.path); const tmpRepoNpmRoot = tmpRepoPackagePath.slice(0, -`/node_modules/${packageDetails.name}`.length); const tmpRepoPackageJsonPath = path_1.join(tmpRepoNpmRoot, "package.json"); try { const patchesDir = path_1.resolve(path_1.join(appPath, patchDir)); console.info(chalk_1.default.grey("•"), "Creating temporary folder"); // make a blank package.json fs_extra_1.mkdirpSync(tmpRepoNpmRoot); fs_extra_1.writeFileSync(tmpRepoPackageJsonPath, JSON.stringify({ dependencies: { [packageDetails.name]: getPackageResolution_1.getPackageResolution({ packageDetails, packageManager, appPath, }), }, resolutions: resolveRelativeFileDependencies_1.resolveRelativeFileDependencies(appPath, appPackageJson.resolutions || {}), })); const packageVersion = getPackageVersion_1.getPackageVersion(path_1.join(path_1.resolve(packageDetails.path), "package.json")); [".npmrc", ".yarnrc"].forEach((rcFile) => { const rcPath = path_1.join(appPath, rcFile); if (fs_extra_1.existsSync(rcPath)) { fs_extra_2.copySync(rcPath, path_1.join(tmpRepo.name, rcFile)); } }); if (packageManager === "yarn") { console.info(chalk_1.default.grey("•"), `Installing ${packageDetails.name}@${packageVersion} with yarn`); try { // try first without ignoring scripts in case they are required // this works in 99.99% of cases spawnSafe_1.spawnSafeSync(`yarn`, ["install", "--ignore-engines"], { cwd: tmpRepoNpmRoot, logStdErrOnError: false, }); } catch (e) { // try again while ignoring scripts in case the script depends on // an implicit context which we havn't reproduced spawnSafe_1.spawnSafeSync(`yarn`, ["install", "--ignore-engines", "--ignore-scripts"], { cwd: tmpRepoNpmRoot, }); } } else { console.info(chalk_1.default.grey("•"), `Installing ${packageDetails.name}@${packageVersion} with npm`); try { // try first without ignoring scripts in case they are required // this works in 99.99% of cases spawnSafe_1.spawnSafeSync(`npm`, ["i", "--force"], { cwd: tmpRepoNpmRoot, logStdErrOnError: false, stdio: "ignore", }); } catch (e) { // try again while ignoring scripts in case the script depends on // an implicit context which we havn't reproduced spawnSafe_1.spawnSafeSync(`npm`, ["i", "--ignore-scripts", "--force"], { cwd: tmpRepoNpmRoot, stdio: "ignore", }); } } const git = (...args) => spawnSafe_1.spawnSafeSync("git", args, { cwd: tmpRepo.name, env: Object.assign(Object.assign({}, process.env), { HOME: tmpRepo.name }), maxBuffer: 1024 * 1024 * 100, }); // remove nested node_modules just to be safe rimraf_1.sync(path_1.join(tmpRepoPackagePath, "node_modules")); // remove .git just to be safe rimraf_1.sync(path_1.join(tmpRepoPackagePath, ".git")); // commit the package console.info(chalk_1.default.grey("•"), "Diffing your files with clean files"); fs_extra_1.writeFileSync(path_1.join(tmpRepo.name, ".gitignore"), "!/node_modules\n\n"); git("init"); git("config", "--local", "user.name", "patch-package"); git("config", "--local", "user.email", "patch@pack.age"); // remove ignored files first filterFiles_1.removeIgnoredFiles(tmpRepoPackagePath, includePaths, excludePaths); git("add", "-f", packageDetails.path); git("commit", "--allow-empty", "-m", "init"); // replace package with user's version rimraf_1.sync(tmpRepoPackagePath); // pnpm installs packages as symlinks, copySync would copy only the symlink fs_extra_2.copySync(fs_extra_1.realpathSync(packagePath), tmpRepoPackagePath); // remove nested node_modules just to be safe rimraf_1.sync(path_1.join(tmpRepoPackagePath, "node_modules")); // remove .git just to be safe rimraf_1.sync(path_1.join(tmpRepoPackagePath, ".git")); // also remove ignored files like before filterFiles_1.removeIgnoredFiles(tmpRepoPackagePath, includePaths, excludePaths); // stage all files git("add", "-f", packageDetails.path); // get diff of changes const diffResult = git("diff", "--cached", "--no-color", "--ignore-space-at-eol", "--no-ext-diff"); if (diffResult.stdout.length === 0) { console.warn(`⁉️ Not creating patch file for package '${packagePathSpecifier}'`); console.warn(`⁉️ There don't appear to be any changes.`); process.exit(1); return; } try { parse_1.parsePatchFile(diffResult.stdout.toString()); } catch (e) { if (e.message.includes("Unexpected file mode string: 120000")) { console.error(` ⛔️ ${chalk_1.default.red.bold("ERROR")} Your changes involve creating symlinks. patch-package does not yet support symlinks. ️Please use ${chalk_1.default.bold("--include")} and/or ${chalk_1.default.bold("--exclude")} to narrow the scope of your patch if this was unintentional. `); } else { const outPath = "./patch-package-error.json.gz"; fs_extra_1.writeFileSync(outPath, zlib_1.gzipSync(JSON.stringify({ error: { message: e.message, stack: e.stack }, patch: diffResult.stdout.toString(), }))); console.error(` ⛔️ ${chalk_1.default.red.bold("ERROR")} patch-package was unable to read the patch-file made by git. This should not happen. A diagnostic file was written to ${outPath} Please attach it to a github issue https://github.com/ds300/patch-package/issues/new?title=New+patch+parse+failed&body=Please+attach+the+diagnostic+file+by+dragging+it+into+here+🙏 Note that this diagnostic file will contain code from the package you were attempting to patch. `); } process.exit(1); return; } // maybe delete existing patchFs_1.getPatchFiles(patchDir).forEach((filename) => { const deets = PackageDetails_1.getPackageDetailsFromPatchFilename(filename); if (deets && deets.path === packageDetails.path) { fs_extra_1.unlinkSync(path_1.join(patchDir, filename)); } }); const patchFileName = createPatchFileName({ packageDetails, packageVersion, }); const patchPath = path_1.join(patchesDir, patchFileName); if (!fs_extra_1.existsSync(path_1.dirname(patchPath))) { // scoped package fs_extra_1.mkdirSync(path_1.dirname(patchPath)); } fs_extra_1.writeFileSync(patchPath, diffResult.stdout); console.log(`${chalk_1.default.green("✔")} Created file ${path_1.join(patchDir, patchFileName)}\n`); if (createIssue) { createIssue_1.openIssueCreationLink({ packageDetails, patchFileContents: diffResult.stdout.toString(), packageVersion, }); } else { createIssue_1.maybePrintIssueCreationPrompt(packageDetails, packageManager); } } catch (e) { console.error(e); throw e; } finally { tmpRepo.removeCallback(); } } exports.makePatch = makePatch; function createPatchFileName({ packageDetails, packageVersion, }) { const packageNames = packageDetails.packageNames .map((name) => name.replace(/\//g, "+")) .join("++"); return `${packageNames}+${packageVersion}.patch`; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"makePatch.js","sourceRoot":"","sources":["../src/makePatch.ts"],"names":[],"mappings":"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","sourcesContent":["import chalk from \"chalk\"\nimport { join, dirname, resolve } from \"./path\"\nimport { spawnSafeSync } from \"./spawnSafe\"\nimport { PackageManager } from \"./detectPackageManager\"\nimport { removeIgnoredFiles } from \"./filterFiles\"\nimport {\n  writeFileSync,\n  existsSync,\n  mkdirSync,\n  unlinkSync,\n  mkdirpSync,\n  realpathSync,\n} from \"fs-extra\"\nimport { sync as rimraf } from \"rimraf\"\nimport { copySync } from \"fs-extra\"\nimport { dirSync } from \"tmp\"\nimport { getPatchFiles } from \"./patchFs\"\nimport {\n  getPatchDetailsFromCliString,\n  getPackageDetailsFromPatchFilename,\n  PackageDetails,\n} from \"./PackageDetails\"\nimport { resolveRelativeFileDependencies } from \"./resolveRelativeFileDependencies\"\nimport { getPackageResolution } from \"./getPackageResolution\"\nimport { parsePatchFile } from \"./patch/parse\"\nimport { gzipSync } from \"zlib\"\nimport { getPackageVersion } from \"./getPackageVersion\"\nimport {\n  maybePrintIssueCreationPrompt,\n  openIssueCreationLink,\n} from \"./createIssue\"\n\nfunction printNoPackageFoundError(\n  packageName: string,\n  packageJsonPath: string,\n) {\n  console.error(\n    `No such package ${packageName}\n\n  File not found: ${packageJsonPath}`,\n  )\n}\n\nexport function makePatch({\n  packagePathSpecifier,\n  appPath,\n  packageManager,\n  includePaths,\n  excludePaths,\n  patchDir,\n  createIssue,\n}: {\n  packagePathSpecifier: string\n  appPath: string\n  packageManager: PackageManager\n  includePaths: RegExp\n  excludePaths: RegExp\n  patchDir: string\n  createIssue: boolean\n}) {\n  const packageDetails = getPatchDetailsFromCliString(packagePathSpecifier)\n\n  if (!packageDetails) {\n    console.error(\"No such package\", packagePathSpecifier)\n    return\n  }\n  const appPackageJson = require(join(appPath, \"package.json\"))\n  const packagePath = join(appPath, packageDetails.path)\n  const packageJsonPath = join(packagePath, \"package.json\")\n\n  if (!existsSync(packageJsonPath)) {\n    printNoPackageFoundError(packagePathSpecifier, packageJsonPath)\n    process.exit(1)\n  }\n\n  const tmpRepo = dirSync({ unsafeCleanup: true })\n  const tmpRepoPackagePath = join(tmpRepo.name, packageDetails.path)\n  const tmpRepoNpmRoot = tmpRepoPackagePath.slice(\n    0,\n    -`/node_modules/${packageDetails.name}`.length,\n  )\n\n  const tmpRepoPackageJsonPath = join(tmpRepoNpmRoot, \"package.json\")\n\n  try {\n    const patchesDir = resolve(join(appPath, patchDir))\n\n    console.info(chalk.grey(\"•\"), \"Creating temporary folder\")\n\n    // make a blank package.json\n    mkdirpSync(tmpRepoNpmRoot)\n    writeFileSync(\n      tmpRepoPackageJsonPath,\n      JSON.stringify({\n        dependencies: {\n          [packageDetails.name]: getPackageResolution({\n            packageDetails,\n            packageManager,\n            appPath,\n          }),\n        },\n        resolutions: resolveRelativeFileDependencies(\n          appPath,\n          appPackageJson.resolutions || {},\n        ),\n      }),\n    )\n\n    const packageVersion = getPackageVersion(\n      join(resolve(packageDetails.path), \"package.json\"),\n    )\n\n    // copy .npmrc/.yarnrc in case packages are hosted in private registry\n    // tslint:disable-next-line:align\n    ;[\".npmrc\", \".yarnrc\"].forEach((rcFile) => {\n      const rcPath = join(appPath, rcFile)\n      if (existsSync(rcPath)) {\n        copySync(rcPath, join(tmpRepo.name, rcFile))\n      }\n    })\n\n    if (packageManager === \"yarn\") {\n      console.info(\n        chalk.grey(\"•\"),\n        `Installing ${packageDetails.name}@${packageVersion} with yarn`,\n      )\n      try {\n        // try first without ignoring scripts in case they are required\n        // this works in 99.99% of cases\n        spawnSafeSync(`yarn`, [\"install\", \"--ignore-engines\"], {\n          cwd: tmpRepoNpmRoot,\n          logStdErrOnError: false,\n        })\n      } catch (e) {\n        // try again while ignoring scripts in case the script depends on\n        // an implicit context which we havn't reproduced\n        spawnSafeSync(\n          `yarn`,\n          [\"install\", \"--ignore-engines\", \"--ignore-scripts\"],\n          {\n            cwd: tmpRepoNpmRoot,\n          },\n        )\n      }\n    } else {\n      console.info(\n        chalk.grey(\"•\"),\n        `Installing ${packageDetails.name}@${packageVersion} with npm`,\n      )\n      try {\n        // try first without ignoring scripts in case they are required\n        // this works in 99.99% of cases\n        spawnSafeSync(`npm`, [\"i\", \"--force\"], {\n          cwd: tmpRepoNpmRoot,\n          logStdErrOnError: false,\n          stdio: \"ignore\",\n        })\n      } catch (e) {\n        // try again while ignoring scripts in case the script depends on\n        // an implicit context which we havn't reproduced\n        spawnSafeSync(`npm`, [\"i\", \"--ignore-scripts\", \"--force\"], {\n          cwd: tmpRepoNpmRoot,\n          stdio: \"ignore\",\n        })\n      }\n    }\n\n    const git = (...args: string[]) =>\n      spawnSafeSync(\"git\", args, {\n        cwd: tmpRepo.name,\n        env: { ...process.env, HOME: tmpRepo.name },\n        maxBuffer: 1024 * 1024 * 100,\n      })\n\n    // remove nested node_modules just to be safe\n    rimraf(join(tmpRepoPackagePath, \"node_modules\"))\n    // remove .git just to be safe\n    rimraf(join(tmpRepoPackagePath, \".git\"))\n\n    // commit the package\n    console.info(chalk.grey(\"•\"), \"Diffing your files with clean files\")\n    writeFileSync(join(tmpRepo.name, \".gitignore\"), \"!/node_modules\\n\\n\")\n    git(\"init\")\n    git(\"config\", \"--local\", \"user.name\", \"patch-package\")\n    git(\"config\", \"--local\", \"user.email\", \"patch@pack.age\")\n\n    // remove ignored files first\n    removeIgnoredFiles(tmpRepoPackagePath, includePaths, excludePaths)\n\n    git(\"add\", \"-f\", packageDetails.path)\n    git(\"commit\", \"--allow-empty\", \"-m\", \"init\")\n\n    // replace package with user's version\n    rimraf(tmpRepoPackagePath)\n\n    // pnpm installs packages as symlinks, copySync would copy only the symlink\n    copySync(realpathSync(packagePath), tmpRepoPackagePath)\n\n    // remove nested node_modules just to be safe\n    rimraf(join(tmpRepoPackagePath, \"node_modules\"))\n    // remove .git just to be safe\n    rimraf(join(tmpRepoPackagePath, \".git\"))\n\n    // also remove ignored files like before\n    removeIgnoredFiles(tmpRepoPackagePath, includePaths, excludePaths)\n\n    // stage all files\n    git(\"add\", \"-f\", packageDetails.path)\n\n    // get diff of changes\n    const diffResult = git(\n      \"diff\",\n      \"--cached\",\n      \"--no-color\",\n      \"--ignore-space-at-eol\",\n      \"--no-ext-diff\",\n    )\n\n    if (diffResult.stdout.length === 0) {\n      console.warn(\n        `⁉️  Not creating patch file for package '${packagePathSpecifier}'`,\n      )\n      console.warn(`⁉️  There don't appear to be any changes.`)\n      process.exit(1)\n      return\n    }\n\n    try {\n      parsePatchFile(diffResult.stdout.toString())\n    } catch (e) {\n      if (\n        (e as Error).message.includes(\"Unexpected file mode string: 120000\")\n      ) {\n        console.error(`\n⛔️ ${chalk.red.bold(\"ERROR\")}\n\n  Your changes involve creating symlinks. patch-package does not yet support\n  symlinks.\n  \n  ️Please use ${chalk.bold(\"--include\")} and/or ${chalk.bold(\n          \"--exclude\",\n        )} to narrow the scope of your patch if\n  this was unintentional.\n`)\n      } else {\n        const outPath = \"./patch-package-error.json.gz\"\n        writeFileSync(\n          outPath,\n          gzipSync(\n            JSON.stringify({\n              error: { message: e.message, stack: e.stack },\n              patch: diffResult.stdout.toString(),\n            }),\n          ),\n        )\n        console.error(`\n⛔️ ${chalk.red.bold(\"ERROR\")}\n        \n  patch-package was unable to read the patch-file made by git. This should not\n  happen.\n  \n  A diagnostic file was written to\n  \n    ${outPath}\n  \n  Please attach it to a github issue\n  \n    https://github.com/ds300/patch-package/issues/new?title=New+patch+parse+failed&body=Please+attach+the+diagnostic+file+by+dragging+it+into+here+🙏\n  \n  Note that this diagnostic file will contain code from the package you were\n  attempting to patch.\n\n`)\n      }\n      process.exit(1)\n      return\n    }\n\n    // maybe delete existing\n    getPatchFiles(patchDir).forEach((filename) => {\n      const deets = getPackageDetailsFromPatchFilename(filename)\n      if (deets && deets.path === packageDetails.path) {\n        unlinkSync(join(patchDir, filename))\n      }\n    })\n\n    const patchFileName = createPatchFileName({\n      packageDetails,\n      packageVersion,\n    })\n\n    const patchPath = join(patchesDir, patchFileName)\n    if (!existsSync(dirname(patchPath))) {\n      // scoped package\n      mkdirSync(dirname(patchPath))\n    }\n    writeFileSync(patchPath, diffResult.stdout)\n    console.log(\n      `${chalk.green(\"✔\")} Created file ${join(patchDir, patchFileName)}\\n`,\n    )\n    if (createIssue) {\n      openIssueCreationLink({\n        packageDetails,\n        patchFileContents: diffResult.stdout.toString(),\n        packageVersion,\n      })\n    } else {\n      maybePrintIssueCreationPrompt(packageDetails, packageManager)\n    }\n  } catch (e) {\n    console.error(e)\n    throw e\n  } finally {\n    tmpRepo.removeCallback()\n  }\n}\n\nfunction createPatchFileName({\n  packageDetails,\n  packageVersion,\n}: {\n  packageDetails: PackageDetails\n  packageVersion: string\n}) {\n  const packageNames = packageDetails.packageNames\n    .map((name) => name.replace(/\\//g, \"+\"))\n    .join(\"++\")\n\n  return `${packageNames}+${packageVersion}.patch`\n}\n"]}