Xdiff.php 2.3 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667
  1. <?php
  2. /**
  3. * Class used internally by Diff to actually compute the diffs.
  4. *
  5. * This class uses the xdiff PECL package (http://pecl.php.net/package/xdiff)
  6. * to compute the differences between the two input arrays.
  7. *
  8. * Copyright 2004-2012 Horde LLC (http://www.horde.org/)
  9. *
  10. * See the enclosed file COPYING for license information (LGPL). If you did
  11. * not receive this file, see http://www.horde.org/licenses/lgpl21.
  12. *
  13. * @author Jon Parise <jon@horde.org>
  14. * @package Text_Diff
  15. */
  16. class Horde_Text_Diff_Engine_Xdiff
  17. {
  18. /**
  19. */
  20. public function diff($from_lines, $to_lines)
  21. {
  22. if (!extension_loaded('xdiff')) {
  23. throw new Horde_Text_Diff_Exception('The xdiff extension is required for this diff engine');
  24. }
  25. array_walk($from_lines, ['Horde_Text_Diff', 'trimNewlines']);
  26. array_walk($to_lines, ['Horde_Text_Diff', 'trimNewlines']);
  27. /* Convert the two input arrays into strings for xdiff processing. */
  28. $from_string = implode("\n", $from_lines);
  29. $to_string = implode("\n", $to_lines);
  30. /* Diff the two strings and convert the result to an array. */
  31. $diff = xdiff_string_diff($from_string, $to_string, count($to_lines));
  32. $diff = explode("\n", $diff);
  33. /* Walk through the diff one line at a time. We build the $edits
  34. * array of diff operations by reading the first character of the
  35. * xdiff output (which is in the "unified diff" format).
  36. *
  37. * Note that we don't have enough information to detect "changed"
  38. * lines using this approach, so we can't add Horde_Text_Diff_Op_Changed
  39. * instances to the $edits array. The result is still perfectly
  40. * valid, albeit a little less descriptive and efficient. */
  41. $edits = [];
  42. foreach ($diff as $line) {
  43. if (!strlen($line)) {
  44. continue;
  45. }
  46. switch ($line[0]) {
  47. case ' ':
  48. $edits[] = new Horde_Text_Diff_Op_Copy([substr($line, 1)]);
  49. break;
  50. case '+':
  51. $edits[] = new Horde_Text_Diff_Op_Add([substr($line, 1)]);
  52. break;
  53. case '-':
  54. $edits[] = new Horde_Text_Diff_Op_Delete([substr($line, 1)]);
  55. break;
  56. }
  57. }
  58. return $edits;
  59. }
  60. }