2023-11-22 23:14:28 +01:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Tests\Unit;
|
|
|
|
|
2023-11-27 22:38:43 +01:00
|
|
|
use BookStack\Entities\Tools\PageIncludeContent;
|
2023-11-22 23:14:28 +01:00
|
|
|
use BookStack\Entities\Tools\PageIncludeParser;
|
2023-11-27 22:38:43 +01:00
|
|
|
use BookStack\Entities\Tools\PageIncludeTag;
|
2023-11-27 20:54:47 +01:00
|
|
|
use BookStack\Util\HtmlDocument;
|
2023-11-22 23:14:28 +01:00
|
|
|
use Tests\TestCase;
|
|
|
|
|
|
|
|
class PageIncludeParserTest extends TestCase
|
|
|
|
{
|
2023-11-25 00:39:16 +01:00
|
|
|
public function test_simple_inline_text()
|
2023-11-22 23:14:28 +01:00
|
|
|
{
|
|
|
|
$this->runParserTest(
|
|
|
|
'<p>{{@45#content}}</p>',
|
|
|
|
['45' => '<p id="content">Testing</p>'],
|
|
|
|
'<p>Testing</p>',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-11-25 00:39:16 +01:00
|
|
|
public function test_simple_inline_text_with_existing_siblings()
|
2023-11-22 23:14:28 +01:00
|
|
|
{
|
|
|
|
$this->runParserTest(
|
|
|
|
'<p>{{@45#content}} <strong>Hi</strong>there!</p>',
|
|
|
|
['45' => '<p id="content">Testing</p>'],
|
|
|
|
'<p>Testing <strong>Hi</strong>there!</p>',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-11-25 00:39:16 +01:00
|
|
|
public function test_simple_inline_text_within_other_text()
|
2023-11-22 23:14:28 +01:00
|
|
|
{
|
|
|
|
$this->runParserTest(
|
|
|
|
'<p>Hello {{@45#content}}there!</p>',
|
|
|
|
['45' => '<p id="content">Testing</p>'],
|
|
|
|
'<p>Hello Testingthere!</p>',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-11-25 18:32:00 +01:00
|
|
|
public function test_complex_inline_text_within_other_text()
|
|
|
|
{
|
|
|
|
$this->runParserTest(
|
|
|
|
'<p>Hello {{@45#content}}there!</p>',
|
|
|
|
['45' => '<p id="content"><strong>Testing</strong> with<em>some</em><i>extra</i>tags</p>'],
|
|
|
|
'<p>Hello <strong>Testing</strong> with<em>some</em><i>extra</i>tagsthere!</p>',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-11-25 00:39:16 +01:00
|
|
|
public function test_block_content_types()
|
|
|
|
{
|
|
|
|
$inputs = [
|
|
|
|
'<table id="content"><td>Text</td></table>',
|
|
|
|
'<ul id="content"><li>Item A</li></ul>',
|
|
|
|
'<ol id="content"><li>Item A</li></ol>',
|
|
|
|
'<pre id="content">Code</pre>',
|
|
|
|
];
|
|
|
|
|
|
|
|
foreach ($inputs as $input) {
|
|
|
|
$this->runParserTest(
|
|
|
|
'<p>A{{@45#content}}B</p>',
|
|
|
|
['45' => $input],
|
|
|
|
'<p>A</p>' . $input . '<p>B</p>',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_block_content_nested_origin_gets_placed_before()
|
|
|
|
{
|
|
|
|
$this->runParserTest(
|
|
|
|
'<p><strong>A {{@45#content}} there!</strong></p>',
|
|
|
|
['45' => '<pre id="content">Testing</pre>'],
|
|
|
|
'<pre id="content">Testing</pre><p><strong>A there!</strong></p>',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_block_content_nested_origin_gets_placed_after()
|
|
|
|
{
|
|
|
|
$this->runParserTest(
|
|
|
|
'<p><strong>Some really good {{@45#content}} there!</strong></p>',
|
|
|
|
['45' => '<pre id="content">Testing</pre>'],
|
|
|
|
'<p><strong>Some really good there!</strong></p><pre id="content">Testing</pre>',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_block_content_in_shallow_origin_gets_split()
|
|
|
|
{
|
|
|
|
$this->runParserTest(
|
|
|
|
'<p>Some really good {{@45#content}} there!</p>',
|
|
|
|
['45' => '<pre id="content">doggos</pre>'],
|
|
|
|
'<p>Some really good </p><pre id="content">doggos</pre><p> there!</p>',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_block_content_in_shallow_origin_split_does_not_duplicate_id()
|
|
|
|
{
|
|
|
|
$this->runParserTest(
|
|
|
|
'<p id="test" title="Hi">Some really good {{@45#content}} there!</p>',
|
|
|
|
['45' => '<pre id="content">doggos</pre>'],
|
|
|
|
'<p title="Hi">Some really good </p><pre id="content">doggos</pre><p id="test" title="Hi"> there!</p>',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_block_content_in_shallow_origin_does_not_leave_empty_nodes()
|
|
|
|
{
|
|
|
|
$this->runParserTest(
|
|
|
|
'<p>{{@45#content}}</p>',
|
|
|
|
['45' => '<pre id="content">doggos</pre>'],
|
|
|
|
'<pre id="content">doggos</pre>',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-11-25 18:32:00 +01:00
|
|
|
public function test_block_content_in_allowable_parent_element()
|
|
|
|
{
|
|
|
|
$this->runParserTest(
|
|
|
|
'<div>{{@45#content}}</div>',
|
|
|
|
['45' => '<pre id="content">doggos</pre>'],
|
|
|
|
'<div><pre id="content">doggos</pre></div>',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_block_content_in_paragraph_origin_with_allowable_grandparent()
|
|
|
|
{
|
|
|
|
$this->runParserTest(
|
|
|
|
'<div><p>{{@45#content}}</p></div>',
|
|
|
|
['45' => '<pre id="content">doggos</pre>'],
|
|
|
|
'<div><pre id="content">doggos</pre></div>',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_block_content_in_paragraph_origin_with_allowable_grandparent_with_adjacent_content()
|
|
|
|
{
|
|
|
|
$this->runParserTest(
|
|
|
|
'<div><p>Cute {{@45#content}} over there!</p></div>',
|
|
|
|
['45' => '<pre id="content">doggos</pre>'],
|
|
|
|
'<div><p>Cute </p><pre id="content">doggos</pre><p> over there!</p></div>',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_block_content_in_child_within_paragraph_origin_with_allowable_grandparent_with_adjacent_content()
|
|
|
|
{
|
|
|
|
$this->runParserTest(
|
|
|
|
'<div><p><strong>Cute {{@45#content}} over there!</strong></p></div>',
|
|
|
|
['45' => '<pre id="content">doggos</pre>'],
|
|
|
|
'<div><pre id="content">doggos</pre><p><strong>Cute over there!</strong></p></div>',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_block_content_in_paragraph_origin_within_details()
|
|
|
|
{
|
|
|
|
$this->runParserTest(
|
|
|
|
'<details><p>{{@45#content}}</p></details>',
|
|
|
|
['45' => '<pre id="content">doggos</pre>'],
|
|
|
|
'<details><pre id="content">doggos</pre></details>',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-11-25 00:39:16 +01:00
|
|
|
public function test_simple_whole_document()
|
|
|
|
{
|
|
|
|
$this->runParserTest(
|
|
|
|
'<p>{{@45}}</p>',
|
|
|
|
['45' => '<p id="content">Testing</p>'],
|
|
|
|
'<p id="content">Testing</p>',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_multi_source_elem_whole_document()
|
|
|
|
{
|
|
|
|
$this->runParserTest(
|
|
|
|
'<p>{{@45}}</p>',
|
|
|
|
['45' => '<p>Testing</p><blockquote>This</blockquote>'],
|
|
|
|
'<p>Testing</p><blockquote>This</blockquote>',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_multi_source_elem_whole_document_with_shared_content_origin()
|
|
|
|
{
|
|
|
|
$this->runParserTest(
|
|
|
|
'<p>This is {{@45}} some text</p>',
|
|
|
|
['45' => '<p>Testing</p><blockquote>This</blockquote>'],
|
|
|
|
'<p>This is </p><p>Testing</p><blockquote>This</blockquote><p> some text</p>',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-11-27 21:16:27 +01:00
|
|
|
public function test_multi_source_elem_whole_document_with_nested_content_origin()
|
|
|
|
{
|
|
|
|
$this->runParserTest(
|
|
|
|
'<p><strong>{{@45}}</strong></p>',
|
|
|
|
['45' => '<p>Testing</p><blockquote>This</blockquote>'],
|
|
|
|
'<p>Testing</p><blockquote>This</blockquote>',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-11-25 18:32:00 +01:00
|
|
|
public function test_multiple_tags_in_same_origin_with_inline_content()
|
|
|
|
{
|
|
|
|
$this->runParserTest(
|
|
|
|
'<p>This {{@45#content}}{{@45#content}} content is {{@45#content}}</p>',
|
|
|
|
['45' => '<p id="content">inline</p>'],
|
|
|
|
'<p>This inlineinline content is inline</p>',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_multiple_tags_in_same_origin_with_block_content()
|
|
|
|
{
|
|
|
|
$this->runParserTest(
|
|
|
|
'<p>This {{@45#content}}{{@45#content}} content is {{@45#content}}</p>',
|
|
|
|
['45' => '<pre id="content">block</pre>'],
|
|
|
|
'<p>This </p><pre id="content">block</pre><pre id="content">block</pre><p> content is </p><pre id="content">block</pre>',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_multiple_tags_in_differing_origin_levels_with_block_content()
|
|
|
|
{
|
|
|
|
$this->runParserTest(
|
|
|
|
'<div><p>This <strong>{{@45#content}}</strong> content is {{@45#content}}</p>{{@45#content}}</div>',
|
|
|
|
['45' => '<pre id="content">block</pre>'],
|
2023-11-27 21:16:27 +01:00
|
|
|
'<div><pre id="content">block</pre><p>This content is </p><pre id="content">block</pre><pre id="content">block</pre></div>',
|
2023-11-25 18:32:00 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_multiple_tags_in_shallow_origin_with_multi_block_content()
|
|
|
|
{
|
|
|
|
$this->runParserTest(
|
|
|
|
'<p>{{@45}}C{{@45}}</p><div>{{@45}}{{@45}}</div>',
|
|
|
|
['45' => '<p>A</p><p>B</p>'],
|
|
|
|
'<p>A</p><p>B</p><p>C</p><p>A</p><p>B</p><div><p>A</p><p>B</p><p>A</p><p>B</p></div>',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-11-27 20:54:47 +01:00
|
|
|
protected function runParserTest(string $html, array $contentById, string $expected): void
|
2023-11-22 23:14:28 +01:00
|
|
|
{
|
2023-11-27 20:54:47 +01:00
|
|
|
$doc = new HtmlDocument($html);
|
2023-11-27 22:38:43 +01:00
|
|
|
$parser = new PageIncludeParser($doc, function (PageIncludeTag $tag) use ($contentById): PageIncludeContent {
|
|
|
|
$html = $contentById[strval($tag->getPageId())] ?? '';
|
|
|
|
return PageIncludeContent::fromHtmlAndTag($html, $tag);
|
2023-11-22 23:14:28 +01:00
|
|
|
});
|
|
|
|
|
2023-11-27 20:54:47 +01:00
|
|
|
$parser->parse();
|
|
|
|
$this->assertEquals($expected, $doc->getBodyInnerHtml());
|
2023-11-22 23:14:28 +01:00
|
|
|
}
|
|
|
|
}
|