92 lines
2.7 KiB
PHP
92 lines
2.7 KiB
PHP
|
<?php
|
||
|
|
||
|
/*
|
||
|
* This file is part of the Symfony package.
|
||
|
*
|
||
|
* (c) Fabien Potencier <fabien@symfony.com>
|
||
|
*
|
||
|
* For the full copyright and license information, please view the LICENSE
|
||
|
* file that was distributed with this source code.
|
||
|
*/
|
||
|
|
||
|
namespace Sensio\Bundle\DistributionBundle\Diff;
|
||
|
|
||
|
/**
|
||
|
* Computes a Diff between files (line by line).
|
||
|
*
|
||
|
* Implements the Longest common subsequence problem algorithm.
|
||
|
*
|
||
|
* @see http://en.wikipedia.org/wiki/Longest_common_subsequence_problem
|
||
|
*
|
||
|
* @author Fabien Potencier <fabien@symfony.com>
|
||
|
*/
|
||
|
class Diff
|
||
|
{
|
||
|
private $diff;
|
||
|
|
||
|
public function __construct($str1, $str2)
|
||
|
{
|
||
|
$lines1 = explode("\n", $str1);
|
||
|
$lines2 = explode("\n", $str2);
|
||
|
|
||
|
$this->diff = $this->computeDiff($this->computeLcs($lines1, $lines2), $lines1, $lines2, count($lines1) - 1, count($lines2) - 1);
|
||
|
}
|
||
|
|
||
|
public function getDiff()
|
||
|
{
|
||
|
$diff = array();
|
||
|
for ($i = 0, $max = count($this->diff); $i < $max; $i++) {
|
||
|
if ('' != $this->diff[$i][0]) {
|
||
|
$diff[] = array('@', sprintf(' Line %s', $this->diff[$i][2]));
|
||
|
|
||
|
do {
|
||
|
$diff[] = $this->diff[$i++];
|
||
|
} while ('' != $this->diff[$i][0]);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $diff;
|
||
|
}
|
||
|
|
||
|
public function computeDiff(array $c, array $lines1, array $lines2, $i, $j)
|
||
|
{
|
||
|
$diff = array();
|
||
|
|
||
|
if ($i > -1 && $j > -1 && $lines1[$i] == $lines2[$j]) {
|
||
|
$diff = array_merge($diff, $this->computeDiff($c, $lines1, $lines2, $i - 1, $j - 1));
|
||
|
$diff[] = array('', $lines1[$i], $i, $j);
|
||
|
} else {
|
||
|
if ($j > -1 && ($i == -1 || $c[$i][$j - 1] >= $c[$i - 1][$j])) {
|
||
|
$diff = array_merge($diff, $this->computeDiff($c, $lines1, $lines2, $i, $j - 1));
|
||
|
$diff[] = array('+', $lines2[$j], $i, $j);
|
||
|
} elseif ($i > -1 && ($j == -1 || $c[$i][$j - 1] < $c[$i - 1][$j])) {
|
||
|
$diff = array_merge($diff, $this->computeDiff($c, $lines1, $lines2, $i - 1, $j));
|
||
|
$diff[] = array('-', $lines1[$i], $i, $j);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $diff;
|
||
|
}
|
||
|
|
||
|
private function computeLcs(array $lines1, array $lines2)
|
||
|
{
|
||
|
for ($i = -1, $len1 = count($lines1); $i < $len1; $i++) {
|
||
|
for ($j = -1, $len2 = count($lines2); $j < $len2; $j++) {
|
||
|
$c[$i][$j] = 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
for ($i = 0; $i < $len1; $i++) {
|
||
|
for ($j = 0; $j < $len2; $j++) {
|
||
|
if ($lines1[$i] == $lines2[$j]) {
|
||
|
$c[$i][$j] = $c[$i - 1][$j - 1] + 1;
|
||
|
} else {
|
||
|
$c[$i][$j] = max($c[$i][$j - 1], $c[$i - 1][$j]);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $c;
|
||
|
}
|
||
|
}
|