@unts/patch-package
Version:
Fix broken node modules with no fuss
116 lines (109 loc) • 14.3 kB
JavaScript
;
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.openIssueCreationLink = exports.maybePrintIssueCreationPrompt = exports.shouldRecommendIssue = exports.getPackageVCSDetails = void 0;
const open_1 = __importDefault(require("open"));
const picocolors_1 = __importDefault(require("picocolors"));
const querystring_1 = require("querystring");
const path_1 = require("./path");
const repoSpecifier = /^([\w.-]+)\/([\w.-]+)$/;
const githubURL = /github.com(:|\/)([\w.-]+\/[\w.-]+?)(.git|\/.*)?$/;
function parseRepoString(repository) {
if (repository.startsWith("github:")) {
repository = repository.replace(/^github:/, "");
}
const urlMatch = repository.match(githubURL);
if (urlMatch) {
repository = urlMatch[2];
}
const specMatch = repository.match(repoSpecifier);
if (!specMatch) {
return null;
}
const [, org, repo] = specMatch;
return { org, repo, provider: "GitHub" };
}
function getPackageVCSDetails(packageDetails) {
const repository = require((0, path_1.resolve)((0, path_1.join)(packageDetails.path, "package.json")))
.repository;
if (!repository) {
return null;
}
if (typeof repository === "string") {
return parseRepoString(repository);
}
else if (typeof repository === "object" &&
typeof repository.url === "string") {
return parseRepoString(repository.url);
}
}
exports.getPackageVCSDetails = getPackageVCSDetails;
function createIssueUrl({ vcs, packageDetails, packageVersion, diff, }) {
return `https://github.com/${vcs === null || vcs === void 0 ? void 0 : vcs.org}/${vcs === null || vcs === void 0 ? void 0 : vcs.repo}/issues/new?${(0, querystring_1.stringify)({
title: "",
body: `Hi! 👋
Firstly, thanks for your work on this project! 🙂
Today I used [patch-package](https://github.com/ds300/patch-package) to patch \`${packageDetails.name}@${packageVersion}\` for the project I'm working on.
<!-- 🔺️🔺️🔺️ PLEASE REPLACE THIS BLOCK with a description of your problem, and any other relevant context 🔺️🔺️🔺️ -->
Here is the diff that solved my problem:
\`\`\`diff
${diff}
\`\`\`
<em>This issue body was [partially generated by patch-package](https://github.com/ds300/patch-package/issues/296).</em>
`,
})}`;
}
function shouldRecommendIssue(vcsDetails) {
if (!vcsDetails) {
return true;
}
const { repo, org } = vcsDetails;
if (repo === "DefinitelyTyped" && org === "DefinitelyTyped") {
return false;
}
return true;
}
exports.shouldRecommendIssue = shouldRecommendIssue;
function maybePrintIssueCreationPrompt(vcs, packageDetails, packageManager) {
if (vcs) {
console.log(`💡 ${picocolors_1.default.bold(packageDetails.name)} is on ${vcs.provider}! To draft an issue based on your patch run
${packageManager === "yarn" ? "yarn" : "npx"} patch-package ${packageDetails.pathSpecifier} --create-issue
`);
}
}
exports.maybePrintIssueCreationPrompt = maybePrintIssueCreationPrompt;
function openIssueCreationLink({ packageDetails, patchFileContents, packageVersion, patchPath, }) {
const vcs = getPackageVCSDetails(packageDetails);
if (!vcs) {
console.log(`Error: Couldn't find VCS details for ${packageDetails.pathSpecifier}`);
process.exit(1);
}
// trim off trailing newline since we add an extra one in the markdown block
if (patchFileContents.endsWith("\n")) {
patchFileContents = patchFileContents.slice(0, -1);
}
let issueUrl = createIssueUrl({
vcs,
packageDetails,
packageVersion,
diff: patchFileContents,
});
const urlExceedsLimit = patchFileContents.length > 1950;
if (urlExceedsLimit) {
const diffMessage = `<!-- 🔺️🔺️🔺️ PLEASE REPLACE THIS BLOCK with the diff contents of ${patchPath
.split("/")
.pop()}. 🔺️🔺️🔺️ -->`;
console.log(`📋 Copy the contents in [ ${patchPath} ] and paste it in the new issue's diff section.`);
issueUrl = createIssueUrl({
vcs,
packageDetails,
packageVersion,
diff: diffMessage,
});
}
(0, open_1.default)(issueUrl);
}
exports.openIssueCreationLink = openIssueCreationLink;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"createIssue.js","sourceRoot":"","sources":["../src/createIssue.ts"],"names":[],"mappings":"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","sourcesContent":["import open from \"open\"\nimport colors from \"picocolors\"\nimport { stringify } from \"querystring\"\nimport { PackageManager } from \"./detectPackageManager\"\nimport { PackageDetails } from \"./PackageDetails\"\nimport { join, resolve } from \"./path\"\n\nconst repoSpecifier = /^([\\w.-]+)\\/([\\w.-]+)$/\nconst githubURL = /github.com(:|\\/)([\\w.-]+\\/[\\w.-]+?)(.git|\\/.*)?$/\n\ntype VCS =\n  | {\n      repo: string\n      org: string\n      provider: \"GitHub\"\n    }\n  | null\n  | undefined\n\nfunction parseRepoString(repository: string): VCS {\n  if (repository.startsWith(\"github:\")) {\n    repository = repository.replace(/^github:/, \"\")\n  }\n  const urlMatch = repository.match(githubURL)\n  if (urlMatch) {\n    repository = urlMatch[2]\n  }\n\n  const specMatch = repository.match(repoSpecifier)\n\n  if (!specMatch) {\n    return null\n  }\n  const [, org, repo] = specMatch\n\n  return { org, repo, provider: \"GitHub\" }\n}\n\nexport function getPackageVCSDetails(packageDetails: PackageDetails): VCS {\n  const repository = require(resolve(join(packageDetails.path, \"package.json\")))\n    .repository as undefined | string | { url: string }\n\n  if (!repository) {\n    return null\n  }\n  if (typeof repository === \"string\") {\n    return parseRepoString(repository)\n  } else if (\n    typeof repository === \"object\" &&\n    typeof repository.url === \"string\"\n  ) {\n    return parseRepoString(repository.url)\n  }\n}\n\nfunction createIssueUrl({\n  vcs,\n  packageDetails,\n  packageVersion,\n  diff,\n}: {\n  vcs: VCS\n  packageDetails: PackageDetails\n  packageVersion: string\n  diff: string\n}): string {\n  return `https://github.com/${vcs?.org}/${vcs?.repo}/issues/new?${stringify({\n    title: \"\",\n    body: `Hi! 👋 \n      \nFirstly, thanks for your work on this project! 🙂\n\nToday I used [patch-package](https://github.com/ds300/patch-package) to patch \\`${packageDetails.name}@${packageVersion}\\` for the project I'm working on.\n\n<!-- 🔺️🔺️🔺️ PLEASE REPLACE THIS BLOCK with a description of your problem, and any other relevant context 🔺️🔺️🔺️ -->\n\nHere is the diff that solved my problem:\n\n\\`\\`\\`diff\n${diff}\n\\`\\`\\`\n\n<em>This issue body was [partially generated by patch-package](https://github.com/ds300/patch-package/issues/296).</em>\n`,\n  })}`\n}\n\nexport function shouldRecommendIssue(\n  vcsDetails: ReturnType<typeof getPackageVCSDetails>,\n) {\n  if (!vcsDetails) {\n    return true\n  }\n\n  const { repo, org } = vcsDetails\n  if (repo === \"DefinitelyTyped\" && org === \"DefinitelyTyped\") {\n    return false\n  }\n  return true\n}\n\nexport function maybePrintIssueCreationPrompt(\n  vcs: ReturnType<typeof getPackageVCSDetails>,\n  packageDetails: PackageDetails,\n  packageManager: PackageManager,\n) {\n  if (vcs) {\n    console.log(`💡 ${colors.bold(packageDetails.name)} is on ${\n      vcs.provider\n    }! To draft an issue based on your patch run\n\n    ${packageManager === \"yarn\" ? \"yarn\" : \"npx\"} patch-package ${\n      packageDetails.pathSpecifier\n    } --create-issue\n`)\n  }\n}\n\nexport function openIssueCreationLink({\n  packageDetails,\n  patchFileContents,\n  packageVersion,\n  patchPath,\n}: {\n  packageDetails: PackageDetails\n  patchFileContents: string\n  packageVersion: string\n  patchPath: string\n}) {\n  const vcs = getPackageVCSDetails(packageDetails)\n\n  if (!vcs) {\n    console.log(\n      `Error: Couldn't find VCS details for ${packageDetails.pathSpecifier}`,\n    )\n    process.exit(1)\n  }\n\n  // trim off trailing newline since we add an extra one in the markdown block\n  if (patchFileContents.endsWith(\"\\n\")) {\n    patchFileContents = patchFileContents.slice(0, -1)\n  }\n\n  let issueUrl = createIssueUrl({\n    vcs,\n    packageDetails,\n    packageVersion,\n    diff: patchFileContents,\n  })\n\n  const urlExceedsLimit = patchFileContents.length > 1950\n\n  if (urlExceedsLimit) {\n    const diffMessage = `<!-- 🔺️🔺️🔺️ PLEASE REPLACE THIS BLOCK with the diff contents of ${patchPath\n      .split(\"/\")\n      .pop()}. 🔺️🔺️🔺️ -->`\n    console.log(\n      `📋 Copy the contents in [ ${patchPath} ] and paste it in the new issue's diff section.`,\n    )\n    issueUrl = createIssueUrl({\n      vcs,\n      packageDetails,\n      packageVersion,\n      diff: diffMessage,\n    })\n  }\n  open(issueUrl)\n}\n"]}