Input WYSIWYG: Updated tests, Added simple html limiting

This commit is contained in:
Dan Brown 2023-12-19 15:10:29 +00:00
parent 077b9709d4
commit 7fd6d5b2cc
No known key found for this signature in database
GPG Key ID: 46D9F943C24A2EF9
9 changed files with 134 additions and 36 deletions

View File

@ -10,6 +10,7 @@ use BookStack\Exceptions\ImageUploadException;
use BookStack\References\ReferenceStore; use BookStack\References\ReferenceStore;
use BookStack\References\ReferenceUpdater; use BookStack\References\ReferenceUpdater;
use BookStack\Uploads\ImageRepo; use BookStack\Uploads\ImageRepo;
use BookStack\Util\HtmlDescriptionFilter;
use Illuminate\Http\UploadedFile; use Illuminate\Http\UploadedFile;
class BaseRepo class BaseRepo
@ -111,7 +112,7 @@ class BaseRepo
/** @var HasHtmlDescription $entity */ /** @var HasHtmlDescription $entity */
if (isset($input['description_html'])) { if (isset($input['description_html'])) {
$entity->description_html = $input['description_html']; $entity->description_html = HtmlDescriptionFilter::filterFromString($input['description_html']);
$entity->description = html_entity_decode(strip_tags($input['description_html'])); $entity->description = html_entity_decode(strip_tags($input['description_html']));
} else if (isset($input['description'])) { } else if (isset($input['description'])) {
$entity->description = $input['description']; $entity->description = $input['description'];

View File

@ -0,0 +1,75 @@
<?php
namespace BookStack\Util;
use DOMAttr;
use DOMElement;
use DOMNamedNodeMap;
use DOMNode;
/**
* Filter to ensure HTML input for description content remains simple and
* to a limited allow-list of elements and attributes.
* More for consistency and to prevent nuisance rather than for security
* (which would be done via a separate content filter and CSP).
*/
class HtmlDescriptionFilter
{
/**
* @var array<string, string[]>
*/
protected static array $allowedAttrsByElements = [
'p' => [],
'a' => ['href', 'title'],
'ol' => [],
'ul' => [],
'li' => [],
'strong' => [],
'em' => [],
'br' => [],
];
public static function filterFromString(string $html): string
{
$doc = new HtmlDocument($html);
$topLevel = [...$doc->getBodyChildren()];
foreach ($topLevel as $child) {
/** @var DOMNode $child */
if ($child instanceof DOMElement) {
static::filterElement($child);
} else {
$child->parentNode->removeChild($child);
}
}
return $doc->getBodyInnerHtml();
}
protected static function filterElement(DOMElement $element): void
{
$elType = strtolower($element->tagName);
$allowedAttrs = static::$allowedAttrsByElements[$elType] ?? null;
if (is_null($allowedAttrs)) {
$element->remove();
return;
}
/** @var DOMNamedNodeMap $attrs */
$attrs = $element->attributes;
for ($i = $attrs->length - 1; $i >= 0; $i--) {
/** @var DOMAttr $attr */
$attr = $attrs->item($i);
$name = strtolower($attr->name);
if (!in_array($name, $allowedAttrs)) {
$element->removeAttribute($attr->name);
}
}
foreach ($element->childNodes as $child) {
if ($child instanceof DOMElement) {
static::filterElement($child);
}
}
}
}

View File

@ -21,10 +21,12 @@ class BookFactory extends Factory
*/ */
public function definition() public function definition()
{ {
$description = $this->faker->paragraph();
return [ return [
'name' => $this->faker->sentence(), 'name' => $this->faker->sentence(),
'slug' => Str::random(10), 'slug' => Str::random(10),
'description' => $this->faker->paragraph(), 'description' => $description,
'description_html' => '<p>' . e($description) . '</p>'
]; ];
} }
} }

View File

@ -21,10 +21,12 @@ class BookshelfFactory extends Factory
*/ */
public function definition() public function definition()
{ {
$description = $this->faker->paragraph();
return [ return [
'name' => $this->faker->sentence, 'name' => $this->faker->sentence,
'slug' => Str::random(10), 'slug' => Str::random(10),
'description' => $this->faker->paragraph, 'description' => $description,
'description_html' => '<p>' . e($description) . '</p>'
]; ];
} }
} }

View File

@ -21,10 +21,12 @@ class ChapterFactory extends Factory
*/ */
public function definition() public function definition()
{ {
$description = $this->faker->paragraph();
return [ return [
'name' => $this->faker->sentence(), 'name' => $this->faker->sentence(),
'slug' => Str::random(10), 'slug' => Str::random(10),
'description' => $this->faker->paragraph(), 'description' => $description,
'description_html' => '<p>' . e($description) . '</p>'
]; ];
} }
} }

