| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  | <?php | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace BookStack\Http\Controllers\Api; | 
					
						
							| 
									
										
										
										
											2019-12-28 22:58:07 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-29 22:05:57 +08:00
										 |  |  | use BookStack\Api\ApiEntityListFormatter; | 
					
						
							| 
									
										
										
										
											2020-11-22 08:17:45 +08:00
										 |  |  | use BookStack\Entities\Models\Book; | 
					
						
							| 
									
										
										
										
											2022-09-29 22:05:57 +08:00
										 |  |  | use BookStack\Entities\Models\Chapter; | 
					
						
							|  |  |  | use BookStack\Entities\Models\Entity; | 
					
						
							| 
									
										
										
										
											2020-01-12 22:45:54 +08:00
										 |  |  | use BookStack\Entities\Repos\BookRepo; | 
					
						
							| 
									
										
										
										
											2022-09-29 22:05:57 +08:00
										 |  |  | use BookStack\Entities\Tools\BookContents; | 
					
						
							| 
									
										
										
										
											2020-01-12 22:45:54 +08:00
										 |  |  | use Illuminate\Http\Request; | 
					
						
							| 
									
										
										
										
											2020-01-16 04:18:02 +08:00
										 |  |  | use Illuminate\Validation\ValidationException; | 
					
						
							| 
									
										
										
										
											2019-12-28 22:58:07 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-23 07:28:41 +08:00
										 |  |  | class BookApiController extends ApiController | 
					
						
							| 
									
										
										
										
											2019-12-28 22:58:07 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-09-29 22:05:57 +08:00
										 |  |  |     protected BookRepo $bookRepo; | 
					
						
							| 
									
										
										
										
											2020-01-12 22:45:54 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     public function __construct(BookRepo $bookRepo) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->bookRepo = $bookRepo; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-28 22:58:07 +08:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Get a listing of books visible to the user. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2020-01-18 22:03:11 +08:00
										 |  |  |     public function list() | 
					
						
							| 
									
										
										
										
											2019-12-28 22:58:07 +08:00
										 |  |  |     { | 
					
						
							|  |  |  |         $books = Book::visible(); | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-28 22:58:07 +08:00
										 |  |  |         return $this->apiListingResponse($books, [ | 
					
						
							| 
									
										
										
										
											2022-06-07 21:27:45 +08:00
										 |  |  |             'id', 'name', 'slug', 'description', 'created_at', 'updated_at', 'created_by', 'updated_by', 'owned_by', | 
					
						
							| 
									
										
										
										
											2019-12-28 22:58:07 +08:00
										 |  |  |         ]); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-01-02 00:33:47 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-12 22:45:54 +08:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2020-01-16 04:18:02 +08:00
										 |  |  |      * Create a new book in the system. | 
					
						
							| 
									
										
										
										
											2022-06-20 00:26:23 +08:00
										 |  |  |      * The cover image of a book can be set by sending a file via an 'image' property within a 'multipart/form-data' request. | 
					
						
							|  |  |  |      * If the 'image' property is null then the book cover image will be removed. | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2020-01-16 04:18:02 +08:00
										 |  |  |      * @throws ValidationException | 
					
						
							| 
									
										
										
										
											2020-01-12 22:45:54 +08:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function create(Request $request) | 
					
						
							| 
									
										
										
										
											2020-01-02 00:33:47 +08:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-01-12 22:45:54 +08:00
										 |  |  |         $this->checkPermission('book-create-all'); | 
					
						
							| 
									
										
										
										
											2022-06-20 00:26:23 +08:00
										 |  |  |         $requestData = $this->validate($request, $this->rules()['create']); | 
					
						
							| 
									
										
										
										
											2020-01-12 22:45:54 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $book = $this->bookRepo->create($requestData); | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-12 22:45:54 +08:00
										 |  |  |         return response()->json($book); | 
					
						
							| 
									
										
										
										
											2020-01-02 00:33:47 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-12 22:45:54 +08:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * View the details of a single book. | 
					
						
							| 
									
										
										
										
											2022-09-29 22:05:57 +08:00
										 |  |  |      * The response data will contain 'content' property listing the chapter and pages directly within, in | 
					
						
							|  |  |  |      * the same structure as you'd see within the BookStack interface when viewing a book. Top-level | 
					
						
							|  |  |  |      * contents will have a 'type' property to distinguish between pages & chapters. | 
					
						
							| 
									
										
										
										
											2020-01-12 22:45:54 +08:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function read(string $id) | 
					
						
							| 
									
										
										
										
											2020-01-02 00:33:47 +08:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-01-04 06:29:58 +08:00
										 |  |  |         $book = Book::visible()->with(['tags', 'cover', 'createdBy', 'updatedBy', 'ownedBy'])->findOrFail($id); | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-29 22:05:57 +08:00
										 |  |  |         $contents = (new BookContents($book))->getTree(true, false)->all(); | 
					
						
							|  |  |  |         $contentsApiData = (new ApiEntityListFormatter($contents)) | 
					
						
							|  |  |  |             ->withType() | 
					
						
							|  |  |  |             ->withField('pages', function (Entity $entity) { | 
					
						
							|  |  |  |                 if ($entity instanceof Chapter) { | 
					
						
							|  |  |  |                     return (new ApiEntityListFormatter($entity->pages->all()))->format(); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 return null; | 
					
						
							|  |  |  |             })->format(); | 
					
						
							|  |  |  |         $book->setAttribute('contents', $contentsApiData); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-12 22:45:54 +08:00
										 |  |  |         return response()->json($book); | 
					
						
							| 
									
										
										
										
											2020-01-02 00:33:47 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-12 22:45:54 +08:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Update the details of a single book. | 
					
						
							| 
									
										
										
										
											2022-06-20 00:26:23 +08:00
										 |  |  |      * The cover image of a book can be set by sending a file via an 'image' property within a 'multipart/form-data' request. | 
					
						
							|  |  |  |      * If the 'image' property is null then the book cover image will be removed. | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2020-01-16 04:18:02 +08:00
										 |  |  |      * @throws ValidationException | 
					
						
							| 
									
										
										
										
											2020-01-12 22:45:54 +08:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function update(Request $request, string $id) | 
					
						
							| 
									
										
										
										
											2020-01-02 00:33:47 +08:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-01-12 22:45:54 +08:00
										 |  |  |         $book = Book::visible()->findOrFail($id); | 
					
						
							|  |  |  |         $this->checkOwnablePermission('book-update', $book); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-20 00:26:23 +08:00
										 |  |  |         $requestData = $this->validate($request, $this->rules()['update']); | 
					
						
							| 
									
										
										
										
											2020-01-12 22:45:54 +08:00
										 |  |  |         $book = $this->bookRepo->update($book, $requestData); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return response()->json($book); | 
					
						
							| 
									
										
										
										
											2020-01-02 00:33:47 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-12 22:45:54 +08:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2020-11-28 23:21:54 +08:00
										 |  |  |      * Delete a single book. | 
					
						
							|  |  |  |      * This will typically send the book to the recycle bin. | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2020-11-22 08:17:45 +08:00
										 |  |  |      * @throws \Exception | 
					
						
							| 
									
										
										
										
											2020-01-12 22:45:54 +08:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function delete(string $id) | 
					
						
							| 
									
										
										
										
											2020-01-02 00:33:47 +08:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-01-12 22:45:54 +08:00
										 |  |  |         $book = Book::visible()->findOrFail($id); | 
					
						
							|  |  |  |         $this->checkOwnablePermission('book-delete', $book); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $this->bookRepo->destroy($book); | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-12 22:45:54 +08:00
										 |  |  |         return response('', 204); | 
					
						
							| 
									
										
										
										
											2020-01-02 00:33:47 +08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-06-14 00:20:21 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-20 01:14:53 +08:00
										 |  |  |     protected function rules(): array | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2022-06-14 00:20:21 +08:00
										 |  |  |         return [ | 
					
						
							|  |  |  |             'create' => [ | 
					
						
							|  |  |  |                 'name'        => ['required', 'string', 'max:255'], | 
					
						
							|  |  |  |                 'description' => ['string', 'max:1000'], | 
					
						
							|  |  |  |                 'tags'        => ['array'], | 
					
						
							|  |  |  |                 'image'       => array_merge(['nullable'], $this->getImageValidationRules()), | 
					
						
							|  |  |  |             ], | 
					
						
							|  |  |  |             'update' => [ | 
					
						
							|  |  |  |                 'name'        => ['string', 'min:1', 'max:255'], | 
					
						
							|  |  |  |                 'description' => ['string', 'max:1000'], | 
					
						
							|  |  |  |                 'tags'        => ['array'], | 
					
						
							|  |  |  |                 'image'       => array_merge(['nullable'], $this->getImageValidationRules()), | 
					
						
							|  |  |  |             ], | 
					
						
							|  |  |  |         ]; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-03-08 06:24:05 +08:00
										 |  |  | } |