2011-09-12 17:50:02 +02:00
|
|
|
<?php
|
2016-08-19 00:51:35 +02:00
|
|
|
|
2016-10-14 03:30:05 +02:00
|
|
|
namespace SilverStripe\View\Tests\Parsers;
|
|
|
|
|
2023-02-08 04:15:32 +01:00
|
|
|
use SebastianBergmann\Diff\Differ;
|
2016-08-19 00:51:35 +02:00
|
|
|
use SilverStripe\Dev\SapphireTest;
|
2023-02-02 00:37:32 +01:00
|
|
|
use SilverStripe\View\Parsers\HtmlDiff;
|
2024-09-18 03:53:44 +02:00
|
|
|
use PHPUnit\Framework\Attributes\DataProvider;
|
|
|
|
use SebastianBergmann\Diff\Output\DiffOnlyOutputBuilder;
|
2016-08-19 00:51:35 +02:00
|
|
|
|
2023-02-02 00:37:32 +01:00
|
|
|
class HtmlDiffTest extends SapphireTest
|
2016-12-16 05:34:21 +01:00
|
|
|
{
|
|
|
|
|
2023-01-10 22:45:26 +01:00
|
|
|
/**
|
|
|
|
* Most if not all other tests strip out the whitespace from comparisons to avoid complexities with checking
|
|
|
|
* if the actual HTML content is correct, since whitespace in HTML isn't all that vital and the algorithm
|
|
|
|
* can add some extra spaces where they're not stricly necessary but don't affect anything.
|
|
|
|
*
|
|
|
|
* This test is here to ensure that spaces _are_ kept where they're actually needed (i.e. between text)
|
|
|
|
*/
|
|
|
|
public function testKeepsSpacesBetweenText()
|
|
|
|
{
|
|
|
|
$from = '<span>Some text</span> <span>more text</span>';
|
|
|
|
$to = '<span>Other text</span> <span>more text</span>';
|
2023-02-02 00:37:32 +01:00
|
|
|
$diff = HtmlDiff::compareHtml($from, $to);
|
2023-01-10 22:45:26 +01:00
|
|
|
$this->assertEquals('<span><del>Some</del> <ins>Other</ins> text</span> <span>more text</span>', $diff, false);
|
|
|
|
|
|
|
|
// Note that the end result here isn't perfect (there are new spaces where there weren't before)...
|
|
|
|
// If we make improvements later on that keep only the original spaces, that would be preferred.
|
|
|
|
// This test is more here to protect against any unexpected changes to the spacing, so that we can make an intentional
|
|
|
|
// decision as to whether those changes are desirable.
|
2023-02-02 00:37:32 +01:00
|
|
|
$diff = HtmlDiff::compareHtml($from, $to, true);
|
2023-01-10 22:45:26 +01:00
|
|
|
$this->assertEquals('<span> <del>Some</del> <ins>Other</ins> text </span> <span> more text </span>', $diff, true);
|
|
|
|
}
|
|
|
|
|
2023-02-08 04:15:32 +01:00
|
|
|
/**
|
|
|
|
* The underlying SebastianBergmann\Diff\Differ class has a special constant for end of line differences
|
|
|
|
* but we shouldn't ever encounter that because of the way we're modifying the values before passing them
|
|
|
|
* in to that class.
|
|
|
|
*/
|
|
|
|
public function testEndOfLineDoesntNeedHandling()
|
|
|
|
{
|
|
|
|
$from = 'some change' . "\r";
|
|
|
|
$to = 'some change' . "\n";
|
|
|
|
|
|
|
|
// If we encounter an end-of-line difference, these should throw exceptions
|
|
|
|
$this->assertSame('some change', HtmlDiff::compareHtml($from, $to, true));
|
|
|
|
$this->assertSame('some change', HtmlDiff::compareHtml($from, $to));
|
|
|
|
$this->assertSame('some change', HtmlDiff::compareHtml($to, $from, true));
|
|
|
|
$this->assertSame('some change', HtmlDiff::compareHtml($to, $from));
|
|
|
|
|
|
|
|
// Ensure that this test is valid and that those changes would include an end-of-line warning
|
|
|
|
// in a direct call to the underlying differ
|
2024-09-18 03:53:44 +02:00
|
|
|
$differ = new Differ(new DiffOnlyOutputBuilder());
|
2023-02-08 04:15:32 +01:00
|
|
|
$expected = [
|
|
|
|
[
|
|
|
|
'#Warning: Strings contain different line endings!' . "\n",
|
|
|
|
Differ::DIFF_LINE_END_WARNING,
|
|
|
|
],
|
|
|
|
[
|
|
|
|
$from,
|
|
|
|
Differ::REMOVED,
|
|
|
|
],
|
|
|
|
[
|
|
|
|
$to,
|
|
|
|
Differ::ADDED,
|
|
|
|
],
|
|
|
|
];
|
|
|
|
$this->assertSame($expected, $differ->diffToArray($from, $to));
|
|
|
|
}
|
|
|
|
|
2024-09-18 03:53:44 +02:00
|
|
|
public static function provideCompareHtml(): array
|
2023-01-10 22:45:26 +01:00
|
|
|
{
|
|
|
|
return [
|
|
|
|
[
|
|
|
|
'from' => '<p><span>Some text</span></p>',
|
|
|
|
'to' => '<p><span>Other text</span></p>',
|
|
|
|
'escape' => false,
|
|
|
|
'expected' => '<p><span><del>Some</del><ins>Other</ins> text</span></p>',
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'from' => '<p><span>Some text</span></p>',
|
|
|
|
'to' => '<span>Other text</span>',
|
|
|
|
'escape' => false,
|
|
|
|
'expected' => '<del><p><span>Some text</span></p></del><ins><span>Other text</span></ins>',
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'from' => '<p><span>Some text</span></p>',
|
|
|
|
'to' => '<p>Other text</p>',
|
|
|
|
'escape' => false,
|
|
|
|
'expected' => '<p><del><span>Some text</span></del><ins>Other text</ins></p>',
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'from' => '<h2 class="mb-3 h4">About</h2>',
|
|
|
|
'to' => '<h2 class="mb-3 h2">About</h2>',
|
|
|
|
'escape' => false,
|
|
|
|
'expected' => '<del><h2 class="mb-3 h4">About</h2></del><ins><h2 class="mb-3 h2">About</h2></ins>',
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'from' => '<div class="BorderGrid-cell"><h2 class="mb-3 h4">About</h2><p class="f4 my-3">A comprehensive Library</p></div>',
|
|
|
|
'to' => '<div class="BorderGrid-cell"><h2 class="mb-3 h4">About</h2><span class="etc"><p class="f4 my-3">A comprehensive</p></span></div>',
|
|
|
|
'escape' => false,
|
|
|
|
'expected' => '<div class="BorderGrid-cell"><h2 class="mb-3 h4">About</h2><del><p class="f4 my-3">A comprehensive Library</p></del><ins><p class="f4 my-3"><span class="etc">A comprehensive</span></p></ins></div>',
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'from' => '<p><span>Some text</span><span>more stuff</span></p>',
|
|
|
|
'to' => '<p><span>Some text</span></p><p>more stuff</p>',
|
|
|
|
'escape' => false,
|
|
|
|
'expected' => '<p><span>Sometext</span><del><span>morestuff</span></del></p><ins><p>morestuff</p></ins>',
|
|
|
|
],
|
|
|
|
// Same examples as above, but with escaped HTML
|
|
|
|
[
|
|
|
|
'from' => '<p><span>Some text</span></p>',
|
|
|
|
'to' => '<p><span>Other text</span></p>',
|
|
|
|
'escape' => true,
|
|
|
|
'expected' => '<p><span><del>Some</del><ins>Other</ins> text</span></p>',
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'from' => '<p><span>Some text</span></p>',
|
|
|
|
'to' => '<span>Other text</span>',
|
|
|
|
'escape' => true,
|
|
|
|
'expected' => '<del><p></del><span><del>Some</del><ins>Other</ins> text</span><del></p></del>',
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'from' => '<p><span>Some text</span></p>',
|
|
|
|
'to' => '<p>Other text</p>',
|
|
|
|
'escape' => true,
|
|
|
|
'expected' => '<p><del><span>Some</del><ins>Other</ins> text<del></span></del></p>',
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'from' => '<h2 class="mb-3 h4">About</h2>',
|
|
|
|
'to' => '<h2 class="mb-3 h2">About</h2>',
|
|
|
|
'escape' => true,
|
|
|
|
// Note: This sees the whole h2 tag as being changed because of the initial call to explodeToHtmlChunks.
|
|
|
|
// There is room to improve this in the future, but care would have to be taken not to aversely affect other scenarios.
|
|
|
|
'expected' => '<del><h2 class="mb-3 h4"></del><ins><h2 class="mb-3 h2"></ins>About</h2>',
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'from' => '<div class="BorderGrid-cell"><h2 class="mb-3 h4">About</h2><p class="f4 my-3">A comprehensive Library</p></div>',
|
|
|
|
'to' => '<div class="BorderGrid-cell"><h2 class="mb-3 h4">About</h2><span class="etc"><p class="f4 my-3">A comprehensive</p></span></div>',
|
|
|
|
'escape' => true,
|
|
|
|
'expected' => '<div class="BorderGrid-cell"><h2 class="mb-3 h4">About</h2><ins><span class="etc"></ins><p class="f4 my-3">A comprehensive<del> Library</del></p><ins></span></ins></div>',
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'from' => '<p><span>Some text</span><span>more stuff</span></p>',
|
|
|
|
'to' => '<p><span>Some text</span></p><p>more stuff</p>',
|
|
|
|
'escape' => true,
|
|
|
|
'expected' => '<p><span>Some text</span><del><span></del><ins></p><p></ins>more stuff<del></span></del></p>',
|
|
|
|
],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
2024-09-18 03:53:44 +02:00
|
|
|
#[DataProvider('provideCompareHtml')]
|
2023-01-10 22:45:26 +01:00
|
|
|
public function testCompareHTML(string|array $from, string|array $to, bool $escape, string $expected)
|
|
|
|
{
|
2023-02-02 00:37:32 +01:00
|
|
|
$diff = HtmlDiff::compareHtml($from, $to, $escape);
|
2023-01-10 22:45:26 +01:00
|
|
|
$this->assertEquals($this->removeWhiteSpace($expected), $this->removeWhiteSpace($diff));
|
|
|
|
}
|
|
|
|
|
2016-12-16 05:34:21 +01:00
|
|
|
/**
|
|
|
|
* @see https://groups.google.com/forum/#!topic/silverstripe-dev/yHcluCvuszo
|
|
|
|
*/
|
|
|
|
public function testTableDiff()
|
|
|
|
{
|
|
|
|
if (!class_exists('DOMDocument')) {
|
|
|
|
$this->markTestSkipped('"DOMDocument" required');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-01-10 22:45:26 +01:00
|
|
|
$from = '<table>
|
2023-02-02 01:10:24 +01:00
|
|
|
<tbody>
|
|
|
|
<tr class="blah">
|
|
|
|
<td colspan="2">Row 1</td>
|
|
|
|
</tr>
|
|
|
|
<tr class="foo">
|
|
|
|
<td>Row 2</td>
|
|
|
|
<td>Row 2</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<td>Row 3</td>
|
|
|
|
<td>Row 3</td>
|
|
|
|
</tr>
|
2023-02-08 04:20:25 +01:00
|
|
|
</tbody>
|
2023-02-02 01:10:24 +01:00
|
|
|
</table>';
|
2011-09-12 17:50:02 +02:00
|
|
|
|
2023-01-10 22:45:26 +01:00
|
|
|
$to = '<table class="new-class">
|
2023-02-02 01:10:24 +01:00
|
|
|
<tbody>
|
|
|
|
<tr class="blah">
|
|
|
|
<td colspan="2">Row 1</td>
|
|
|
|
</tr>
|
|
|
|
<tr class="foo">
|
|
|
|
<td>Row 2</td>
|
|
|
|
<td>Row 2</td>
|
|
|
|
</tr>
|
|
|
|
</tbody>
|
|
|
|
</table>';
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2023-01-10 22:45:26 +01:00
|
|
|
$expected = '<del>' . $from . '</del>' . '<ins>' . $to . '</ins>';
|
2023-02-02 00:37:32 +01:00
|
|
|
$compare = HtmlDiff::compareHtml($from, $to);
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2023-01-10 22:45:26 +01:00
|
|
|
$this->assertEquals($this->removeWhiteSpace($expected), $this->removeWhiteSpace($compare));
|
2018-06-01 01:16:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @see https://github.com/silverstripe/silverstripe-framework/issues/8053
|
|
|
|
*/
|
|
|
|
public function testLegacyEachStatement()
|
|
|
|
{
|
|
|
|
$sentenceOne =
|
|
|
|
'Lorem ipsum dolor sit amet, consectetur adipiscing elit.';
|
|
|
|
$sentenceTwo =
|
|
|
|
'Nulla porttitor, ex quis commodo pharetra, diam dui efficitur justo, eu gravida elit eros vel libero.';
|
|
|
|
|
|
|
|
$from = "$sentenceOne $sentenceTwo";
|
|
|
|
$to = "$sentenceTwo $sentenceOne";
|
|
|
|
|
2018-06-01 03:02:49 +02:00
|
|
|
// We're cheating our test a little bit here, because depending on what HTML cleaner you have, you'll get
|
|
|
|
// spaces added or not added around the tags.
|
2023-01-10 22:45:26 +01:00
|
|
|
$quotedOne = preg_quote($sentenceOne, '/');
|
|
|
|
$quotedTwo = preg_quote($sentenceTwo, '/');
|
|
|
|
$expected = '/^ *<del>' . $quotedOne . '<\/del> *' . $quotedTwo . ' *<ins>' . $quotedOne . '<\/ins> *$/';
|
2023-02-02 00:37:32 +01:00
|
|
|
$actual = HtmlDiff::compareHtml($from, $to);
|
2018-06-01 01:16:03 +02:00
|
|
|
|
2021-10-27 04:39:47 +02:00
|
|
|
$this->assertMatchesRegularExpression($expected, $actual);
|
2016-12-16 05:34:21 +01:00
|
|
|
}
|
2020-03-17 03:57:28 +01:00
|
|
|
|
|
|
|
public function testDiffArray()
|
|
|
|
{
|
|
|
|
$from = ['Lorem', ['array here please ignore'], 'ipsum dolor'];
|
|
|
|
$to = 'Lorem,ipsum';
|
2023-01-10 22:45:26 +01:00
|
|
|
$expected = '/^Lorem,ipsum *<del>dolor<\/del> *$/';
|
2023-02-02 00:37:32 +01:00
|
|
|
$actual = HtmlDiff::compareHtml($from, $to);
|
2020-03-17 03:57:28 +01:00
|
|
|
|
2021-10-27 04:39:47 +02:00
|
|
|
$this->assertMatchesRegularExpression($expected, $actual);
|
2020-03-17 03:57:28 +01:00
|
|
|
}
|
2023-01-10 22:45:26 +01:00
|
|
|
|
|
|
|
private function removeWhiteSpace(string $value): string
|
|
|
|
{
|
|
|
|
return preg_replace('/[\s]*/', '', $value);
|
|
|
|
}
|
2012-03-24 04:04:52 +01:00
|
|
|
}
|