| Total Complexity | 87 |
| Complexity/F | 4.35 |
| Lines of Code | 366 |
| Function Count | 20 |
| Duplicated Lines | 6 |
| Ratio | 1.64 % |
| Changes | 0 | ||
Duplicate code is one of the most pungent code smells. A rule that is often used is to re-structure code once it is duplicated in three or more places.
Common duplication problems, and corresponding solutions are:
Complex classes like node_modules/diff/lib/patch/merge.js often do a lot of different things. To break such a class down, we need to identify a cohesive component within that class. A common approach to find such a component is to look for fields/methods that share the same prefixes, or suffixes.
Once you have determined the fields that belong together, you can apply the Extract Class refactoring. If the component makes sense as a sub-class, Extract Subclass is also a candidate, and is often faster.
| 1 | /*istanbul ignore start*/'use strict'; |
||
| 2 | |||
| 3 | exports.__esModule = true; |
||
| 4 | exports. /*istanbul ignore end*/calcLineCount = calcLineCount; |
||
| 5 | /*istanbul ignore start*/exports. /*istanbul ignore end*/merge = merge; |
||
| 6 | |||
| 7 | var /*istanbul ignore start*/_create = require('./create') /*istanbul ignore end*/; |
||
| 8 | |||
| 9 | var /*istanbul ignore start*/_parse = require('./parse') /*istanbul ignore end*/; |
||
| 10 | |||
| 11 | var /*istanbul ignore start*/_array = require('../util/array') /*istanbul ignore end*/; |
||
| 12 | |||
| 13 | /*istanbul ignore start*/function _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } } |
||
|
|
|||
| 14 | |||
| 15 | View Code Duplication | /*istanbul ignore end*/function calcLineCount(hunk) { |
|
| 16 | var conflicted = false; |
||
| 17 | |||
| 18 | hunk.oldLines = 0; |
||
| 19 | hunk.newLines = 0; |
||
| 20 | |||
| 21 | hunk.lines.forEach(function (line) { |
||
| 22 | if (typeof line !== 'string') { |
||
| 23 | conflicted = true; |
||
| 24 | return; |
||
| 25 | } |
||
| 26 | |||
| 27 | if (line[0] === '+' || line[0] === ' ') { |
||
| 28 | hunk.newLines++; |
||
| 29 | } |
||
| 30 | if (line[0] === '-' || line[0] === ' ') { |
||
| 31 | hunk.oldLines++; |
||
| 32 | } |
||
| 33 | }); |
||
| 34 | |||
| 35 | if (conflicted) { |
||
| 36 | delete hunk.oldLines; |
||
| 37 | delete hunk.newLines; |
||
| 38 | } |
||
| 39 | } |
||
| 40 | |||
| 41 | View Code Duplication | function merge(mine, theirs, base) { |
|
| 42 | mine = loadPatch(mine, base); |
||
| 43 | theirs = loadPatch(theirs, base); |
||
| 44 | |||
| 45 | var ret = {}; |
||
| 46 | |||
| 47 | // For index we just let it pass through as it doesn't have any necessary meaning. |
||
| 48 | // Leaving sanity checks on this to the API consumer that may know more about the |
||
| 49 | // meaning in their own context. |
||
| 50 | if (mine.index || theirs.index) { |
||
| 51 | ret.index = mine.index || theirs.index; |
||
| 52 | } |
||
| 53 | |||
| 54 | if (mine.newFileName || theirs.newFileName) { |
||
| 55 | if (!fileNameChanged(mine)) { |
||
| 56 | // No header or no change in ours, use theirs (and ours if theirs does not exist) |
||
| 57 | ret.oldFileName = theirs.oldFileName || mine.oldFileName; |
||
| 58 | ret.newFileName = theirs.newFileName || mine.newFileName; |
||
| 59 | ret.oldHeader = theirs.oldHeader || mine.oldHeader; |
||
| 60 | ret.newHeader = theirs.newHeader || mine.newHeader; |
||
| 61 | } else if (!fileNameChanged(theirs)) { |
||
| 62 | // No header or no change in theirs, use ours |
||
| 63 | ret.oldFileName = mine.oldFileName; |
||
| 64 | ret.newFileName = mine.newFileName; |
||
| 65 | ret.oldHeader = mine.oldHeader; |
||
| 66 | ret.newHeader = mine.newHeader; |
||
| 67 | } else { |
||
| 68 | // Both changed... figure it out |
||
| 69 | ret.oldFileName = selectField(ret, mine.oldFileName, theirs.oldFileName); |
||
| 70 | ret.newFileName = selectField(ret, mine.newFileName, theirs.newFileName); |
||
| 71 | ret.oldHeader = selectField(ret, mine.oldHeader, theirs.oldHeader); |
||
| 72 | ret.newHeader = selectField(ret, mine.newHeader, theirs.newHeader); |
||
| 73 | } |
||
| 74 | } |
||
| 75 | |||
| 76 | ret.hunks = []; |
||
| 77 | |||
| 78 | var mineIndex = 0, |
||
| 79 | theirsIndex = 0, |
||
| 80 | mineOffset = 0, |
||
| 81 | theirsOffset = 0; |
||
| 82 | |||
| 83 | while (mineIndex < mine.hunks.length || theirsIndex < theirs.hunks.length) { |
||
| 84 | var mineCurrent = mine.hunks[mineIndex] || { oldStart: Infinity }, |
||
| 85 | theirsCurrent = theirs.hunks[theirsIndex] || { oldStart: Infinity }; |
||
| 86 | |||
| 87 | if (hunkBefore(mineCurrent, theirsCurrent)) { |
||
| 88 | // This patch does not overlap with any of the others, yay. |
||
| 89 | ret.hunks.push(cloneHunk(mineCurrent, mineOffset)); |
||
| 90 | mineIndex++; |
||
| 91 | theirsOffset += mineCurrent.newLines - mineCurrent.oldLines; |
||
| 92 | } else if (hunkBefore(theirsCurrent, mineCurrent)) { |
||
| 93 | // This patch does not overlap with any of the others, yay. |
||
| 94 | ret.hunks.push(cloneHunk(theirsCurrent, theirsOffset)); |
||
| 95 | theirsIndex++; |
||
| 96 | mineOffset += theirsCurrent.newLines - theirsCurrent.oldLines; |
||
| 97 | } else { |
||
| 98 | // Overlap, merge as best we can |
||
| 99 | var mergedHunk = { |
||
| 100 | oldStart: Math.min(mineCurrent.oldStart, theirsCurrent.oldStart), |
||
| 101 | oldLines: 0, |
||
| 102 | newStart: Math.min(mineCurrent.newStart + mineOffset, theirsCurrent.oldStart + theirsOffset), |
||
| 103 | newLines: 0, |
||
| 104 | lines: [] |
||
| 105 | }; |
||
| 106 | mergeLines(mergedHunk, mineCurrent.oldStart, mineCurrent.lines, theirsCurrent.oldStart, theirsCurrent.lines); |
||
| 107 | theirsIndex++; |
||
| 108 | mineIndex++; |
||
| 109 | |||
| 110 | ret.hunks.push(mergedHunk); |
||
| 111 | } |
||
| 112 | } |
||
| 113 | |||
| 114 | return ret; |
||
| 115 | } |
||
| 116 | |||
| 117 | function loadPatch(param, base) { |
||
| 118 | if (typeof param === 'string') { |
||
| 119 | if (/^@@/m.test(param) || /^Index:/m.test(param)) { |
||
| 120 | return (/*istanbul ignore start*/(0, _parse.parsePatch) /*istanbul ignore end*/(param)[0] |
||
| 121 | ); |
||
| 122 | } |
||
| 123 | |||
| 124 | if (!base) { |
||
| 125 | throw new Error('Must provide a base reference or pass in a patch'); |
||
| 126 | } |
||
| 127 | return (/*istanbul ignore start*/(0, _create.structuredPatch) /*istanbul ignore end*/(undefined, undefined, base, param) |
||
| 128 | ); |
||
| 129 | } |
||
| 130 | |||
| 131 | return param; |
||
| 132 | } |
||
| 133 | |||
| 134 | function fileNameChanged(patch) { |
||
| 135 | return patch.newFileName && patch.newFileName !== patch.oldFileName; |
||
| 136 | } |
||
| 137 | |||
| 138 | function selectField(index, mine, theirs) { |
||
| 139 | if (mine === theirs) { |
||
| 140 | return mine; |
||
| 141 | } else { |
||
| 142 | index.conflict = true; |
||
| 143 | return { mine: mine, theirs: theirs }; |
||
| 144 | } |
||
| 145 | } |
||
| 146 | |||
| 147 | function hunkBefore(test, check) { |
||
| 148 | return test.oldStart < check.oldStart && test.oldStart + test.oldLines < check.oldStart; |
||
| 149 | } |
||
| 150 | |||
| 151 | function cloneHunk(hunk, offset) { |
||
| 152 | return { |
||
| 153 | oldStart: hunk.oldStart, oldLines: hunk.oldLines, |
||
| 154 | newStart: hunk.newStart + offset, newLines: hunk.newLines, |
||
| 155 | lines: hunk.lines |
||
| 156 | }; |
||
| 157 | } |
||
| 158 | |||
| 159 | View Code Duplication | function mergeLines(hunk, mineOffset, mineLines, theirOffset, theirLines) { |
|
| 160 | // This will generally result in a conflicted hunk, but there are cases where the context |
||
| 161 | // is the only overlap where we can successfully merge the content here. |
||
| 162 | var mine = { offset: mineOffset, lines: mineLines, index: 0 }, |
||
| 163 | their = { offset: theirOffset, lines: theirLines, index: 0 }; |
||
| 164 | |||
| 165 | // Handle any leading content |
||
| 166 | insertLeading(hunk, mine, their); |
||
| 167 | insertLeading(hunk, their, mine); |
||
| 168 | |||
| 169 | // Now in the overlap content. Scan through and select the best changes from each. |
||
| 170 | while (mine.index < mine.lines.length && their.index < their.lines.length) { |
||
| 171 | var mineCurrent = mine.lines[mine.index], |
||
| 172 | theirCurrent = their.lines[their.index]; |
||
| 173 | |||
| 174 | if ((mineCurrent[0] === '-' || mineCurrent[0] === '+') && (theirCurrent[0] === '-' || theirCurrent[0] === '+')) { |
||
| 175 | // Both modified ... |
||
| 176 | mutualChange(hunk, mine, their); |
||
| 177 | } else if (mineCurrent[0] === '+' && theirCurrent[0] === ' ') { |
||
| 178 | /*istanbul ignore start*/var _hunk$lines; |
||
| 179 | |||
| 180 | /*istanbul ignore end*/ // Mine inserted |
||
| 181 | /*istanbul ignore start*/(_hunk$lines = /*istanbul ignore end*/hunk.lines).push. /*istanbul ignore start*/apply /*istanbul ignore end*/( /*istanbul ignore start*/_hunk$lines /*istanbul ignore end*/, /*istanbul ignore start*/_toConsumableArray( /*istanbul ignore end*/collectChange(mine))); |
||
| 182 | } else if (theirCurrent[0] === '+' && mineCurrent[0] === ' ') { |
||
| 183 | /*istanbul ignore start*/var _hunk$lines2; |
||
| 184 | |||
| 185 | /*istanbul ignore end*/ // Theirs inserted |
||
| 186 | /*istanbul ignore start*/(_hunk$lines2 = /*istanbul ignore end*/hunk.lines).push. /*istanbul ignore start*/apply /*istanbul ignore end*/( /*istanbul ignore start*/_hunk$lines2 /*istanbul ignore end*/, /*istanbul ignore start*/_toConsumableArray( /*istanbul ignore end*/collectChange(their))); |
||
| 187 | } else if (mineCurrent[0] === '-' && theirCurrent[0] === ' ') { |
||
| 188 | // Mine removed or edited |
||
| 189 | removal(hunk, mine, their); |
||
| 190 | } else if (theirCurrent[0] === '-' && mineCurrent[0] === ' ') { |
||
| 191 | // Their removed or edited |
||
| 192 | removal(hunk, their, mine, true); |
||
| 193 | } else if (mineCurrent === theirCurrent) { |
||
| 194 | // Context identity |
||
| 195 | hunk.lines.push(mineCurrent); |
||
| 196 | mine.index++; |
||
| 197 | their.index++; |
||
| 198 | } else { |
||
| 199 | // Context mismatch |
||
| 200 | conflict(hunk, collectChange(mine), collectChange(their)); |
||
| 201 | } |
||
| 202 | } |
||
| 203 | |||
| 204 | // Now push anything that may be remaining |
||
| 205 | insertTrailing(hunk, mine); |
||
| 206 | insertTrailing(hunk, their); |
||
| 207 | |||
| 208 | calcLineCount(hunk); |
||
| 209 | } |
||
| 210 | |||
| 211 | View Code Duplication | function mutualChange(hunk, mine, their) { |
|
| 212 | var myChanges = collectChange(mine), |
||
| 213 | theirChanges = collectChange(their); |
||
| 214 | |||
| 215 | if (allRemoves(myChanges) && allRemoves(theirChanges)) { |
||
| 216 | // Special case for remove changes that are supersets of one another |
||
| 217 | if ( /*istanbul ignore start*/(0, _array.arrayStartsWith) /*istanbul ignore end*/(myChanges, theirChanges) && skipRemoveSuperset(their, myChanges, myChanges.length - theirChanges.length)) { |
||
| 218 | /*istanbul ignore start*/var _hunk$lines3; |
||
| 219 | |||
| 220 | /*istanbul ignore end*/ /*istanbul ignore start*/(_hunk$lines3 = /*istanbul ignore end*/hunk.lines).push. /*istanbul ignore start*/apply /*istanbul ignore end*/( /*istanbul ignore start*/_hunk$lines3 /*istanbul ignore end*/, /*istanbul ignore start*/_toConsumableArray( /*istanbul ignore end*/myChanges)); |
||
| 221 | return; |
||
| 222 | } else if ( /*istanbul ignore start*/(0, _array.arrayStartsWith) /*istanbul ignore end*/(theirChanges, myChanges) && skipRemoveSuperset(mine, theirChanges, theirChanges.length - myChanges.length)) { |
||
| 223 | /*istanbul ignore start*/var _hunk$lines4; |
||
| 224 | |||
| 225 | /*istanbul ignore end*/ /*istanbul ignore start*/(_hunk$lines4 = /*istanbul ignore end*/hunk.lines).push. /*istanbul ignore start*/apply /*istanbul ignore end*/( /*istanbul ignore start*/_hunk$lines4 /*istanbul ignore end*/, /*istanbul ignore start*/_toConsumableArray( /*istanbul ignore end*/theirChanges)); |
||
| 226 | return; |
||
| 227 | } |
||
| 228 | } else if ( /*istanbul ignore start*/(0, _array.arrayEqual) /*istanbul ignore end*/(myChanges, theirChanges)) { |
||
| 229 | /*istanbul ignore start*/var _hunk$lines5; |
||
| 230 | |||
| 231 | /*istanbul ignore end*/ /*istanbul ignore start*/(_hunk$lines5 = /*istanbul ignore end*/hunk.lines).push. /*istanbul ignore start*/apply /*istanbul ignore end*/( /*istanbul ignore start*/_hunk$lines5 /*istanbul ignore end*/, /*istanbul ignore start*/_toConsumableArray( /*istanbul ignore end*/myChanges)); |
||
| 232 | return; |
||
| 233 | } |
||
| 234 | |||
| 235 | conflict(hunk, myChanges, theirChanges); |
||
| 236 | } |
||
| 237 | |||
| 238 | function removal(hunk, mine, their, swap) { |
||
| 239 | var myChanges = collectChange(mine), |
||
| 240 | theirChanges = collectContext(their, myChanges); |
||
| 241 | if (theirChanges.merged) { |
||
| 242 | /*istanbul ignore start*/var _hunk$lines6; |
||
| 243 | |||
| 244 | /*istanbul ignore end*/ /*istanbul ignore start*/(_hunk$lines6 = /*istanbul ignore end*/hunk.lines).push. /*istanbul ignore start*/apply /*istanbul ignore end*/( /*istanbul ignore start*/_hunk$lines6 /*istanbul ignore end*/, /*istanbul ignore start*/_toConsumableArray( /*istanbul ignore end*/theirChanges.merged)); |
||
| 245 | } else { |
||
| 246 | conflict(hunk, swap ? theirChanges : myChanges, swap ? myChanges : theirChanges); |
||
| 247 | } |
||
| 248 | } |
||
| 249 | |||
| 250 | function conflict(hunk, mine, their) { |
||
| 251 | hunk.conflict = true; |
||
| 252 | hunk.lines.push({ |
||
| 253 | conflict: true, |
||
| 254 | mine: mine, |
||
| 255 | theirs: their |
||
| 256 | }); |
||
| 257 | } |
||
| 258 | |||
| 259 | function insertLeading(hunk, insert, their) { |
||
| 260 | while (insert.offset < their.offset && insert.index < insert.lines.length) { |
||
| 261 | var line = insert.lines[insert.index++]; |
||
| 262 | hunk.lines.push(line); |
||
| 263 | insert.offset++; |
||
| 264 | } |
||
| 265 | } |
||
| 266 | function insertTrailing(hunk, insert) { |
||
| 267 | while (insert.index < insert.lines.length) { |
||
| 268 | var line = insert.lines[insert.index++]; |
||
| 269 | hunk.lines.push(line); |
||
| 270 | } |
||
| 271 | } |
||
| 272 | |||
| 273 | View Code Duplication | function collectChange(state) { |
|
| 274 | var ret = [], |
||
| 275 | operation = state.lines[state.index][0]; |
||
| 276 | while (state.index < state.lines.length) { |
||
| 277 | var line = state.lines[state.index]; |
||
| 278 | |||
| 279 | // Group additions that are immediately after subtractions and treat them as one "atomic" modify change. |
||
| 280 | if (operation === '-' && line[0] === '+') { |
||
| 281 | operation = '+'; |
||
| 282 | } |
||
| 283 | |||
| 284 | if (operation === line[0]) { |
||
| 285 | ret.push(line); |
||
| 286 | state.index++; |
||
| 287 | } else { |
||
| 288 | break; |
||
| 289 | } |
||
| 290 | } |
||
| 291 | |||
| 292 | return ret; |
||
| 293 | } |
||
| 294 | View Code Duplication | function collectContext(state, matchChanges) { |
|
| 295 | var changes = [], |
||
| 296 | merged = [], |
||
| 297 | matchIndex = 0, |
||
| 298 | contextChanges = false, |
||
| 299 | conflicted = false; |
||
| 300 | while (matchIndex < matchChanges.length && state.index < state.lines.length) { |
||
| 301 | var change = state.lines[state.index], |
||
| 302 | match = matchChanges[matchIndex]; |
||
| 303 | |||
| 304 | // Once we've hit our add, then we are done |
||
| 305 | if (match[0] === '+') { |
||
| 306 | break; |
||
| 307 | } |
||
| 308 | |||
| 309 | contextChanges = contextChanges || change[0] !== ' '; |
||
| 310 | |||
| 311 | merged.push(match); |
||
| 312 | matchIndex++; |
||
| 313 | |||
| 314 | // Consume any additions in the other block as a conflict to attempt |
||
| 315 | // to pull in the remaining context after this |
||
| 316 | if (change[0] === '+') { |
||
| 317 | conflicted = true; |
||
| 318 | |||
| 319 | while (change[0] === '+') { |
||
| 320 | changes.push(change); |
||
| 321 | change = state.lines[++state.index]; |
||
| 322 | } |
||
| 323 | } |
||
| 324 | |||
| 325 | if (match.substr(1) === change.substr(1)) { |
||
| 326 | changes.push(change); |
||
| 327 | state.index++; |
||
| 328 | } else { |
||
| 329 | conflicted = true; |
||
| 330 | } |
||
| 331 | } |
||
| 332 | |||
| 333 | if ((matchChanges[matchIndex] || '')[0] === '+' && contextChanges) { |
||
| 334 | conflicted = true; |
||
| 335 | } |
||
| 336 | |||
| 337 | if (conflicted) { |
||
| 338 | return changes; |
||
| 339 | } |
||
| 340 | |||
| 341 | while (matchIndex < matchChanges.length) { |
||
| 342 | merged.push(matchChanges[matchIndex++]); |
||
| 343 | } |
||
| 344 | |||
| 345 | return { |
||
| 346 | merged: merged, |
||
| 347 | changes: changes |
||
| 348 | }; |
||
| 349 | } |
||
| 350 | |||
| 351 | function allRemoves(changes) { |
||
| 352 | return changes.reduce(function (prev, change) { |
||
| 353 | return prev && change[0] === '-'; |
||
| 354 | }, true); |
||
| 355 | } |
||
| 356 | function skipRemoveSuperset(state, removeChanges, delta) { |
||
| 357 | for (var i = 0; i < delta; i++) { |
||
| 358 | var changeContent = removeChanges[removeChanges.length - delta + i].substr(1); |
||
| 359 | if (state.lines[state.index + i] !== ' ' + changeContent) { |
||
| 360 | return false; |
||
| 361 | } |
||
| 362 | } |
||
| 363 | |||
| 364 | state.index += delta; |
||
| 365 | return true; |
||
| 366 | } |
||
| 367 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/patch/merge.js"],"names":["calcLineCount","merge","hunk","conflicted","oldLines","newLines","lines","forEach","line","mine","theirs","base","loadPatch","ret","index","newFileName","fileNameChanged","oldFileName","oldHeader","newHeader","selectField","hunks","mineIndex","theirsIndex","mineOffset","theirsOffset","length","mineCurrent","oldStart","Infinity","theirsCurrent","hunkBefore","push","cloneHunk","mergedHunk","Math","min","newStart","mergeLines","param","test","Error","undefined","patch","conflict","check","offset","mineLines","theirOffset","theirLines","their","insertLeading","theirCurrent","mutualChange","collectChange","removal","insertTrailing","myChanges","theirChanges","allRemoves","skipRemoveSuperset","swap","collectContext","merged","insert","state","operation","matchChanges","changes","matchIndex","contextChanges","change","match","substr","reduce","prev","removeChanges","delta","i","changeContent"],"mappings":"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","file":"merge.js","sourcesContent":["import {structuredPatch} from './create';\nimport {parsePatch} from './parse';\n\nimport {arrayEqual, arrayStartsWith} from '../util/array';\n\nexport function calcLineCount(hunk) {\n  let conflicted = false;\n\n  hunk.oldLines = 0;\n  hunk.newLines = 0;\n\n  hunk.lines.forEach(function(line) {\n    if (typeof line !== 'string') {\n      conflicted = true;\n      return;\n    }\n\n    if (line[0] === '+' || line[0] === ' ') {\n      hunk.newLines++;\n    }\n    if (line[0] === '-' || line[0] === ' ') {\n      hunk.oldLines++;\n    }\n  });\n\n  if (conflicted) {\n    delete hunk.oldLines;\n    delete hunk.newLines;\n  }\n}\n\nexport function merge(mine, theirs, base) {\n  mine = loadPatch(mine, base);\n  theirs = loadPatch(theirs, base);\n\n  let ret = {};\n\n  // For index we just let it pass through as it doesn't have any necessary meaning.\n  // Leaving sanity checks on this to the API consumer that may know more about the\n  // meaning in their own context.\n  if (mine.index || theirs.index) {\n    ret.index = mine.index || theirs.index;\n  }\n\n  if (mine.newFileName || theirs.newFileName) {\n    if (!fileNameChanged(mine)) {\n      // No header or no change in ours, use theirs (and ours if theirs does not exist)\n      ret.oldFileName = theirs.oldFileName || mine.oldFileName;\n      ret.newFileName = theirs.newFileName || mine.newFileName;\n      ret.oldHeader = theirs.oldHeader || mine.oldHeader;\n      ret.newHeader = theirs.newHeader || mine.newHeader;\n    } else if (!fileNameChanged(theirs)) {\n      // No header or no change in theirs, use ours\n      ret.oldFileName = mine.oldFileName;\n      ret.newFileName = mine.newFileName;\n      ret.oldHeader = mine.oldHeader;\n      ret.newHeader = mine.newHeader;\n    } else {\n      // Both changed... figure it out\n      ret.oldFileName = selectField(ret, mine.oldFileName, theirs.oldFileName);\n      ret.newFileName = selectField(ret, mine.newFileName, theirs.newFileName);\n      ret.oldHeader = selectField(ret, mine.oldHeader, theirs.oldHeader);\n      ret.newHeader = selectField(ret, mine.newHeader, theirs.newHeader);\n    }\n  }\n\n  ret.hunks = [];\n\n  let mineIndex = 0,\n      theirsIndex = 0,\n      mineOffset = 0,\n      theirsOffset = 0;\n\n  while (mineIndex < mine.hunks.length || theirsIndex < theirs.hunks.length) {\n    let mineCurrent = mine.hunks[mineIndex] || {oldStart: Infinity},\n        theirsCurrent = theirs.hunks[theirsIndex] || {oldStart: Infinity};\n\n    if (hunkBefore(mineCurrent, theirsCurrent)) {\n      // This patch does not overlap with any of the others, yay.\n      ret.hunks.push(cloneHunk(mineCurrent, mineOffset));\n      mineIndex++;\n      theirsOffset += mineCurrent.newLines - mineCurrent.oldLines;\n    } else if (hunkBefore(theirsCurrent, mineCurrent)) {\n      // This patch does not overlap with any of the others, yay.\n      ret.hunks.push(cloneHunk(theirsCurrent, theirsOffset));\n      theirsIndex++;\n      mineOffset += theirsCurrent.newLines - theirsCurrent.oldLines;\n    } else {\n      // Overlap, merge as best we can\n      let mergedHunk = {\n        oldStart: Math.min(mineCurrent.oldStart, theirsCurrent.oldStart),\n        oldLines: 0,\n        newStart: Math.min(mineCurrent.newStart + mineOffset, theirsCurrent.oldStart + theirsOffset),\n        newLines: 0,\n        lines: []\n      };\n      mergeLines(mergedHunk, mineCurrent.oldStart, mineCurrent.lines, theirsCurrent.oldStart, theirsCurrent.lines);\n      theirsIndex++;\n      mineIndex++;\n\n      ret.hunks.push(mergedHunk);\n    }\n  }\n\n  return ret;\n}\n\nfunction loadPatch(param, base) {\n  if (typeof param === 'string') {\n    if (/^@@/m.test(param) || (/^Index:/m.test(param))) {\n      return parsePatch(param)[0];\n    }\n\n    if (!base) {\n      throw new Error('Must provide a base reference or pass in a patch');\n    }\n    return structuredPatch(undefined, undefined, base, param);\n  }\n\n  return param;\n}\n\nfunction fileNameChanged(patch) {\n  return patch.newFileName && patch.newFileName !== patch.oldFileName;\n}\n\nfunction selectField(index, mine, theirs) {\n  if (mine === theirs) {\n    return mine;\n  } else {\n    index.conflict = true;\n    return {mine, theirs};\n  }\n}\n\nfunction hunkBefore(test, check) {\n  return test.oldStart < check.oldStart\n    && (test.oldStart + test.oldLines) < check.oldStart;\n}\n\nfunction cloneHunk(hunk, offset) {\n  return {\n    oldStart: hunk.oldStart, oldLines: hunk.oldLines,\n    newStart: hunk.newStart + offset, newLines: hunk.newLines,\n    lines: hunk.lines\n  };\n}\n\nfunction mergeLines(hunk, mineOffset, mineLines, theirOffset, theirLines) {\n  // This will generally result in a conflicted hunk, but there are cases where the context\n  // is the only overlap where we can successfully merge the content here.\n  let mine = {offset: mineOffset, lines: mineLines, index: 0},\n      their = {offset: theirOffset, lines: theirLines, index: 0};\n\n  // Handle any leading content\n  insertLeading(hunk, mine, their);\n  insertLeading(hunk, their, mine);\n\n  // Now in the overlap content. Scan through and select the best changes from each.\n  while (mine.index < mine.lines.length && their.index < their.lines.length) {\n    let mineCurrent = mine.lines[mine.index],\n        theirCurrent = their.lines[their.index];\n\n    if ((mineCurrent[0] === '-' || mineCurrent[0] === '+')\n        && (theirCurrent[0] === '-' || theirCurrent[0] === '+')) {\n      // Both modified ...\n      mutualChange(hunk, mine, their);\n    } else if (mineCurrent[0] === '+' && theirCurrent[0] === ' ') {\n      // Mine inserted\n      hunk.lines.push(... collectChange(mine));\n    } else if (theirCurrent[0] === '+' && mineCurrent[0] === ' ') {\n      // Theirs inserted\n      hunk.lines.push(... collectChange(their));\n    } else if (mineCurrent[0] === '-' && theirCurrent[0] === ' ') {\n      // Mine removed or edited\n      removal(hunk, mine, their);\n    } else if (theirCurrent[0] === '-' && mineCurrent[0] === ' ') {\n      // Their removed or edited\n      removal(hunk, their, mine, true);\n    } else if (mineCurrent === theirCurrent) {\n      // Context identity\n      hunk.lines.push(mineCurrent);\n      mine.index++;\n      their.index++;\n    } else {\n      // Context mismatch\n      conflict(hunk, collectChange(mine), collectChange(their));\n    }\n  }\n\n  // Now push anything that may be remaining\n  insertTrailing(hunk, mine);\n  insertTrailing(hunk, their);\n\n  calcLineCount(hunk);\n}\n\nfunction mutualChange(hunk, mine, their) {\n  let myChanges = collectChange(mine),\n      theirChanges = collectChange(their);\n\n  if (allRemoves(myChanges) && allRemoves(theirChanges)) {\n    // Special case for remove changes that are supersets of one another\n    if (arrayStartsWith(myChanges, theirChanges)\n        && skipRemoveSuperset(their, myChanges, myChanges.length - theirChanges.length)) {\n      hunk.lines.push(... myChanges);\n      return;\n    } else if (arrayStartsWith(theirChanges, myChanges)\n        && skipRemoveSuperset(mine, theirChanges, theirChanges.length - myChanges.length)) {\n      hunk.lines.push(... theirChanges);\n      return;\n    }\n  } else if (arrayEqual(myChanges, theirChanges)) {\n    hunk.lines.push(... myChanges);\n    return;\n  }\n\n  conflict(hunk, myChanges, theirChanges);\n}\n\nfunction removal(hunk, mine, their, swap) {\n  let myChanges = collectChange(mine),\n      theirChanges = collectContext(their, myChanges);\n  if (theirChanges.merged) {\n    hunk.lines.push(... theirChanges.merged);\n  } else {\n    conflict(hunk, swap ? theirChanges : myChanges, swap ? myChanges : theirChanges);\n  }\n}\n\nfunction conflict(hunk, mine, their) {\n  hunk.conflict = true;\n  hunk.lines.push({\n    conflict: true,\n    mine: mine,\n    theirs: their\n  });\n}\n\nfunction insertLeading(hunk, insert, their) {\n  while (insert.offset < their.offset && insert.index < insert.lines.length) {\n    let line = insert.lines[insert.index++];\n    hunk.lines.push(line);\n    insert.offset++;\n  }\n}\nfunction insertTrailing(hunk, insert) {\n  while (insert.index < insert.lines.length) {\n    let line = insert.lines[insert.index++];\n    hunk.lines.push(line);\n  }\n}\n\nfunction collectChange(state) {\n  let ret = [],\n      operation = state.lines[state.index][0];\n  while (state.index < state.lines.length) {\n    let line = state.lines[state.index];\n\n    // Group additions that are immediately after subtractions and treat them as one \"atomic\" modify change.\n    if (operation === '-' && line[0] === '+') {\n      operation = '+';\n    }\n\n    if (operation === line[0]) {\n      ret.push(line);\n      state.index++;\n    } else {\n      break;\n    }\n  }\n\n  return ret;\n}\nfunction collectContext(state, matchChanges) {\n  let changes = [],\n      merged = [],\n      matchIndex = 0,\n      contextChanges = false,\n      conflicted = false;\n  while (matchIndex < matchChanges.length\n        && state.index < state.lines.length) {\n    let change = state.lines[state.index],\n        match = matchChanges[matchIndex];\n\n    // Once we've hit our add, then we are done\n    if (match[0] === '+') {\n      break;\n    }\n\n    contextChanges = contextChanges || change[0] !== ' ';\n\n    merged.push(match);\n    matchIndex++;\n\n    // Consume any additions in the other block as a conflict to attempt\n    // to pull in the remaining context after this\n    if (change[0] === '+') {\n      conflicted = true;\n\n      while (change[0] === '+') {\n        changes.push(change);\n        change = state.lines[++state.index];\n      }\n    }\n\n    if (match.substr(1) === change.substr(1)) {\n      changes.push(change);\n      state.index++;\n    } else {\n      conflicted = true;\n    }\n  }\n\n  if ((matchChanges[matchIndex] || '')[0] === '+'\n      && contextChanges) {\n    conflicted = true;\n  }\n\n  if (conflicted) {\n    return changes;\n  }\n\n  while (matchIndex < matchChanges.length) {\n    merged.push(matchChanges[matchIndex++]);\n  }\n\n  return {\n    merged,\n    changes\n  };\n}\n\nfunction allRemoves(changes) {\n  return changes.reduce(function(prev, change) {\n    return prev && change[0] === '-';\n  }, true);\n}\nfunction skipRemoveSuperset(state, removeChanges, delta) {\n  for (let i = 0; i < delta; i++) {\n    let changeContent = removeChanges[removeChanges.length - delta + i].substr(1);\n    if (state.lines[state.index + i] !== ' ' + changeContent) {\n      return false;\n    }\n  }\n\n  state.index += delta;\n  return true;\n}\n"]} |
||
| 368 |