| 
									
										
										
										
											2021-10-19 00:46:55 +08:00
										 |  |  | <?php | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace BookStack\Http\Controllers\Api; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | use BookStack\Entities\Models\Page; | 
					
						
							|  |  |  | use BookStack\Exceptions\FileUploadException; | 
					
						
							|  |  |  | use BookStack\Uploads\Attachment; | 
					
						
							|  |  |  | use BookStack\Uploads\AttachmentService; | 
					
						
							|  |  |  | use Exception; | 
					
						
							|  |  |  | use Illuminate\Contracts\Filesystem\FileNotFoundException; | 
					
						
							|  |  |  | use Illuminate\Http\Request; | 
					
						
							|  |  |  | use Illuminate\Validation\ValidationException; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class AttachmentApiController extends ApiController | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     protected $attachmentService; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     protected $rules = [ | 
					
						
							|  |  |  |         'create' => [ | 
					
						
							| 
									
										
										
										
											2021-10-20 17:49:45 +08:00
										 |  |  |             'name'        => 'required|min:1|max:255|string', | 
					
						
							| 
									
										
										
										
											2021-10-19 00:46:55 +08:00
										 |  |  |             'uploaded_to' => 'required|integer|exists:pages,id', | 
					
						
							| 
									
										
										
										
											2021-10-20 17:49:45 +08:00
										 |  |  |             'file'        => 'required_without:link|file', | 
					
						
							|  |  |  |             'link'        => 'required_without:file|min:1|max:255|safe_url', | 
					
						
							| 
									
										
										
										
											2021-10-19 00:46:55 +08:00
										 |  |  |         ], | 
					
						
							|  |  |  |         'update' => [ | 
					
						
							| 
									
										
										
										
											2021-10-20 17:49:45 +08:00
										 |  |  |             'name'        => 'min:1|max:255|string', | 
					
						
							| 
									
										
										
										
											2021-10-19 00:46:55 +08:00
										 |  |  |             'uploaded_to' => 'integer|exists:pages,id', | 
					
						
							| 
									
										
										
										
											2021-10-20 17:49:45 +08:00
										 |  |  |             'file'        => 'file', | 
					
						
							|  |  |  |             'link'        => 'min:1|max:255|safe_url', | 
					
						
							| 
									
										
										
										
											2021-10-19 00:46:55 +08:00
										 |  |  |         ], | 
					
						
							|  |  |  |     ]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public function __construct(AttachmentService $attachmentService) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->attachmentService = $attachmentService; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Get a listing of attachments visible to the user. | 
					
						
							|  |  |  |      * The external property indicates whether the attachment is simple a link. | 
					
						
							|  |  |  |      * A false value for the external property would indicate a file upload. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function list() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return $this->apiListingResponse(Attachment::visible(), [ | 
					
						
							|  |  |  |             'id', 'name', 'extension', 'uploaded_to', 'external', 'order', 'created_at', 'updated_at', 'created_by', 'updated_by', | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Create a new attachment in the system. | 
					
						
							|  |  |  |      * An uploaded_to value must be provided containing an ID of the page | 
					
						
							|  |  |  |      * that this upload will be related to. | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2021-10-20 17:43:03 +08:00
										 |  |  |      * If you're uploading a file the POST data should be provided via | 
					
						
							|  |  |  |      * a multipart/form-data type request instead of JSON. | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2021-10-19 00:46:55 +08:00
										 |  |  |      * @throws ValidationException | 
					
						
							|  |  |  |      * @throws FileUploadException | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function create(Request $request) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->checkPermission('attachment-create-all'); | 
					
						
							|  |  |  |         $requestData = $this->validate($request, $this->rules['create']); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $pageId = $request->get('uploaded_to'); | 
					
						
							|  |  |  |         $page = Page::visible()->findOrFail($pageId); | 
					
						
							|  |  |  |         $this->checkOwnablePermission('page-update', $page); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ($request->hasFile('file')) { | 
					
						
							|  |  |  |             $uploadedFile = $request->file('file'); | 
					
						
							|  |  |  |             $attachment = $this->attachmentService->saveNewUpload($uploadedFile, $page->id); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             $attachment = $this->attachmentService->saveNewFromLink( | 
					
						
							| 
									
										
										
										
											2021-10-20 17:49:45 +08:00
										 |  |  |                 $requestData['name'], | 
					
						
							|  |  |  |                 $requestData['link'], | 
					
						
							|  |  |  |                 $page->id | 
					
						
							| 
									
										
										
										
											2021-10-19 00:46:55 +08:00
										 |  |  |             ); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $this->attachmentService->updateFile($attachment, $requestData); | 
					
						
							| 
									
										
										
										
											2021-10-20 17:49:45 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-19 00:46:55 +08:00
										 |  |  |         return response()->json($attachment); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Get the details & content of a single attachment of the given ID. | 
					
						
							|  |  |  |      * The attachment link or file content is provided via a 'content' property. | 
					
						
							|  |  |  |      * For files the content will be base64 encoded. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @throws FileNotFoundException | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function read(string $id) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         /** @var Attachment $attachment */ | 
					
						
							| 
									
										
										
										
											2021-10-20 07:58:56 +08:00
										 |  |  |         $attachment = Attachment::visible() | 
					
						
							|  |  |  |             ->with(['createdBy', 'updatedBy']) | 
					
						
							|  |  |  |             ->findOrFail($id); | 
					
						
							| 
									
										
										
										
											2021-10-19 00:46:55 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $attachment->setAttribute('links', [ | 
					
						
							|  |  |  |             'html'     => $attachment->htmlLink(), | 
					
						
							|  |  |  |             'markdown' => $attachment->markdownLink(), | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!$attachment->external) { | 
					
						
							|  |  |  |             $attachmentContents = $this->attachmentService->getAttachmentFromStorage($attachment); | 
					
						
							|  |  |  |             $attachment->setAttribute('content', base64_encode($attachmentContents)); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             $attachment->setAttribute('content', $attachment->path); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return response()->json($attachment); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Update the details of a single attachment. | 
					
						
							| 
									
										
										
										
											2021-10-20 17:43:03 +08:00
										 |  |  |      * As per the create endpoint, if a file is being provided as the attachment content | 
					
						
							|  |  |  |      * the request should be formatted as a multipart/form-data request instead of JSON. | 
					
						
							| 
									
										
										
										
											2021-10-19 00:46:55 +08:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @throws ValidationException | 
					
						
							|  |  |  |      * @throws FileUploadException | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function update(Request $request, string $id) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $requestData = $this->validate($request, $this->rules['update']); | 
					
						
							|  |  |  |         /** @var Attachment $attachment */ | 
					
						
							|  |  |  |         $attachment = Attachment::visible()->findOrFail($id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $page = $attachment->page; | 
					
						
							|  |  |  |         if ($requestData['uploaded_to'] ?? false) { | 
					
						
							|  |  |  |             $pageId = $request->get('uploaded_to'); | 
					
						
							|  |  |  |             $page = Page::visible()->findOrFail($pageId); | 
					
						
							|  |  |  |             $attachment->uploaded_to = $requestData['uploaded_to']; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $this->checkOwnablePermission('page-view', $page); | 
					
						
							|  |  |  |         $this->checkOwnablePermission('page-update', $page); | 
					
						
							|  |  |  |         $this->checkOwnablePermission('attachment-update', $attachment); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ($request->hasFile('file')) { | 
					
						
							|  |  |  |             $uploadedFile = $request->file('file'); | 
					
						
							| 
									
										
										
										
											2021-10-20 07:58:56 +08:00
										 |  |  |             $attachment = $this->attachmentService->saveUpdatedUpload($uploadedFile, $attachment); | 
					
						
							| 
									
										
										
										
											2021-10-19 00:46:55 +08:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $this->attachmentService->updateFile($attachment, $requestData); | 
					
						
							| 
									
										
										
										
											2021-10-20 17:49:45 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-19 00:46:55 +08:00
										 |  |  |         return response()->json($attachment); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Delete an attachment of the given ID. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @throws Exception | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function delete(string $id) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         /** @var Attachment $attachment */ | 
					
						
							|  |  |  |         $attachment = Attachment::visible()->findOrFail($id); | 
					
						
							|  |  |  |         $this->checkOwnablePermission('attachment-delete', $attachment); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $this->attachmentService->deleteFile($attachment); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return response('', 204); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |