UNPKG

projen

Version:

CDK for software projects

170 lines 23.5 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.unifiedDiff = unifiedDiff; const chalk_1 = require("chalk"); /** * Compute a unified diff between two strings with context lines, hunk headers, * and optional colorized output with character-level inline highlighting. * * @param oldContent the previous content. * @param newContent the new content. * @param colorize whether to colorize the output. * @param contextSize number of unchanged context lines around each change. * @returns the diff as an array of lines, or `undefined` if there are no changes. */ function unifiedDiff(oldContent, newContent, colorize, contextSize) { const oldLines = oldContent.split("\n"); const newLines = newContent.split("\n"); const edits = computeEdits(oldLines, newLines); // Find indices of changed edits const changeIndices = []; for (let i = 0; i < edits.length; i++) { if (edits[i].type !== "equal") { changeIndices.push(i); } } if (changeIndices.length === 0) { return undefined; } // Group changes into hunks with context, merging overlapping ones const hunks = []; for (const ci of changeIndices) { const start = Math.max(0, ci - contextSize); const end = Math.min(edits.length - 1, ci + contextSize); if (hunks.length > 0 && start <= hunks[hunks.length - 1].end + 1) { hunks[hunks.length - 1].end = end; } else { hunks.push({ start, end }); } } // Format hunks const output = []; for (const hunk of hunks) { // Compute line numbers for the header let oldStart = 1; let newStart = 1; for (let i = 0; i < hunk.start; i++) { const e = edits[i]; if (e.type === "equal" || e.type === "delete" || e.type === "replace") { oldStart++; } if (e.type === "equal" || e.type === "insert" || e.type === "replace") { newStart++; } } let oldCount = 0; let newCount = 0; for (let i = hunk.start; i <= hunk.end; i++) { const e = edits[i]; if (e.type === "equal" || e.type === "delete" || e.type === "replace") { oldCount++; } if (e.type === "equal" || e.type === "insert" || e.type === "replace") { newCount++; } } const header = `@@ -${oldStart},${oldCount} +${newStart},${newCount} @@`; output.push(colorize ? (0, chalk_1.cyan)(header) : header); for (let i = hunk.start; i <= hunk.end; i++) { const e = edits[i]; switch (e.type) { case "equal": output.push(` ${oldLines[e.oldIdx]}`); break; case "delete": output.push(colorize ? `${chalk_1.bold.red("-")} ${(0, chalk_1.red)(oldLines[e.oldIdx])}` : `- ${oldLines[e.oldIdx]}`); break; case "insert": output.push(colorize ? `${chalk_1.bold.green("+")} ${(0, chalk_1.green)(newLines[e.newIdx])}` : `+ ${newLines[e.newIdx]}`); break; case "replace": output.push(...inlineDiff(oldLines[e.oldIdx], newLines[e.newIdx], colorize)); break; } } } return output; } /** * Compute edit operations between two arrays of lines using LCS. */ function computeEdits(oldLines, newLines) { const oldLen = oldLines.length; const newLen = newLines.length; // Build LCS table const lcs = Array.from({ length: oldLen + 1 }, () => new Array(newLen + 1).fill(0)); for (let i = oldLen - 1; i >= 0; i--) { for (let j = newLen - 1; j >= 0; j--) { if (oldLines[i] === newLines[j]) { lcs[i][j] = lcs[i + 1][j + 1] + 1; } else { lcs[i][j] = Math.max(lcs[i + 1][j], lcs[i][j + 1]); } } } // Walk the LCS table to produce edit operations const edits = []; let i = 0; let j = 0; while (i < oldLen || j < newLen) { if (i < oldLen && j < newLen && oldLines[i] === newLines[j]) { edits.push({ type: "equal", oldIdx: i, newIdx: j }); i++; j++; } else if (i < oldLen && j < newLen && lcs[i + 1][j] === lcs[i][j + 1]) { // Both sides advance equally — treat as a replacement edits.push({ type: "replace", oldIdx: i, newIdx: j }); i++; j++; } else if (j < newLen && (i >= oldLen || lcs[i][j + 1] >= lcs[i + 1][j])) { edits.push({ type: "insert", newIdx: j }); j++; } else { edits.push({ type: "delete", oldIdx: i }); i++; } } return edits; } /** * Produces a remove/add line pair with character-level highlighting * of the parts that changed. */ function inlineDiff(oldLine, newLine, colorize) { // Find common prefix and suffix let prefix = 0; while (prefix < oldLine.length && prefix < newLine.length && oldLine[prefix] === newLine[prefix]) { prefix++; } let oldSuffix = oldLine.length; let newSuffix = newLine.length; while (oldSuffix > prefix && newSuffix > prefix && oldLine[oldSuffix - 1] === newLine[newSuffix - 1]) { oldSuffix--; newSuffix--; } if (colorize) { const before = oldLine.substring(0, prefix); const after = oldLine.substring(oldSuffix); const oldChanged = oldLine.substring(prefix, oldSuffix); const newChanged = newLine.substring(prefix, newSuffix); return [ `${chalk_1.bold.red("-")} ${(0, chalk_1.red)(before)}${chalk_1.red.bold.underline(oldChanged)}${(0, chalk_1.red)(after)}`, `${chalk_1.bold.green("+")} ${(0, chalk_1.green)(before)}${chalk_1.green.bold.underline(newChanged)}${(0, chalk_1.green)(after)}`, ]; } return [`- ${oldLine}`, `+ ${newLine}`]; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"diff.js","sourceRoot":"","sources":["../../src/util/diff.ts"],"names":[],"mappings":"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","sourcesContent":["import { bold, cyan, green, red } from \"chalk\";\n\n/**\n * Compute a unified diff between two strings with context lines, hunk headers,\n * and optional colorized output with character-level inline highlighting.\n *\n * @param oldContent the previous content.\n * @param newContent the new content.\n * @param colorize whether to colorize the output.\n * @param contextSize number of unchanged context lines around each change.\n * @returns the diff as an array of lines, or `undefined` if there are no changes.\n */\nexport function unifiedDiff(\n  oldContent: string,\n  newContent: string,\n  colorize: boolean,\n  contextSize: number,\n): string[] | undefined {\n  const oldLines = oldContent.split(\"\\n\");\n  const newLines = newContent.split(\"\\n\");\n  const edits = computeEdits(oldLines, newLines);\n\n  // Find indices of changed edits\n  const changeIndices: number[] = [];\n  for (let i = 0; i < edits.length; i++) {\n    if (edits[i].type !== \"equal\") {\n      changeIndices.push(i);\n    }\n  }\n  if (changeIndices.length === 0) {\n    return undefined;\n  }\n\n  // Group changes into hunks with context, merging overlapping ones\n  const hunks: Array<{ start: number; end: number }> = [];\n  for (const ci of changeIndices) {\n    const start = Math.max(0, ci - contextSize);\n    const end = Math.min(edits.length - 1, ci + contextSize);\n    if (hunks.length > 0 && start <= hunks[hunks.length - 1].end + 1) {\n      hunks[hunks.length - 1].end = end;\n    } else {\n      hunks.push({ start, end });\n    }\n  }\n\n  // Format hunks\n  const output: string[] = [];\n  for (const hunk of hunks) {\n    // Compute line numbers for the header\n    let oldStart = 1;\n    let newStart = 1;\n    for (let i = 0; i < hunk.start; i++) {\n      const e = edits[i];\n      if (e.type === \"equal\" || e.type === \"delete\" || e.type === \"replace\") {\n        oldStart++;\n      }\n      if (e.type === \"equal\" || e.type === \"insert\" || e.type === \"replace\") {\n        newStart++;\n      }\n    }\n\n    let oldCount = 0;\n    let newCount = 0;\n    for (let i = hunk.start; i <= hunk.end; i++) {\n      const e = edits[i];\n      if (e.type === \"equal\" || e.type === \"delete\" || e.type === \"replace\") {\n        oldCount++;\n      }\n      if (e.type === \"equal\" || e.type === \"insert\" || e.type === \"replace\") {\n        newCount++;\n      }\n    }\n\n    const header = `@@ -${oldStart},${oldCount} +${newStart},${newCount} @@`;\n    output.push(colorize ? cyan(header) : header);\n\n    for (let i = hunk.start; i <= hunk.end; i++) {\n      const e = edits[i];\n      switch (e.type) {\n        case \"equal\":\n          output.push(`  ${oldLines[e.oldIdx]}`);\n          break;\n        case \"delete\":\n          output.push(\n            colorize\n              ? `${bold.red(\"-\")} ${red(oldLines[e.oldIdx])}`\n              : `- ${oldLines[e.oldIdx]}`,\n          );\n          break;\n        case \"insert\":\n          output.push(\n            colorize\n              ? `${bold.green(\"+\")} ${green(newLines[e.newIdx])}`\n              : `+ ${newLines[e.newIdx]}`,\n          );\n          break;\n        case \"replace\":\n          output.push(\n            ...inlineDiff(oldLines[e.oldIdx], newLines[e.newIdx], colorize),\n          );\n          break;\n      }\n    }\n  }\n\n  return output;\n}\n\ntype EditOp =\n  | { type: \"equal\"; oldIdx: number; newIdx: number }\n  | { type: \"delete\"; oldIdx: number }\n  | { type: \"insert\"; newIdx: number }\n  | { type: \"replace\"; oldIdx: number; newIdx: number };\n\n/**\n * Compute edit operations between two arrays of lines using LCS.\n */\nfunction computeEdits(oldLines: string[], newLines: string[]): EditOp[] {\n  const oldLen = oldLines.length;\n  const newLen = newLines.length;\n\n  // Build LCS table\n  const lcs: number[][] = Array.from({ length: oldLen + 1 }, () =>\n    new Array(newLen + 1).fill(0),\n  );\n  for (let i = oldLen - 1; i >= 0; i--) {\n    for (let j = newLen - 1; j >= 0; j--) {\n      if (oldLines[i] === newLines[j]) {\n        lcs[i][j] = lcs[i + 1][j + 1] + 1;\n      } else {\n        lcs[i][j] = Math.max(lcs[i + 1][j], lcs[i][j + 1]);\n      }\n    }\n  }\n\n  // Walk the LCS table to produce edit operations\n  const edits: EditOp[] = [];\n  let i = 0;\n  let j = 0;\n  while (i < oldLen || j < newLen) {\n    if (i < oldLen && j < newLen && oldLines[i] === newLines[j]) {\n      edits.push({ type: \"equal\", oldIdx: i, newIdx: j });\n      i++;\n      j++;\n    } else if (i < oldLen && j < newLen && lcs[i + 1][j] === lcs[i][j + 1]) {\n      // Both sides advance equally — treat as a replacement\n      edits.push({ type: \"replace\", oldIdx: i, newIdx: j });\n      i++;\n      j++;\n    } else if (j < newLen && (i >= oldLen || lcs[i][j + 1] >= lcs[i + 1][j])) {\n      edits.push({ type: \"insert\", newIdx: j });\n      j++;\n    } else {\n      edits.push({ type: \"delete\", oldIdx: i });\n      i++;\n    }\n  }\n\n  return edits;\n}\n\n/**\n * Produces a remove/add line pair with character-level highlighting\n * of the parts that changed.\n */\nfunction inlineDiff(\n  oldLine: string,\n  newLine: string,\n  colorize: boolean,\n): string[] {\n  // Find common prefix and suffix\n  let prefix = 0;\n  while (\n    prefix < oldLine.length &&\n    prefix < newLine.length &&\n    oldLine[prefix] === newLine[prefix]\n  ) {\n    prefix++;\n  }\n  let oldSuffix = oldLine.length;\n  let newSuffix = newLine.length;\n  while (\n    oldSuffix > prefix &&\n    newSuffix > prefix &&\n    oldLine[oldSuffix - 1] === newLine[newSuffix - 1]\n  ) {\n    oldSuffix--;\n    newSuffix--;\n  }\n\n  if (colorize) {\n    const before = oldLine.substring(0, prefix);\n    const after = oldLine.substring(oldSuffix);\n    const oldChanged = oldLine.substring(prefix, oldSuffix);\n    const newChanged = newLine.substring(prefix, newSuffix);\n    return [\n      `${bold.red(\"-\")} ${red(before)}${red.bold.underline(oldChanged)}${red(after)}`,\n      `${bold.green(\"+\")} ${green(before)}${green.bold.underline(newChanged)}${green(after)}`,\n    ];\n  }\n\n  return [`- ${oldLine}`, `+ ${newLine}`];\n}\n"]}