| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  | <?php | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Tests\Api; | 
					
						
							| 
									
										
										
										
											2020-04-10 22:19:18 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-22 08:17:45 +08:00
										 |  |  | use BookStack\Entities\Models\Book; | 
					
						
							|  |  |  | use BookStack\Entities\Models\Bookshelf; | 
					
						
							| 
									
										
										
										
											2022-04-05 00:24:05 +08:00
										 |  |  | use Carbon\Carbon; | 
					
						
							|  |  |  | use Illuminate\Support\Facades\DB; | 
					
						
							| 
									
										
										
										
											2020-04-10 22:19:18 +08:00
										 |  |  | use Tests\TestCase; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ShelvesApiTest extends TestCase | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     use TestsApi; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-05 00:24:05 +08:00
										 |  |  |     protected string $baseEndpoint = '/api/shelves'; | 
					
						
							| 
									
										
										
										
											2020-04-10 22:19:18 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     public function test_index_endpoint_returns_expected_shelf() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->actingAsApiEditor(); | 
					
						
							|  |  |  |         $firstBookshelf = Bookshelf::query()->orderBy('id', 'asc')->first(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $resp = $this->getJson($this->baseEndpoint . '?count=1&sort=+id'); | 
					
						
							|  |  |  |         $resp->assertJson(['data' => [ | 
					
						
							|  |  |  |             [ | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |                 'id'   => $firstBookshelf->id, | 
					
						
							| 
									
										
										
										
											2020-04-10 22:19:18 +08:00
										 |  |  |                 'name' => $firstBookshelf->name, | 
					
						
							|  |  |  |                 'slug' => $firstBookshelf->slug, | 
					
						
							| 
									
										
										
										
											2024-02-11 23:42:37 +08:00
										 |  |  |                 'owned_by' => $firstBookshelf->owned_by, | 
					
						
							|  |  |  |                 'created_by' => $firstBookshelf->created_by, | 
					
						
							|  |  |  |                 'updated_by' => $firstBookshelf->updated_by, | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |             ], | 
					
						
							| 
									
										
										
										
											2020-04-10 22:19:18 +08:00
										 |  |  |         ]]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public function test_create_endpoint() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->actingAsApiEditor(); | 
					
						
							|  |  |  |         $books = Book::query()->take(2)->get(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $details = [ | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |             'name'        => 'My API shelf', | 
					
						
							| 
									
										
										
										
											2020-04-10 22:19:18 +08:00
										 |  |  |             'description' => 'A shelf created via the API', | 
					
						
							|  |  |  |         ]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $resp = $this->postJson($this->baseEndpoint, array_merge($details, ['books' => [$books[0]->id, $books[1]->id]])); | 
					
						
							|  |  |  |         $resp->assertStatus(200); | 
					
						
							|  |  |  |         $newItem = Bookshelf::query()->orderByDesc('id')->where('name', '=', $details['name'])->first(); | 
					
						
							| 
									
										
										
										
											2023-12-22 21:17:23 +08:00
										 |  |  |         $resp->assertJson(array_merge($details, [ | 
					
						
							|  |  |  |             'id' => $newItem->id, | 
					
						
							|  |  |  |             'slug' => $newItem->slug, | 
					
						
							|  |  |  |             'description_html' => '<p>A shelf created via the API</p>', | 
					
						
							|  |  |  |         ])); | 
					
						
							| 
									
										
										
										
											2020-04-10 22:19:18 +08:00
										 |  |  |         $this->assertActivityExists('bookshelf_create', $newItem); | 
					
						
							|  |  |  |         foreach ($books as $index => $book) { | 
					
						
							|  |  |  |             $this->assertDatabaseHas('bookshelves_books', [ | 
					
						
							|  |  |  |                 'bookshelf_id' => $newItem->id, | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |                 'book_id'      => $book->id, | 
					
						
							|  |  |  |                 'order'        => $index, | 
					
						
							| 
									
										
										
										
											2020-04-10 22:19:18 +08:00
										 |  |  |             ]); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-22 21:17:23 +08:00
										 |  |  |     public function test_create_endpoint_with_html() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->actingAsApiEditor(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $details = [ | 
					
						
							|  |  |  |             'name'             => 'My API shelf', | 
					
						
							|  |  |  |             'description_html' => '<p>A <strong>shelf</strong> created via the API</p>', | 
					
						
							|  |  |  |         ]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $resp = $this->postJson($this->baseEndpoint, $details); | 
					
						
							|  |  |  |         $resp->assertStatus(200); | 
					
						
							|  |  |  |         $newItem = Bookshelf::query()->orderByDesc('id')->where('name', '=', $details['name'])->first(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $expectedDetails = array_merge($details, [ | 
					
						
							|  |  |  |             'id'          => $newItem->id, | 
					
						
							|  |  |  |             'description' => 'A shelf created via the API', | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $resp->assertJson($expectedDetails); | 
					
						
							|  |  |  |         $this->assertDatabaseHas('bookshelves', $expectedDetails); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-10 22:19:18 +08:00
										 |  |  |     public function test_shelf_name_needed_to_create() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->actingAsApiEditor(); | 
					
						
							|  |  |  |         $details = [ | 
					
						
							|  |  |  |             'description' => 'A shelf created via the API', | 
					
						
							|  |  |  |         ]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $resp = $this->postJson($this->baseEndpoint, $details); | 
					
						
							|  |  |  |         $resp->assertStatus(422); | 
					
						
							|  |  |  |         $resp->assertJson([ | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |             'error' => [ | 
					
						
							|  |  |  |                 'message'    => 'The given data was invalid.', | 
					
						
							|  |  |  |                 'validation' => [ | 
					
						
							|  |  |  |                     'name' => ['The name field is required.'], | 
					
						
							| 
									
										
										
										
											2020-04-10 22:19:18 +08:00
										 |  |  |                 ], | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |                 'code' => 422, | 
					
						
							| 
									
										
										
										
											2020-04-10 22:19:18 +08:00
										 |  |  |             ], | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public function test_read_endpoint() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->actingAsApiEditor(); | 
					
						
							|  |  |  |         $shelf = Bookshelf::visible()->first(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $resp = $this->getJson($this->baseEndpoint . "/{$shelf->id}"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $resp->assertStatus(200); | 
					
						
							|  |  |  |         $resp->assertJson([ | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |             'id'         => $shelf->id, | 
					
						
							|  |  |  |             'slug'       => $shelf->slug, | 
					
						
							| 
									
										
										
										
											2020-04-10 22:19:18 +08:00
										 |  |  |             'created_by' => [ | 
					
						
							|  |  |  |                 'name' => $shelf->createdBy->name, | 
					
						
							|  |  |  |             ], | 
					
						
							|  |  |  |             'updated_by' => [ | 
					
						
							|  |  |  |                 'name' => $shelf->createdBy->name, | 
					
						
							| 
									
										
										
										
											2021-01-04 06:29:58 +08:00
										 |  |  |             ], | 
					
						
							|  |  |  |             'owned_by' => [ | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |                 'name' => $shelf->ownedBy->name, | 
					
						
							| 
									
										
										
										
											2021-01-04 06:29:58 +08:00
										 |  |  |             ], | 
					
						
							| 
									
										
										
										
											2020-04-10 22:19:18 +08:00
										 |  |  |         ]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public function test_update_endpoint() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->actingAsApiEditor(); | 
					
						
							|  |  |  |         $shelf = Bookshelf::visible()->first(); | 
					
						
							|  |  |  |         $details = [ | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |             'name'        => 'My updated API shelf', | 
					
						
							| 
									
										
										
										
											2023-12-22 21:17:23 +08:00
										 |  |  |             'description' => 'A shelf updated via the API', | 
					
						
							| 
									
										
										
										
											2020-04-10 22:19:18 +08:00
										 |  |  |         ]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $resp = $this->putJson($this->baseEndpoint . "/{$shelf->id}", $details); | 
					
						
							|  |  |  |         $shelf->refresh(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $resp->assertStatus(200); | 
					
						
							| 
									
										
										
										
											2023-12-22 21:17:23 +08:00
										 |  |  |         $resp->assertJson(array_merge($details, [ | 
					
						
							|  |  |  |             'id' => $shelf->id, | 
					
						
							|  |  |  |             'slug' => $shelf->slug, | 
					
						
							|  |  |  |             'description_html' => '<p>A shelf updated via the API</p>', | 
					
						
							|  |  |  |         ])); | 
					
						
							| 
									
										
										
										
											2020-04-10 22:19:18 +08:00
										 |  |  |         $this->assertActivityExists('bookshelf_update', $shelf); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-22 21:17:23 +08:00
										 |  |  |     public function test_update_endpoint_with_html() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->actingAsApiEditor(); | 
					
						
							|  |  |  |         $shelf = Bookshelf::visible()->first(); | 
					
						
							|  |  |  |         $details = [ | 
					
						
							|  |  |  |             'name'             => 'My updated API shelf', | 
					
						
							|  |  |  |             'description_html' => '<p>A shelf <em>updated</em> via the API</p>', | 
					
						
							|  |  |  |         ]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $resp = $this->putJson($this->baseEndpoint . "/{$shelf->id}", $details); | 
					
						
							|  |  |  |         $resp->assertStatus(200); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $this->assertDatabaseHas('bookshelves', array_merge($details, ['id' => $shelf->id, 'description' => 'A shelf updated via the API'])); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-05 00:24:05 +08:00
										 |  |  |     public function test_update_increments_updated_date_if_only_tags_are_sent() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->actingAsApiEditor(); | 
					
						
							|  |  |  |         $shelf = Bookshelf::visible()->first(); | 
					
						
							|  |  |  |         DB::table('bookshelves')->where('id', '=', $shelf->id)->update(['updated_at' => Carbon::now()->subWeek()]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $details = [ | 
					
						
							| 
									
										
										
										
											2022-04-25 01:22:40 +08:00
										 |  |  |             'tags' => [['name' => 'Category', 'value' => 'Testing']], | 
					
						
							| 
									
										
										
										
											2022-04-05 00:24:05 +08:00
										 |  |  |         ]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $this->putJson($this->baseEndpoint . "/{$shelf->id}", $details); | 
					
						
							|  |  |  |         $shelf->refresh(); | 
					
						
							|  |  |  |         $this->assertGreaterThan(Carbon::now()->subDay()->unix(), $shelf->updated_at->unix()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-10 22:19:18 +08:00
										 |  |  |     public function test_update_only_assigns_books_if_param_provided() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->actingAsApiEditor(); | 
					
						
							|  |  |  |         $shelf = Bookshelf::visible()->first(); | 
					
						
							|  |  |  |         $this->assertTrue($shelf->books()->count() > 0); | 
					
						
							|  |  |  |         $details = [ | 
					
						
							|  |  |  |             'name' => 'My updated API shelf', | 
					
						
							|  |  |  |         ]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $resp = $this->putJson($this->baseEndpoint . "/{$shelf->id}", $details); | 
					
						
							|  |  |  |         $resp->assertStatus(200); | 
					
						
							|  |  |  |         $this->assertTrue($shelf->books()->count() > 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $resp = $this->putJson($this->baseEndpoint . "/{$shelf->id}", ['books' => []]); | 
					
						
							|  |  |  |         $resp->assertStatus(200); | 
					
						
							|  |  |  |         $this->assertTrue($shelf->books()->count() === 0); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-20 00:26:23 +08:00
										 |  |  |     public function test_update_cover_image_control() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->actingAsApiEditor(); | 
					
						
							|  |  |  |         /** @var Book $shelf */ | 
					
						
							|  |  |  |         $shelf = Bookshelf::visible()->first(); | 
					
						
							|  |  |  |         $this->assertNull($shelf->cover); | 
					
						
							| 
									
										
										
										
											2023-02-08 22:39:13 +08:00
										 |  |  |         $file = $this->files->uploadedImage('image.png'); | 
					
						
							| 
									
										
										
										
											2022-06-20 00:26:23 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Ensure cover image can be set via API
 | 
					
						
							|  |  |  |         $resp = $this->call('PUT', $this->baseEndpoint . "/{$shelf->id}", [ | 
					
						
							|  |  |  |             'name'  => 'My updated API shelf with image', | 
					
						
							|  |  |  |         ], [], ['image' => $file]); | 
					
						
							|  |  |  |         $shelf->refresh(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $resp->assertStatus(200); | 
					
						
							|  |  |  |         $this->assertNotNull($shelf->cover); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Ensure further updates without image do not clear cover image
 | 
					
						
							|  |  |  |         $resp = $this->put($this->baseEndpoint . "/{$shelf->id}", [ | 
					
						
							| 
									
										
										
										
											2022-06-20 01:14:53 +08:00
										 |  |  |             'name' => 'My updated shelf again', | 
					
						
							| 
									
										
										
										
											2022-06-20 00:26:23 +08:00
										 |  |  |         ]); | 
					
						
							|  |  |  |         $shelf->refresh(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $resp->assertStatus(200); | 
					
						
							|  |  |  |         $this->assertNotNull($shelf->cover); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Ensure update with null image property clears image
 | 
					
						
							|  |  |  |         $resp = $this->put($this->baseEndpoint . "/{$shelf->id}", [ | 
					
						
							|  |  |  |             'image' => null, | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  |         $shelf->refresh(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $resp->assertStatus(200); | 
					
						
							|  |  |  |         $this->assertNull($shelf->cover); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-10 22:19:18 +08:00
										 |  |  |     public function test_delete_endpoint() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->actingAsApiEditor(); | 
					
						
							|  |  |  |         $shelf = Bookshelf::visible()->first(); | 
					
						
							|  |  |  |         $resp = $this->deleteJson($this->baseEndpoint . "/{$shelf->id}"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $resp->assertStatus(204); | 
					
						
							|  |  |  |         $this->assertActivityExists('bookshelf_delete'); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  | } |