X-Git-Url: http://source.bookstackapp.com/bookstack/blobdiff_plain/b9a58859a4ef14b7971ca7d07318e1ac6f276b40..refs/pull/5676/head:/tests/Entity/BookShelfTest.php diff --git a/tests/Entity/BookShelfTest.php b/tests/Entity/BookShelfTest.php index fcbc17ea9..fb9862931 100644 --- a/tests/Entity/BookShelfTest.php +++ b/tests/Entity/BookShelfTest.php @@ -2,59 +2,56 @@ namespace Tests\Entity; -use BookStack\Auth\User; use BookStack\Entities\Models\Book; use BookStack\Entities\Models\Bookshelf; use BookStack\Uploads\Image; +use BookStack\Users\Models\User; use Illuminate\Support\Str; use Tests\TestCase; -use Tests\Uploads\UsesImages; class BookShelfTest extends TestCase { - use UsesImages; - public function test_shelves_shows_in_header_if_have_view_permissions() { - $viewer = $this->getViewer(); + $viewer = $this->users->viewer(); $resp = $this->actingAs($viewer)->get('/'); - $resp->assertElementContains('header', 'Shelves'); + $this->withHtml($resp)->assertElementContains('header', 'Shelves'); $viewer->roles()->delete(); - $this->giveUserPermissions($viewer); + $this->permissions->grantUserRolePermissions($viewer, []); $resp = $this->actingAs($viewer)->get('/'); - $resp->assertElementNotContains('header', 'Shelves'); + $this->withHtml($resp)->assertElementNotContains('header', 'Shelves'); - $this->giveUserPermissions($viewer, ['bookshelf-view-all']); + $this->permissions->grantUserRolePermissions($viewer, ['bookshelf-view-all']); $resp = $this->actingAs($viewer)->get('/'); - $resp->assertElementContains('header', 'Shelves'); + $this->withHtml($resp)->assertElementContains('header', 'Shelves'); $viewer->roles()->delete(); - $this->giveUserPermissions($viewer, ['bookshelf-view-own']); + $this->permissions->grantUserRolePermissions($viewer, ['bookshelf-view-own']); $resp = $this->actingAs($viewer)->get('/'); - $resp->assertElementContains('header', 'Shelves'); + $this->withHtml($resp)->assertElementContains('header', 'Shelves'); } public function test_shelves_shows_in_header_if_have_any_shelve_view_permission() { $user = User::factory()->create(); - $this->giveUserPermissions($user, ['image-create-all']); - $shelf = Bookshelf::first(); + $this->permissions->grantUserRolePermissions($user, ['image-create-all']); + $shelf = $this->entities->shelf(); $userRole = $user->roles()->first(); $resp = $this->actingAs($user)->get('/'); - $resp->assertElementNotContains('header', 'Shelves'); + $this->withHtml($resp)->assertElementNotContains('header', 'Shelves'); - $this->setEntityRestrictions($shelf, ['view'], [$userRole]); + $this->permissions->setEntityPermissions($shelf, ['view'], [$userRole]); $resp = $this->get('/'); - $resp->assertElementContains('header', 'Shelves'); + $this->withHtml($resp)->assertElementContains('header', 'Shelves'); } public function test_shelves_page_contains_create_link() { $resp = $this->asEditor()->get('/shelves'); - $resp->assertElementContains('a', 'New Shelf'); + $this->withHtml($resp)->assertElementContains('a', 'New Shelf'); } public function test_book_not_visible_in_shelf_list_view_if_user_cant_view_shelf() @@ -62,14 +59,14 @@ class BookShelfTest extends TestCase config()->set([ 'setting-defaults.user.bookshelves_view_type' => 'list', ]); - $shelf = Bookshelf::query()->first(); + $shelf = $this->entities->shelf(); $book = $shelf->books()->first(); $resp = $this->asEditor()->get('/shelves'); $resp->assertSee($book->name); $resp->assertSee($book->getUrl()); - $this->setEntityRestrictions($book, []); + $this->permissions->setEntityPermissions($book, []); $resp = $this->asEditor()->get('/shelves'); $resp->assertDontSee($book->name); @@ -80,8 +77,8 @@ class BookShelfTest extends TestCase { $booksToInclude = Book::take(2)->get(); $shelfInfo = [ - 'name' => 'My test book' . Str::random(4), - 'description' => 'Test book description ' . Str::random(10), + 'name' => 'My test shelf' . Str::random(4), + 'description_html' => '
Test book description ' . Str::random(10) . '
', ]; $resp = $this->asEditor()->post('/shelves', array_merge($shelfInfo, [ 'books' => $booksToInclude->implode('id', ','), @@ -93,15 +90,15 @@ class BookShelfTest extends TestCase ], ])); $resp->assertRedirect(); - $editorId = $this->getEditor()->id; + $editorId = $this->users->editor()->id; $this->assertDatabaseHas('bookshelves', array_merge($shelfInfo, ['created_by' => $editorId, 'updated_by' => $editorId])); $shelf = Bookshelf::where('name', '=', $shelfInfo['name'])->first(); $shelfPage = $this->get($shelf->getUrl()); $shelfPage->assertSee($shelfInfo['name']); - $shelfPage->assertSee($shelfInfo['description']); - $shelfPage->assertElementContains('.tag-item', 'Test Category'); - $shelfPage->assertElementContains('.tag-item', 'Test Tag Value'); + $shelfPage->assertSee($shelfInfo['description_html'], false); + $this->withHtml($shelfPage)->assertElementContains('.tag-item', 'Test Category'); + $this->withHtml($shelfPage)->assertElementContains('.tag-item', 'Test Tag Value'); $this->assertDatabaseHas('bookshelves_books', ['bookshelf_id' => $shelf->id, 'book_id' => $booksToInclude[0]->id]); $this->assertDatabaseHas('bookshelves_books', ['bookshelf_id' => $shelf->id, 'book_id' => $booksToInclude[1]->id]); @@ -110,11 +107,11 @@ class BookShelfTest extends TestCase public function test_shelves_create_sets_cover_image() { $shelfInfo = [ - 'name' => 'My test book' . Str::random(4), - 'description' => 'Test book description ' . Str::random(10), + 'name' => 'My test shelf' . Str::random(4), + 'description_html' => 'Test book description ' . Str::random(10) . '
', ]; - $imageFile = $this->getTestImage('shelf-test.png'); + $imageFile = $this->files->uploadedImage('shelf-test.png'); $resp = $this->asEditor()->call('POST', '/shelves', $shelfInfo, [], ['image' => $imageFile]); $resp->assertRedirect(); @@ -125,11 +122,12 @@ class BookShelfTest extends TestCase 'image_id' => $lastImage->id, ]); $this->assertEquals($lastImage->id, $shelf->cover->id); + $this->assertEquals('cover_bookshelf', $lastImage->type); } public function test_shelf_view() { - $shelf = Bookshelf::first(); + $shelf = $this->entities->shelf(); $resp = $this->asEditor()->get($shelf->getUrl()); $resp->assertStatus(200); $resp->assertSeeText($shelf->name); @@ -142,16 +140,16 @@ class BookShelfTest extends TestCase public function test_shelf_view_shows_action_buttons() { - $shelf = Bookshelf::first(); + $shelf = $this->entities->shelf(); $resp = $this->asAdmin()->get($shelf->getUrl()); $resp->assertSee($shelf->getUrl('/create-book')); $resp->assertSee($shelf->getUrl('/edit')); $resp->assertSee($shelf->getUrl('/permissions')); $resp->assertSee($shelf->getUrl('/delete')); - $resp->assertElementContains('a', 'New Book'); - $resp->assertElementContains('a', 'Edit'); - $resp->assertElementContains('a', 'Permissions'); - $resp->assertElementContains('a', 'Delete'); + $this->withHtml($resp)->assertElementContains('a', 'New Book'); + $this->withHtml($resp)->assertElementContains('a', 'Edit'); + $this->withHtml($resp)->assertElementContains('a', 'Permissions'); + $this->withHtml($resp)->assertElementContains('a', 'Delete'); $resp = $this->asEditor()->get($shelf->getUrl()); $resp->assertDontSee($shelf->getUrl('/permissions')); @@ -159,10 +157,10 @@ class BookShelfTest extends TestCase public function test_shelf_view_has_sort_control_that_defaults_to_default() { - $shelf = Bookshelf::query()->first(); + $shelf = $this->entities->shelf(); $resp = $this->asAdmin()->get($shelf->getUrl()); - $resp->assertElementExists('form[action$="change-sort/shelf_books"]'); - $resp->assertElementContains('form[action$="change-sort/shelf_books"] [aria-haspopup="true"]', 'Default'); + $this->withHtml($resp)->assertElementExists('form[action$="change-sort/shelf_books"]'); + $this->withHtml($resp)->assertElementContains('form[action$="change-sort/shelf_books"] [aria-haspopup="true"]', 'Default'); } public function test_shelf_view_sort_takes_action() @@ -176,38 +174,63 @@ class BookShelfTest extends TestCase // Set book ordering $this->asAdmin()->put($shelf->getUrl(), [ 'books' => $books->implode('id', ','), - 'tags' => [], 'description' => 'abc', 'name' => 'abc', + 'tags' => [], 'description_html' => 'abc', 'name' => 'abc', ]); $this->assertEquals(3, $shelf->books()->count()); $shelf->refresh(); $resp = $this->asEditor()->get($shelf->getUrl()); - $resp->assertElementContains('.book-content a.grid-card', $books[0]->name, 1); - $resp->assertElementNotContains('.book-content a.grid-card', $books[0]->name, 3); + $this->withHtml($resp)->assertElementContains('.book-content a.grid-card:nth-child(1)', $books[0]->name); + $this->withHtml($resp)->assertElementNotContains('.book-content a.grid-card:nth-child(3)', $books[0]->name); - setting()->putUser($this->getEditor(), 'shelf_books_sort_order', 'desc'); + setting()->putUser($this->users->editor(), 'shelf_books_sort_order', 'desc'); $resp = $this->asEditor()->get($shelf->getUrl()); - $resp->assertElementNotContains('.book-content a.grid-card', $books[0]->name, 1); - $resp->assertElementContains('.book-content a.grid-card', $books[0]->name, 3); + $this->withHtml($resp)->assertElementNotContains('.book-content a.grid-card:nth-child(1)', $books[0]->name); + $this->withHtml($resp)->assertElementContains('.book-content a.grid-card:nth-child(3)', $books[0]->name); - setting()->putUser($this->getEditor(), 'shelf_books_sort_order', 'desc'); - setting()->putUser($this->getEditor(), 'shelf_books_sort', 'name'); + setting()->putUser($this->users->editor(), 'shelf_books_sort_order', 'desc'); + setting()->putUser($this->users->editor(), 'shelf_books_sort', 'name'); $resp = $this->asEditor()->get($shelf->getUrl()); - $resp->assertElementContains('.book-content a.grid-card', 'hdgfgdfg', 1); - $resp->assertElementContains('.book-content a.grid-card', 'bsfsdfsdfsd', 2); - $resp->assertElementContains('.book-content a.grid-card', 'adsfsdfsdfsd', 3); + $this->withHtml($resp)->assertElementContains('.book-content a.grid-card:nth-child(1)', 'hdgfgdfg'); + $this->withHtml($resp)->assertElementContains('.book-content a.grid-card:nth-child(2)', 'bsfsdfsdfsd'); + $this->withHtml($resp)->assertElementContains('.book-content a.grid-card:nth-child(3)', 'adsfsdfsdfsd'); + } + + public function test_shelf_view_sorts_by_name_case_insensitively() + { + $shelf = Bookshelf::query()->whereHas('books')->with('books')->first(); + $books = Book::query()->take(3)->get(['id', 'name']); + $books[0]->fill(['name' => 'Book Ab'])->save(); + $books[1]->fill(['name' => 'Book ac'])->save(); + $books[2]->fill(['name' => 'Book AD'])->save(); + + // Set book ordering + $this->asAdmin()->put($shelf->getUrl(), [ + 'books' => $books->implode('id', ','), + 'tags' => [], 'description_html' => 'abc', 'name' => 'abc', + ]); + $this->assertEquals(3, $shelf->books()->count()); + $shelf->refresh(); + + setting()->putUser($this->users->editor(), 'shelf_books_sort', 'name'); + setting()->putUser($this->users->editor(), 'shelf_books_sort_order', 'asc'); + $html = $this->withHtml($this->asEditor()->get($shelf->getUrl())); + + $html->assertElementContains('.book-content a.grid-card:nth-child(1)', 'Book Ab'); + $html->assertElementContains('.book-content a.grid-card:nth-child(2)', 'Book ac'); + $html->assertElementContains('.book-content a.grid-card:nth-child(3)', 'Book AD'); } public function test_shelf_edit() { - $shelf = Bookshelf::first(); + $shelf = $this->entities->shelf(); $resp = $this->asEditor()->get($shelf->getUrl('/edit')); - $resp->assertSeeText('Edit Bookshelf'); + $resp->assertSeeText('Edit Shelf'); $booksToInclude = Book::take(2)->get(); $shelfInfo = [ - 'name' => 'My test book' . Str::random(4), - 'description' => 'Test book description ' . Str::random(10), + 'name' => 'My test shelf' . Str::random(4), + 'description_html' => 'Test book description ' . Str::random(10) . '
', ]; $resp = $this->asEditor()->put($shelf->getUrl(), array_merge($shelfInfo, [ @@ -223,14 +246,14 @@ class BookShelfTest extends TestCase $resp->assertRedirect($shelf->getUrl()); $this->assertSessionHas('success'); - $editorId = $this->getEditor()->id; + $editorId = $this->users->editor()->id; $this->assertDatabaseHas('bookshelves', array_merge($shelfInfo, ['id' => $shelf->id, 'created_by' => $editorId, 'updated_by' => $editorId])); $shelfPage = $this->get($shelf->getUrl()); $shelfPage->assertSee($shelfInfo['name']); - $shelfPage->assertSee($shelfInfo['description']); - $shelfPage->assertElementContains('.tag-item', 'Test Category'); - $shelfPage->assertElementContains('.tag-item', 'Test Tag Value'); + $shelfPage->assertSee($shelfInfo['description_html'], false); + $this->withHtml($shelfPage)->assertElementContains('.tag-item', 'Test Category'); + $this->withHtml($shelfPage)->assertElementContains('.tag-item', 'Test Tag Value'); $this->assertDatabaseHas('bookshelves_books', ['bookshelf_id' => $shelf->id, 'book_id' => $booksToInclude[0]->id]); $this->assertDatabaseHas('bookshelves_books', ['bookshelf_id' => $shelf->id, 'book_id' => $booksToInclude[1]->id]); @@ -238,7 +261,7 @@ class BookShelfTest extends TestCase public function test_shelf_create_new_book() { - $shelf = Bookshelf::first(); + $shelf = $this->entities->shelf(); $resp = $this->asEditor()->get($shelf->getUrl('/create-book')); $resp->assertSee('Create New Book'); @@ -247,8 +270,8 @@ class BookShelfTest extends TestCase $testName = 'Test Book in Shelf Name'; $createBookResp = $this->asEditor()->post($shelf->getUrl('/create-book'), [ - 'name' => $testName, - 'description' => 'Book in shelf description', + 'name' => $testName, + 'description_html' => 'Book in shelf description', ]); $createBookResp->assertRedirect(); @@ -269,7 +292,7 @@ class BookShelfTest extends TestCase $bookCount = $shelf->books()->count(); $deleteViewReq = $this->asEditor()->get($shelf->getUrl('/delete')); - $deleteViewReq->assertSeeText('Are you sure you want to delete this bookshelf?'); + $deleteViewReq->assertSeeText('Are you sure you want to delete this shelf?'); $deleteReq = $this->delete($shelf->getUrl()); $deleteReq->assertRedirect(url('/shelves')); @@ -282,71 +305,75 @@ class BookShelfTest extends TestCase $this->assertTrue($shelf->deletions()->count() === 1); $redirectReq = $this->get($deleteReq->baseResponse->headers->get('location')); - $redirectReq->assertNotificationContains('Bookshelf Successfully Deleted'); + $this->assertNotificationContains($redirectReq, 'Shelf Successfully Deleted'); } public function test_shelf_copy_permissions() { - $shelf = Bookshelf::first(); + $shelf = $this->entities->shelf(); $resp = $this->asAdmin()->get($shelf->getUrl('/permissions')); $resp->assertSeeText('Copy Permissions'); $resp->assertSee("action=\"{$shelf->getUrl('/copy-permissions')}\"", false); $child = $shelf->books()->first(); - $editorRole = $this->getEditor()->roles()->first(); - $this->assertFalse(boolval($child->restricted), 'Child book should not be restricted by default'); + $editorRole = $this->users->editor()->roles()->first(); + $this->assertFalse($child->hasPermissions(), 'Child book should not be restricted by default'); $this->assertTrue($child->permissions()->count() === 0, 'Child book should have no permissions by default'); - $this->setEntityRestrictions($shelf, ['view', 'update'], [$editorRole]); + $this->permissions->setEntityPermissions($shelf, ['view', 'update'], [$editorRole]); $resp = $this->post($shelf->getUrl('/copy-permissions')); $child = $shelf->books()->first(); $resp->assertRedirect($shelf->getUrl()); - $this->assertTrue(boolval($child->restricted), 'Child book should now be restricted'); + $this->assertTrue($child->hasPermissions(), 'Child book should now be restricted'); $this->assertTrue($child->permissions()->count() === 2, 'Child book should have copied permissions'); - $this->assertDatabaseHas('entity_permissions', ['restrictable_id' => $child->id, 'action' => 'view', 'role_id' => $editorRole->id]); - $this->assertDatabaseHas('entity_permissions', ['restrictable_id' => $child->id, 'action' => 'update', 'role_id' => $editorRole->id]); + $this->assertDatabaseHas('entity_permissions', [ + 'entity_type' => 'book', + 'entity_id' => $child->id, + 'role_id' => $editorRole->id, + 'view' => true, 'update' => true, 'create' => false, 'delete' => false, + ]); } public function test_permission_page_has_a_warning_about_no_cascading() { - $shelf = Bookshelf::first(); + $shelf = $this->entities->shelf(); $resp = $this->asAdmin()->get($shelf->getUrl('/permissions')); - $resp->assertSeeText('Permissions on bookshelves do not automatically cascade to contained books.'); + $resp->assertSeeText('Permissions on shelves do not automatically cascade to contained books.'); } public function test_bookshelves_show_in_breadcrumbs_if_in_context() { - $shelf = Bookshelf::first(); + $shelf = $this->entities->shelf(); $shelfBook = $shelf->books()->first(); $shelfPage = $shelfBook->pages()->first(); $this->asAdmin(); $bookVisit = $this->get($shelfBook->getUrl()); - $bookVisit->assertElementNotContains('.breadcrumbs', 'Shelves'); - $bookVisit->assertElementNotContains('.breadcrumbs', $shelf->getShortName()); + $this->withHtml($bookVisit)->assertElementNotContains('.breadcrumbs', 'Shelves'); + $this->withHtml($bookVisit)->assertElementNotContains('.breadcrumbs', $shelf->getShortName()); $this->get($shelf->getUrl()); $bookVisit = $this->get($shelfBook->getUrl()); - $bookVisit->assertElementContains('.breadcrumbs', 'Shelves'); - $bookVisit->assertElementContains('.breadcrumbs', $shelf->getShortName()); + $this->withHtml($bookVisit)->assertElementContains('.breadcrumbs', 'Shelves'); + $this->withHtml($bookVisit)->assertElementContains('.breadcrumbs', $shelf->getShortName()); $pageVisit = $this->get($shelfPage->getUrl()); - $pageVisit->assertElementContains('.breadcrumbs', 'Shelves'); - $pageVisit->assertElementContains('.breadcrumbs', $shelf->getShortName()); + $this->withHtml($pageVisit)->assertElementContains('.breadcrumbs', 'Shelves'); + $this->withHtml($pageVisit)->assertElementContains('.breadcrumbs', $shelf->getShortName()); $this->get('/books'); $pageVisit = $this->get($shelfPage->getUrl()); - $pageVisit->assertElementNotContains('.breadcrumbs', 'Shelves'); - $pageVisit->assertElementNotContains('.breadcrumbs', $shelf->getShortName()); + $this->withHtml($pageVisit)->assertElementNotContains('.breadcrumbs', 'Shelves'); + $this->withHtml($pageVisit)->assertElementNotContains('.breadcrumbs', $shelf->getShortName()); } public function test_bookshelves_show_on_book() { // Create shelf $shelfInfo = [ - 'name' => 'My test shelf' . Str::random(4), - 'description' => 'Test shelf description ' . Str::random(10), + 'name' => 'My test shelf' . Str::random(4), + 'description_html' => 'Test shelf description ' . Str::random(10) . '
', ]; $this->asEditor()->post('/shelves', $shelfInfo); @@ -354,14 +381,14 @@ class BookShelfTest extends TestCase // Create book and add to shelf $this->asEditor()->post($shelf->getUrl('/create-book'), [ - 'name' => 'Test book name', - 'description' => 'Book in shelf description', + 'name' => 'Test book name', + 'description_html' => 'Book in shelf description
', ]); $newBook = Book::query()->orderBy('id', 'desc')->first(); $resp = $this->asEditor()->get($newBook->getUrl()); - $resp->assertElementContains('.tri-layout-left-contents', $shelfInfo['name']); + $this->withHtml($resp)->assertElementContains('.tri-layout-left-contents', $shelfInfo['name']); // Remove shelf $this->delete($shelf->getUrl()); @@ -372,9 +399,19 @@ class BookShelfTest extends TestCase public function test_cancel_on_child_book_creation_returns_to_original_shelf() { - /** @var Bookshelf $shelf */ - $shelf = Bookshelf::query()->first(); + $shelf = $this->entities->shelf(); $resp = $this->asEditor()->get($shelf->getUrl('/create-book')); - $resp->assertElementContains('form a[href="' . $shelf->getUrl() . '"]', 'Cancel'); + $this->withHtml($resp)->assertElementContains('form a[href="' . $shelf->getUrl() . '"]', 'Cancel'); + } + + public function test_show_view_displays_description_if_no_description_html_set() + { + $shelf = $this->entities->shelf(); + $shelf->description_html = ''; + $shelf->description = "My great\ndescription\n\nwith newlines"; + $shelf->save(); + + $resp = $this->asEditor()->get($shelf->getUrl()); + $resp->assertSee("My great
\ndescription
\n
\nwith newlines