View File

@ -329,7 +329,7 @@ export function buildForInput(options) {
menubar: false, menubar: false,
plugins: 'link autolink lists', plugins: 'link autolink lists',
contextmenu: false, contextmenu: false,
toolbar: 'bold italic underline link bullist numlist', toolbar: 'bold italic link bullist numlist',
content_style: getContentStyle(options), content_style: getContentStyle(options),
color_map: colorMap, color_map: colorMap,
file_picker_types: 'file', file_picker_types: 'file',

View File

@ -77,8 +77,8 @@ class BookShelfTest extends TestCase
{ {
$booksToInclude = Book::take(2)->get(); $booksToInclude = Book::take(2)->get();
$shelfInfo = [ $shelfInfo = [
'name' => 'My test book' . Str::random(4), 'name' => 'My test shelf' . Str::random(4),
'description' => 'Test book description ' . Str::random(10), 'description_html' => '<p>Test book description ' . Str::random(10) . '</p>',
]; ];
$resp = $this->asEditor()->post('/shelves', array_merge($shelfInfo, [ $resp = $this->asEditor()->post('/shelves', array_merge($shelfInfo, [
'books' => $booksToInclude->implode('id', ','), 'books' => $booksToInclude->implode('id', ','),
@ -96,7 +96,7 @@ class BookShelfTest extends TestCase
$shelf = Bookshelf::where('name', '=', $shelfInfo['name'])->first(); $shelf = Bookshelf::where('name', '=', $shelfInfo['name'])->first();
$shelfPage = $this->get($shelf->getUrl()); $shelfPage = $this->get($shelf->getUrl());
$shelfPage->assertSee($shelfInfo['name']); $shelfPage->assertSee($shelfInfo['name']);
$shelfPage->assertSee($shelfInfo['description']); $shelfPage->assertSee($shelfInfo['description_html'], false);
$this->withHtml($shelfPage)->assertElementContains('.tag-item', 'Test Category'); $this->withHtml($shelfPage)->assertElementContains('.tag-item', 'Test Category');
$this->withHtml($shelfPage)->assertElementContains('.tag-item', 'Test Tag Value'); $this->withHtml($shelfPage)->assertElementContains('.tag-item', 'Test Tag Value');
@ -107,8 +107,8 @@ class BookShelfTest extends TestCase
public function test_shelves_create_sets_cover_image() public function test_shelves_create_sets_cover_image()
{ {
$shelfInfo = [ $shelfInfo = [
'name' => 'My test book' . Str::random(4), 'name' => 'My test shelf' . Str::random(4),
'description' => 'Test book description ' . Str::random(10), 'description_html' => '<p>Test book description ' . Str::random(10) . '</p>',
]; ];
$imageFile = $this->files->uploadedImage('shelf-test.png'); $imageFile = $this->files->uploadedImage('shelf-test.png');
@ -174,7 +174,7 @@ class BookShelfTest extends TestCase
// Set book ordering // Set book ordering
$this->asAdmin()->put($shelf->getUrl(), [ $this->asAdmin()->put($shelf->getUrl(), [
'books' => $books->implode('id', ','), 'books' => $books->implode('id', ','),
'tags' => [], 'description' => 'abc', 'name' => 'abc', 'tags' => [], 'description_html' => 'abc', 'name' => 'abc',
]); ]);
$this->assertEquals(3, $shelf->books()->count()); $this->assertEquals(3, $shelf->books()->count());
$shelf->refresh(); $shelf->refresh();
@ -207,7 +207,7 @@ class BookShelfTest extends TestCase
// Set book ordering // Set book ordering
$this->asAdmin()->put($shelf->getUrl(), [ $this->asAdmin()->put($shelf->getUrl(), [
'books' => $books->implode('id', ','), 'books' => $books->implode('id', ','),
'tags' => [], 'description' => 'abc', 'name' => 'abc', 'tags' => [], 'description_html' => 'abc', 'name' => 'abc',
]); ]);
$this->assertEquals(3, $shelf->books()->count()); $this->assertEquals(3, $shelf->books()->count());
$shelf->refresh(); $shelf->refresh();
@ -229,8 +229,8 @@ class BookShelfTest extends TestCase
$booksToInclude = Book::take(2)->get(); $booksToInclude = Book::take(2)->get();
$shelfInfo = [ $shelfInfo = [
'name' => 'My test book' . Str::random(4), 'name' => 'My test shelf' . Str::random(4),
'description' => 'Test book description ' . Str::random(10), 'description_html' => '<p>Test book description ' . Str::random(10) . '</p>',
]; ];
$resp = $this->asEditor()->put($shelf->getUrl(), array_merge($shelfInfo, [ $resp = $this->asEditor()->put($shelf->getUrl(), array_merge($shelfInfo, [
@ -251,7 +251,7 @@ class BookShelfTest extends TestCase
$shelfPage = $this->get($shelf->getUrl()); $shelfPage = $this->get($shelf->getUrl());
$shelfPage->assertSee($shelfInfo['name']); $shelfPage->assertSee($shelfInfo['name']);
$shelfPage->assertSee($shelfInfo['description']); $shelfPage->assertSee($shelfInfo['description_html'], false);
$this->withHtml($shelfPage)->assertElementContains('.tag-item', 'Test Category'); $this->withHtml($shelfPage)->assertElementContains('.tag-item', 'Test Category');
$this->withHtml($shelfPage)->assertElementContains('.tag-item', 'Test Tag Value'); $this->withHtml($shelfPage)->assertElementContains('.tag-item', 'Test Tag Value');
@ -270,8 +270,8 @@ class BookShelfTest extends TestCase
$testName = 'Test Book in Shelf Name'; $testName = 'Test Book in Shelf Name';
$createBookResp = $this->asEditor()->post($shelf->getUrl('/create-book'), [ $createBookResp = $this->asEditor()->post($shelf->getUrl('/create-book'), [
'name' => $testName, 'name' => $testName,
'description' => 'Book in shelf description', 'description_html' => 'Book in shelf description',
]); ]);
$createBookResp->assertRedirect(); $createBookResp->assertRedirect();
@ -372,8 +372,8 @@ class BookShelfTest extends TestCase
{ {
// Create shelf // Create shelf
$shelfInfo = [ $shelfInfo = [
'name' => 'My test shelf' . Str::random(4), 'name' => 'My test shelf' . Str::random(4),
'description' => 'Test shelf description ' . Str::random(10), 'description_html' => '<p>Test shelf description ' . Str::random(10) . '</p>',
]; ];
$this->asEditor()->post('/shelves', $shelfInfo); $this->asEditor()->post('/shelves', $shelfInfo);
@ -381,8 +381,8 @@ class BookShelfTest extends TestCase
// Create book and add to shelf // Create book and add to shelf
$this->asEditor()->post($shelf->getUrl('/create-book'), [ $this->asEditor()->post($shelf->getUrl('/create-book'), [
'name' => 'Test book name', 'name' => 'Test book name',
'description' => 'Book in shelf description', 'description_html' => '<p>Book in shelf description</p>',
]); ]);
$newBook = Book::query()->orderBy('id', 'desc')->first(); $newBook = Book::query()->orderBy('id', 'desc')->first();

View File

@ -22,7 +22,7 @@ class BookTest extends TestCase
$resp = $this->get('/create-book'); $resp = $this->get('/create-book');
$this->withHtml($resp)->assertElementContains('form[action="' . url('/books') . '"][method="POST"]', 'Save Book'); $this->withHtml($resp)->assertElementContains('form[action="' . url('/books') . '"][method="POST"]', 'Save Book');
$resp = $this->post('/books', $book->only('name', 'description')); $resp = $this->post('/books', $book->only('name', 'description_html'));
$resp->assertRedirect('/books/my-first-book'); $resp->assertRedirect('/books/my-first-book');
$resp = $this->get('/books/my-first-book'); $resp = $this->get('/books/my-first-book');
@ -36,8 +36,8 @@ class BookTest extends TestCase
'name' => 'My First Book', 'name' => 'My First Book',
]); ]);
$this->asEditor()->post('/books', $book->only('name', 'description')); $this->asEditor()->post('/books', $book->only('name', 'description_html'));
$this->asEditor()->post('/books', $book->only('name', 'description')); $this->asEditor()->post('/books', $book->only('name', 'description_html'));
$books = Book::query()->where('name', '=', $book->name) $books = Book::query()->where('name', '=', $book->name)
->orderBy('id', 'desc') ->orderBy('id', 'desc')
@ -52,9 +52,9 @@ class BookTest extends TestCase
{ {
// Cheeky initial update to refresh slug // Cheeky initial update to refresh slug
$this->asEditor()->post('books', [ $this->asEditor()->post('books', [
'name' => 'My book with tags', 'name' => 'My book with tags',
'description' => 'A book with tags', 'description_html' => '<p>A book with tags</p>',
'tags' => [ 'tags' => [
[ [
'name' => 'Category', 'name' => 'Category',
'value' => 'Donkey Content', 'value' => 'Donkey Content',
@ -79,23 +79,23 @@ class BookTest extends TestCase
{ {
$book = $this->entities->book(); $book = $this->entities->book();
// Cheeky initial update to refresh slug // Cheeky initial update to refresh slug
$this->asEditor()->put($book->getUrl(), ['name' => $book->name . '5', 'description' => $book->description]); $this->asEditor()->put($book->getUrl(), ['name' => $book->name . '5', 'description_html' => $book->description_html]);
$book->refresh(); $book->refresh();
$newName = $book->name . ' Updated'; $newName = $book->name . ' Updated';
$newDesc = $book->description . ' with more content'; $newDesc = $book->description_html . '<p>with more content</p>';
$resp = $this->get($book->getUrl('/edit')); $resp = $this->get($book->getUrl('/edit'));
$resp->assertSee($book->name); $resp->assertSee($book->name);
$resp->assertSee($book->description); $resp->assertSee($book->description_html);
$this->withHtml($resp)->assertElementContains('form[action="' . $book->getUrl() . '"]', 'Save Book'); $this->withHtml($resp)->assertElementContains('form[action="' . $book->getUrl() . '"]', 'Save Book');
$resp = $this->put($book->getUrl(), ['name' => $newName, 'description' => $newDesc]); $resp = $this->put($book->getUrl(), ['name' => $newName, 'description_html' => $newDesc]);
$resp->assertRedirect($book->getUrl() . '-updated'); $resp->assertRedirect($book->getUrl() . '-updated');
$resp = $this->get($book->getUrl() . '-updated'); $resp = $this->get($book->getUrl() . '-updated');
$resp->assertSee($newName); $resp->assertSee($newName);
$resp->assertSee($newDesc); $resp->assertSee($newDesc, false);
} }
public function test_update_sets_tags() public function test_update_sets_tags()
@ -184,7 +184,7 @@ class BookTest extends TestCase
public function test_recently_viewed_books_updates_as_expected() public function test_recently_viewed_books_updates_as_expected()
{ {
$books = Book::all()->take(2); $books = Book::take(2)->get();
$resp = $this->asAdmin()->get('/books'); $resp = $this->asAdmin()->get('/books');
$this->withHtml($resp)->assertElementNotContains('#recents', $books[0]->name) $this->withHtml($resp)->assertElementNotContains('#recents', $books[0]->name)
@ -200,7 +200,7 @@ class BookTest extends TestCase
public function test_popular_books_updates_upon_visits() public function test_popular_books_updates_upon_visits()
{ {
$books = Book::all()->take(2); $books = Book::take(2)->get();
$resp = $this->asAdmin()->get('/books'); $resp = $this->asAdmin()->get('/books');
$this->withHtml($resp)->assertElementNotContains('#popular', $books[0]->name) $this->withHtml($resp)->assertElementNotContains('#popular', $books[0]->name)
@ -262,6 +262,22 @@ class BookTest extends TestCase
$this->assertEquals('parta-partb-partc', $book->slug); $this->assertEquals('parta-partb-partc', $book->slug);
} }
public function test_description_limited_to_specific_html()
{
$book = $this->entities->book();
$input = '<h1>Test</h1><p id="abc" href="beans">Content<a href="#cat" data-a="b">a</a><section>Hello</section></p>';
$expected = '<p>Content<a href="#cat">a</a></p>';
$this->asEditor()->put($book->getUrl(), [
'name' => $book->name,
'description_html' => $input
]);
$book->refresh();
$this->assertEquals($expected, $book->description_html);
}
public function test_show_view_has_copy_button() public function test_show_view_has_copy_button()
{ {
$book = $this->entities->book(); $book = $this->entities->book();

View File

@ -23,12 +23,12 @@ class ChapterTest extends TestCase
$resp = $this->get($book->getUrl('/create-chapter')); $resp = $this->get($book->getUrl('/create-chapter'));
$this->withHtml($resp)->assertElementContains('form[action="' . $book->getUrl('/create-chapter') . '"][method="POST"]', 'Save Chapter'); $this->withHtml($resp)->assertElementContains('form[action="' . $book->getUrl('/create-chapter') . '"][method="POST"]', 'Save Chapter');
$resp = $this->post($book->getUrl('/create-chapter'), $chapter->only('name', 'description')); $resp = $this->post($book->getUrl('/create-chapter'), $chapter->only('name', 'description_html'));
$resp->assertRedirect($book->getUrl('/chapter/my-first-chapter')); $resp->assertRedirect($book->getUrl('/chapter/my-first-chapter'));
$resp = $this->get($book->getUrl('/chapter/my-first-chapter')); $resp = $this->get($book->getUrl('/chapter/my-first-chapter'));
$resp->assertSee($chapter->name); $resp->assertSee($chapter->name);
$resp->assertSee($chapter->description); $resp->assertSee($chapter->description_html, false);
} }
public function test_delete() public function test_delete()