mirror of
https://github.com/BookStackApp/BookStack.git
synced 2025-01-21 12:48:31 +08:00
a33dbcb04a
Count and reference list would get references then attempt to load entities, which could fail to load if in the recycle bin. This updates the queries to effectively ignore references for items we can't see (in recycle bin). Added test to cover. For #4918
308 lines
11 KiB
PHP
308 lines
11 KiB
PHP
<?php
|
|
|
|
namespace Tests\References;
|
|
|
|
use BookStack\App\Model;
|
|
use BookStack\Entities\Repos\PageRepo;
|
|
use BookStack\Entities\Tools\TrashCan;
|
|
use BookStack\References\Reference;
|
|
use Tests\TestCase;
|
|
|
|
class ReferencesTest extends TestCase
|
|
{
|
|
public function test_references_created_on_page_update()
|
|
{
|
|
$pageA = $this->entities->page();
|
|
$pageB = $this->entities->page();
|
|
|
|
$this->assertDatabaseMissing('references', ['from_id' => $pageA->id, 'from_type' => $pageA->getMorphClass()]);
|
|
|
|
$this->asEditor()->put($pageA->getUrl(), [
|
|
'name' => 'Reference test',
|
|
'html' => '<a href="' . $pageB->getUrl() . '">Testing</a>',
|
|
]);
|
|
|
|
$this->assertDatabaseHas('references', [
|
|
'from_id' => $pageA->id,
|
|
'from_type' => $pageA->getMorphClass(),
|
|
'to_id' => $pageB->id,
|
|
'to_type' => $pageB->getMorphClass(),
|
|
]);
|
|
}
|
|
|
|
public function test_references_created_on_book_chapter_bookshelf_update()
|
|
{
|
|
$entities = [$this->entities->book(), $this->entities->chapter(), $this->entities->shelf()];
|
|
$shelf = $this->entities->shelf();
|
|
|
|
foreach ($entities as $entity) {
|
|
$entity->refresh();
|
|
$this->assertDatabaseMissing('references', ['from_id' => $entity->id, 'from_type' => $entity->getMorphClass()]);
|
|
|
|
$this->asEditor()->put($entity->getUrl(), [
|
|
'name' => 'Reference test',
|
|
'description_html' => '<a href="' . $shelf->getUrl() . '">Testing</a>',
|
|
]);
|
|
|
|
$this->assertDatabaseHas('references', [
|
|
'from_id' => $entity->id,
|
|
'from_type' => $entity->getMorphClass(),
|
|
'to_id' => $shelf->id,
|
|
'to_type' => $shelf->getMorphClass(),
|
|
]);
|
|
}
|
|
}
|
|
|
|
public function test_references_deleted_on_page_delete()
|
|
{
|
|
$pageA = $this->entities->page();
|
|
$pageB = $this->entities->page();
|
|
|
|
$this->createReference($pageA, $pageB);
|
|
$this->createReference($pageB, $pageA);
|
|
|
|
$this->assertDatabaseHas('references', ['from_id' => $pageA->id, 'from_type' => $pageA->getMorphClass()]);
|
|
$this->assertDatabaseHas('references', ['to_id' => $pageA->id, 'to_type' => $pageA->getMorphClass()]);
|
|
|
|
app(PageRepo::class)->destroy($pageA);
|
|
app(TrashCan::class)->empty();
|
|
|
|
$this->assertDatabaseMissing('references', ['from_id' => $pageA->id, 'from_type' => $pageA->getMorphClass()]);
|
|
$this->assertDatabaseMissing('references', ['to_id' => $pageA->id, 'to_type' => $pageA->getMorphClass()]);
|
|
}
|
|
|
|
public function test_references_from_deleted_on_book_chapter_shelf_delete()
|
|
{
|
|
$entities = [$this->entities->chapter(), $this->entities->book(), $this->entities->shelf()];
|
|
$shelf = $this->entities->shelf();
|
|
|
|
foreach ($entities as $entity) {
|
|
$this->createReference($entity, $shelf);
|
|
$this->assertDatabaseHas('references', ['from_id' => $entity->id, 'from_type' => $entity->getMorphClass()]);
|
|
|
|
$this->asEditor()->delete($entity->getUrl());
|
|
app(TrashCan::class)->empty();
|
|
|
|
$this->assertDatabaseMissing('references', [
|
|
'from_id' => $entity->id,
|
|
'from_type' => $entity->getMorphClass()
|
|
]);
|
|
}
|
|
}
|
|
|
|
public function test_references_to_count_visible_on_entity_show_view()
|
|
{
|
|
$entities = $this->entities->all();
|
|
$otherPage = $this->entities->page();
|
|
|
|
$this->asEditor();
|
|
foreach ($entities as $entity) {
|
|
$this->createReference($entities['page'], $entity);
|
|
}
|
|
|
|
foreach ($entities as $entity) {
|
|
$resp = $this->get($entity->getUrl());
|
|
$resp->assertSee('Referenced by 1 item');
|
|
$resp->assertDontSee('Referenced by 1 items');
|
|
}
|
|
|
|
$this->createReference($otherPage, $entities['page']);
|
|
$resp = $this->get($entities['page']->getUrl());
|
|
$resp->assertSee('Referenced by 2 items');
|
|
}
|
|
|
|
public function test_references_to_visible_on_references_page()
|
|
{
|
|
$entities = $this->entities->all();
|
|
$this->asEditor();
|
|
foreach ($entities as $entity) {
|
|
$this->createReference($entities['page'], $entity);
|
|
}
|
|
|
|
foreach ($entities as $entity) {
|
|
$resp = $this->get($entity->getUrl('/references'));
|
|
$resp->assertSee('References');
|
|
$resp->assertSee($entities['page']->name);
|
|
$resp->assertDontSee('There are no tracked references');
|
|
}
|
|
}
|
|
|
|
public function test_reference_not_visible_if_view_permission_does_not_permit()
|
|
{
|
|
$page = $this->entities->page();
|
|
$pageB = $this->entities->page();
|
|
$this->createReference($pageB, $page);
|
|
|
|
$this->permissions->setEntityPermissions($pageB);
|
|
|
|
$this->asEditor()->get($page->getUrl('/references'))->assertDontSee($pageB->name);
|
|
$this->asAdmin()->get($page->getUrl('/references'))->assertSee($pageB->name);
|
|
}
|
|
|
|
public function test_reference_page_shows_empty_state_with_no_references()
|
|
{
|
|
$page = $this->entities->page();
|
|
|
|
$this->asEditor()
|
|
->get($page->getUrl('/references'))
|
|
->assertSee('There are no tracked references');
|
|
}
|
|
|
|
public function test_pages_leading_to_entity_updated_on_url_change()
|
|
{
|
|
$pageA = $this->entities->page();
|
|
$pageB = $this->entities->page();
|
|
$book = $this->entities->book();
|
|
|
|
foreach ([$pageA, $pageB] as $page) {
|
|
$page->html = '<a href="' . $book->getUrl() . '">Link</a>';
|
|
$page->save();
|
|
$this->createReference($page, $book);
|
|
}
|
|
|
|
$this->asEditor()->put($book->getUrl(), [
|
|
'name' => 'my updated book slugaroo',
|
|
]);
|
|
|
|
foreach ([$pageA, $pageB] as $page) {
|
|
$page->refresh();
|
|
$this->assertStringContainsString('href="http://localhost/books/my-updated-book-slugaroo"', $page->html);
|
|
$this->assertDatabaseHas('page_revisions', [
|
|
'page_id' => $page->id,
|
|
'summary' => 'System auto-update of internal links',
|
|
]);
|
|
}
|
|
}
|
|
|
|
public function test_pages_linking_to_other_page_updated_on_parent_book_url_change()
|
|
{
|
|
$bookPage = $this->entities->page();
|
|
$otherPage = $this->entities->page();
|
|
$book = $bookPage->book;
|
|
|
|
$otherPage->html = '<a href="' . $bookPage->getUrl() . '">Link</a>';
|
|
$otherPage->save();
|
|
$this->createReference($otherPage, $bookPage);
|
|
|
|
$this->asEditor()->put($book->getUrl(), [
|
|
'name' => 'my updated book slugaroo',
|
|
]);
|
|
|
|
$otherPage->refresh();
|
|
$this->assertStringContainsString('href="http://localhost/books/my-updated-book-slugaroo/page/' . $bookPage->slug . '"', $otherPage->html);
|
|
$this->assertDatabaseHas('page_revisions', [
|
|
'page_id' => $otherPage->id,
|
|
'summary' => 'System auto-update of internal links',
|
|
]);
|
|
}
|
|
|
|
public function test_pages_linking_to_chapter_updated_on_parent_book_url_change()
|
|
{
|
|
$bookChapter = $this->entities->chapter();
|
|
$otherPage = $this->entities->page();
|
|
$book = $bookChapter->book;
|
|
|
|
$otherPage->html = '<a href="' . $bookChapter->getUrl() . '">Link</a>';
|
|
$otherPage->save();
|
|
$this->createReference($otherPage, $bookChapter);
|
|
|
|
$this->asEditor()->put($book->getUrl(), [
|
|
'name' => 'my updated book slugaroo',
|
|
]);
|
|
|
|
$otherPage->refresh();
|
|
$this->assertStringContainsString('href="http://localhost/books/my-updated-book-slugaroo/chapter/' . $bookChapter->slug . '"', $otherPage->html);
|
|
$this->assertDatabaseHas('page_revisions', [
|
|
'page_id' => $otherPage->id,
|
|
'summary' => 'System auto-update of internal links',
|
|
]);
|
|
}
|
|
|
|
public function test_markdown_links_leading_to_entity_updated_on_url_change()
|
|
{
|
|
$page = $this->entities->page();
|
|
$book = $this->entities->book();
|
|
|
|
$bookUrl = $book->getUrl();
|
|
$markdown = '
|
|
[An awesome link](' . $bookUrl . ')
|
|
[An awesome link with query & hash](' . $bookUrl . '?test=yes#cats)
|
|
[An awesome link with path](' . $bookUrl . '/an/extra/trail)
|
|
[An awesome link with title](' . $bookUrl . ' "title")
|
|
[ref]: ' . $bookUrl . '?test=yes#dogs
|
|
[ref_without_space]:' . $bookUrl . '
|
|
[ref_with_title]: ' . $bookUrl . ' "title"';
|
|
$page->markdown = $markdown;
|
|
$page->save();
|
|
$this->createReference($page, $book);
|
|
|
|
$this->asEditor()->put($book->getUrl(), [
|
|
'name' => 'my updated book slugadoo',
|
|
]);
|
|
|
|
$page->refresh();
|
|
$expected = str_replace($bookUrl, 'http://localhost/books/my-updated-book-slugadoo', $markdown);
|
|
$this->assertEquals($expected, $page->markdown);
|
|
}
|
|
|
|
public function test_description_links_from_book_chapter_shelf_updated_on_url_change()
|
|
{
|
|
$entities = [$this->entities->chapter(), $this->entities->book(), $this->entities->shelf()];
|
|
$shelf = $this->entities->shelf();
|
|
$this->asEditor();
|
|
|
|
foreach ($entities as $entity) {
|
|
$this->put($entity->getUrl(), [
|
|
'name' => 'Reference test',
|
|
'description_html' => '<a href="' . $shelf->getUrl() . '">Testing</a>',
|
|
]);
|
|
}
|
|
|
|
$oldUrl = $shelf->getUrl();
|
|
$this->put($shelf->getUrl(), ['name' => 'My updated shelf link']);
|
|
$shelf->refresh();
|
|
$this->assertNotEquals($oldUrl, $shelf->getUrl());
|
|
|
|
foreach ($entities as $entity) {
|
|
$oldHtml = $entity->description_html;
|
|
$entity->refresh();
|
|
$this->assertNotEquals($oldHtml, $entity->description_html);
|
|
$this->assertStringContainsString($shelf->getUrl(), $entity->description_html);
|
|
}
|
|
}
|
|
|
|
public function test_reference_from_deleted_item_does_not_count_or_show_in_references_page()
|
|
{
|
|
$page = $this->entities->page();
|
|
$referencingPageA = $this->entities->page();
|
|
$referencingPageB = $this->entities->page();
|
|
|
|
$this->asEditor();
|
|
$this->createReference($referencingPageA, $page);
|
|
$this->createReference($referencingPageB, $page);
|
|
|
|
$resp = $this->get($page->getUrl());
|
|
$resp->assertSee('Referenced by 2 items');
|
|
|
|
$this->delete($referencingPageA->getUrl());
|
|
|
|
$resp = $this->get($page->getUrl());
|
|
$resp->assertSee('Referenced by 1 item');
|
|
|
|
$resp = $this->get($page->getUrl('/references'));
|
|
$resp->assertOk();
|
|
$resp->assertSee($referencingPageB->getUrl());
|
|
$resp->assertDontSee($referencingPageA->getUrl());
|
|
}
|
|
|
|
protected function createReference(Model $from, Model $to): void
|
|
{
|
|
(new Reference())->forceFill([
|
|
'from_type' => $from->getMorphClass(),
|
|
'from_id' => $from->id,
|
|
'to_type' => $to->getMorphClass(),
|
|
'to_id' => $to->id,
|
|
])->save();
|
|
}
|
|
}
|