1
0
mirror of https://github.com/BookStackApp/BookStack.git synced 2025-12-13 07:42:23 +03:00

Slugs: Fixed storage bugs, added testing coverage

This commit is contained in:
Dan Brown
2025-11-24 10:46:24 +00:00
parent 291a807d98
commit dd5375f480
5 changed files with 143 additions and 51 deletions

View File

@@ -165,10 +165,7 @@ class BaseRepo
*/
public function refreshSlug(Entity $entity): void
{
if ($entity->id) {
$this->slugHistory->recordForEntity($entity);
}
$this->slugHistory->recordForEntity($entity);
$this->slugGenerator->regenerateForEntity($entity);
}
}

View File

@@ -2,6 +2,7 @@
namespace BookStack\Entities\Tools;
use BookStack\Entities\Models\BookChild;
use BookStack\Entities\Models\Entity;
use Illuminate\Support\Facades\DB;
@@ -12,16 +13,25 @@ class SlugHistory
*/
public function recordForEntity(Entity $entity): void
{
if (!$entity->id || !$entity->slug) {
return;
}
$latest = $this->getLatestEntryForEntity($entity);
if ($latest && $latest->slug === $entity->slug && $latest->parent_slug === $entity->getParent()?->slug) {
return;
}
$parentSlug = null;
if ($entity instanceof BookChild) {
$parentSlug = $entity->book()->first()?->slug;
}
$info = [
'sluggable_type' => $entity->getMorphClass(),
'sluggable_id' => $entity->id,
'slug' => $entity->slug,
'parent_slug' => $entity->getParent()?->slug,
'parent_slug' => $parentSlug,
'created_at' => now(),
'updated_at' => now(),
];

View File

@@ -238,30 +238,6 @@ class BookTest extends TestCase
$this->assertEquals('list', setting()->getUser($editor, 'books_view_type'));
}
public function test_slug_multi_byte_url_safe()
{
$book = $this->entities->newBook([
'name' => 'информация',
]);
$this->assertEquals('informaciia', $book->slug);
$book = $this->entities->newBook([
'name' => '¿Qué?',
]);
$this->assertEquals('que', $book->slug);
}
public function test_slug_format()
{
$book = $this->entities->newBook([
'name' => 'PartA / PartB / PartC',
]);
$this->assertEquals('parta-partb-partc', $book->slug);
}
public function test_description_limited_to_specific_html()
{
$book = $this->entities->book();

View File

@@ -269,28 +269,6 @@ class PageTest extends TestCase
]);
}
public function test_old_page_slugs_redirect_to_new_pages()
{
$page = $this->entities->page();
// Need to save twice since revisions are not generated in seeder.
$this->asAdmin()->put($page->getUrl(), [
'name' => 'super test',
'html' => '<p></p>',
]);
$page->refresh();
$pageUrl = $page->getUrl();
$this->put($pageUrl, [
'name' => 'super test page',
'html' => '<p></p>',
]);
$this->get($pageUrl)
->assertRedirect("/books/{$page->book->slug}/page/super-test-page");
}
public function test_page_within_chapter_deletion_returns_to_chapter()
{
$chapter = $this->entities->chapter();

131
tests/Entity/SlugTest.php Normal file
View File

@@ -0,0 +1,131 @@
<?php
namespace Tests\Entity;
use Tests\TestCase;
class SlugTest extends TestCase
{
public function test_slug_multi_byte_url_safe()
{
$book = $this->entities->newBook([
'name' => 'информация',
]);
$this->assertEquals('informaciia', $book->slug);
$book = $this->entities->newBook([
'name' => '¿Qué?',
]);
$this->assertEquals('que', $book->slug);
}
public function test_slug_format()
{
$book = $this->entities->newBook([
'name' => 'PartA / PartB / PartC',
]);
$this->assertEquals('parta-partb-partc', $book->slug);
}
public function test_old_page_slugs_redirect_to_new_pages()
{
$page = $this->entities->page();
// Need to save twice since revisions are not generated in seeder.
$this->asAdmin()->put($page->getUrl(), [
'name' => 'super test',
'html' => '<p></p>',
]);
$page->refresh();
$pageUrl = $page->getUrl();
$this->put($pageUrl, [
'name' => 'super test page',
'html' => '<p></p>',
]);
$this->get($pageUrl)
->assertRedirect("/books/{$page->book->slug}/page/super-test-page");
}
public function test_slugs_recorded_in_history_on_page_update()
{
$page = $this->entities->page();
$this->asAdmin()->put($page->getUrl(), [
'name' => 'new slug',
'html' => '<p></p>',
]);
$oldSlug = $page->slug;
$page->refresh();
$this->assertNotEquals($oldSlug, $page->slug);
$this->assertDatabaseHas('slug_history', [
'sluggable_id' => $page->id,
'sluggable_type' => 'page',
'slug' => $oldSlug,
'parent_slug' => $page->book->slug,
]);
}
public function test_slugs_recorded_in_history_on_chapter_update()
{
$chapter = $this->entities->chapter();
$this->asAdmin()->put($chapter->getUrl(), [
'name' => 'new slug',
]);
$oldSlug = $chapter->slug;
$chapter->refresh();
$this->assertNotEquals($oldSlug, $chapter->slug);
$this->assertDatabaseHas('slug_history', [
'sluggable_id' => $chapter->id,
'sluggable_type' => 'chapter',
'slug' => $oldSlug,
'parent_slug' => $chapter->book->slug,
]);
}
public function test_slugs_recorded_in_history_on_book_update()
{
$book = $this->entities->book();
$this->asAdmin()->put($book->getUrl(), [
'name' => 'new slug',
]);
$oldSlug = $book->slug;
$book->refresh();
$this->assertNotEquals($oldSlug, $book->slug);
$this->assertDatabaseHas('slug_history', [
'sluggable_id' => $book->id,
'sluggable_type' => 'book',
'slug' => $oldSlug,
'parent_slug' => null,
]);
}
public function test_slugs_recorded_in_history_on_shelf_update()
{
$shelf = $this->entities->shelf();
$this->asAdmin()->put($shelf->getUrl(), [
'name' => 'new slug',
]);
$oldSlug = $shelf->slug;
$shelf->refresh();
$this->assertNotEquals($oldSlug, $shelf->slug);
$this->assertDatabaseHas('slug_history', [
'sluggable_id' => $shelf->id,
'sluggable_type' => 'bookshelf',
'slug' => $oldSlug,
'parent_slug' => null,
]);
}
}