| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  | <?php | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace BookStack\Uploads; | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | use BookStack\Exceptions\FileUploadException; | 
					
						
							|  |  |  | use Exception; | 
					
						
							| 
									
										
										
										
											2020-12-06 20:58:40 +08:00
										 |  |  | use Illuminate\Contracts\Filesystem\Factory as FileSystem; | 
					
						
							| 
									
										
										
										
											2021-06-06 07:51:06 +08:00
										 |  |  | use Illuminate\Contracts\Filesystem\FileNotFoundException; | 
					
						
							| 
									
										
										
										
											2020-12-06 20:58:40 +08:00
										 |  |  | use Illuminate\Contracts\Filesystem\Filesystem as FileSystemInstance; | 
					
						
							| 
									
										
										
										
											2019-09-14 06:58:40 +08:00
										 |  |  | use Illuminate\Support\Str; | 
					
						
							| 
									
										
										
										
											2021-06-06 07:51:06 +08:00
										 |  |  | use Log; | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  | use Symfony\Component\HttpFoundation\File\UploadedFile; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-06 20:58:40 +08:00
										 |  |  | class AttachmentService | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-12-06 20:58:40 +08:00
										 |  |  |     protected $fileSystem; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * AttachmentService constructor. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function __construct(FileSystem $fileSystem) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->fileSystem = $fileSystem; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-13 19:11:23 +08:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Get the storage that will be used for storing files. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2020-12-06 20:58:40 +08:00
										 |  |  |     protected function getStorage(): FileSystemInstance | 
					
						
							| 
									
										
										
										
											2018-01-13 19:11:23 +08:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-06-23 23:01:15 +08:00
										 |  |  |         $storageType = config('filesystems.attachments'); | 
					
						
							| 
									
										
										
										
											2018-01-13 19:11:23 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Override default location if set to local public to ensure not visible.
 | 
					
						
							|  |  |  |         if ($storageType === 'local') { | 
					
						
							|  |  |  |             $storageType = 'local_secure'; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 19:41:52 +08:00
										 |  |  |         return $this->fileSystem->disk($storageType); | 
					
						
							| 
									
										
										
										
											2018-01-13 19:11:23 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Get an attachment from storage. | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2021-06-06 07:51:06 +08:00
										 |  |  |      * @throws FileNotFoundException | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2021-06-06 07:51:06 +08:00
										 |  |  |     public function getAttachmentFromStorage(Attachment $attachment): string | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2018-01-13 19:11:23 +08:00
										 |  |  |         return $this->getStorage()->get($attachment->path); | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Store a new attachment upon user upload. | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |      * @param UploadedFile $uploadedFile | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |      * @param int          $page_id | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |      * @throws FileUploadException | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @return Attachment | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function saveNewUpload(UploadedFile $uploadedFile, $page_id) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $attachmentName = $uploadedFile->getClientOriginalName(); | 
					
						
							| 
									
										
										
										
											2019-03-25 03:07:18 +08:00
										 |  |  |         $attachmentPath = $this->putFileInStorage($uploadedFile); | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |         $largestExistingOrder = Attachment::where('uploaded_to', '=', $page_id)->max('order'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $attachment = Attachment::forceCreate([ | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |             'name'        => $attachmentName, | 
					
						
							|  |  |  |             'path'        => $attachmentPath, | 
					
						
							|  |  |  |             'extension'   => $uploadedFile->getClientOriginalExtension(), | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |             'uploaded_to' => $page_id, | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |             'created_by'  => user()->id, | 
					
						
							|  |  |  |             'updated_by'  => user()->id, | 
					
						
							|  |  |  |             'order'       => $largestExistingOrder + 1, | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |         ]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $attachment; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Store a upload, saving to a file and deleting any existing uploads | 
					
						
							|  |  |  |      * attached to that file. | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |      * @param UploadedFile $uploadedFile | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |      * @param Attachment   $attachment | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |      * @throws FileUploadException | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @return Attachment | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function saveUpdatedUpload(UploadedFile $uploadedFile, Attachment $attachment) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (!$attachment->external) { | 
					
						
							|  |  |  |             $this->deleteFileInStorage($attachment); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $attachmentName = $uploadedFile->getClientOriginalName(); | 
					
						
							| 
									
										
										
										
											2019-03-25 03:07:18 +08:00
										 |  |  |         $attachmentPath = $this->putFileInStorage($uploadedFile); | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $attachment->name = $attachmentName; | 
					
						
							|  |  |  |         $attachment->path = $attachmentPath; | 
					
						
							|  |  |  |         $attachment->external = false; | 
					
						
							|  |  |  |         $attachment->extension = $uploadedFile->getClientOriginalExtension(); | 
					
						
							|  |  |  |         $attachment->save(); | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |         return $attachment; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Save a new File attachment from a given link and name. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2020-10-31 23:01:52 +08:00
										 |  |  |     public function saveNewFromLink(string $name, string $link, int $page_id): Attachment | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |     { | 
					
						
							|  |  |  |         $largestExistingOrder = Attachment::where('uploaded_to', '=', $page_id)->max('order'); | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |         return Attachment::forceCreate([ | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |             'name'        => $name, | 
					
						
							|  |  |  |             'path'        => $link, | 
					
						
							|  |  |  |             'external'    => true, | 
					
						
							|  |  |  |             'extension'   => '', | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |             'uploaded_to' => $page_id, | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |             'created_by'  => user()->id, | 
					
						
							|  |  |  |             'updated_by'  => user()->id, | 
					
						
							|  |  |  |             'order'       => $largestExistingOrder + 1, | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |         ]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							| 
									
										
										
										
											2020-07-01 05:12:45 +08:00
										 |  |  |      * Updates the ordering for a listing of attached files. | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2020-07-01 05:12:45 +08:00
										 |  |  |     public function updateFileOrderWithinPage(array $attachmentOrder, string $pageId) | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-07-01 05:12:45 +08:00
										 |  |  |         foreach ($attachmentOrder as $index => $attachmentId) { | 
					
						
							|  |  |  |             Attachment::query()->where('uploaded_to', '=', $pageId) | 
					
						
							|  |  |  |                 ->where('id', '=', $attachmentId) | 
					
						
							|  |  |  |                 ->update(['order' => $index]); | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Update the details of a file. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2020-10-31 23:01:52 +08:00
										 |  |  |     public function updateFile(Attachment $attachment, array $requestData): Attachment | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |     { | 
					
						
							|  |  |  |         $attachment->name = $requestData['name']; | 
					
						
							| 
									
										
										
										
											2020-10-31 23:01:52 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |         if (isset($requestData['link']) && trim($requestData['link']) !== '') { | 
					
						
							|  |  |  |             $attachment->path = $requestData['link']; | 
					
						
							|  |  |  |             if (!$attachment->external) { | 
					
						
							|  |  |  |                 $this->deleteFileInStorage($attachment); | 
					
						
							|  |  |  |                 $attachment->external = true; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-10-31 23:01:52 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |         $attachment->save(); | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |         return $attachment; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Delete a File from the database and storage. | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |      * @param Attachment $attachment | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2018-01-13 19:11:23 +08:00
										 |  |  |      * @throws Exception | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function deleteFile(Attachment $attachment) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if ($attachment->external) { | 
					
						
							|  |  |  |             $attachment->delete(); | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |         $this->deleteFileInStorage($attachment); | 
					
						
							|  |  |  |         $attachment->delete(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Delete a file from the filesystem it sits on. | 
					
						
							|  |  |  |      * Cleans any empty leftover folders. | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |      * @param Attachment $attachment | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function deleteFileInStorage(Attachment $attachment) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $storage = $this->getStorage(); | 
					
						
							| 
									
										
										
										
											2018-01-13 19:11:23 +08:00
										 |  |  |         $dirPath = dirname($attachment->path); | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-13 19:11:23 +08:00
										 |  |  |         $storage->delete($attachment->path); | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |         if (count($storage->allFiles($dirPath)) === 0) { | 
					
						
							|  |  |  |             $storage->deleteDirectory($dirPath); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |      * Store a file in storage with the given filename. | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |      * @param UploadedFile $uploadedFile | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |      * @throws FileUploadException | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @return string | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2019-03-25 03:07:18 +08:00
										 |  |  |     protected function putFileInStorage(UploadedFile $uploadedFile) | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |     { | 
					
						
							|  |  |  |         $attachmentData = file_get_contents($uploadedFile->getRealPath()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $storage = $this->getStorage(); | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  |         $basePath = 'uploads/files/' . date('Y-m-M') . '/'; | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-14 06:58:40 +08:00
										 |  |  |         $uploadFileName = Str::random(16) . '.' . $uploadedFile->getClientOriginalExtension(); | 
					
						
							| 
									
										
										
										
											2018-01-13 19:11:23 +08:00
										 |  |  |         while ($storage->exists($basePath . $uploadFileName)) { | 
					
						
							| 
									
										
										
										
											2019-09-14 06:58:40 +08:00
										 |  |  |             $uploadFileName = Str::random(3) . $uploadFileName; | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-13 19:11:23 +08:00
										 |  |  |         $attachmentPath = $basePath . $uploadFileName; | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |         try { | 
					
						
							| 
									
										
										
										
											2018-01-13 19:11:23 +08:00
										 |  |  |             $storage->put($attachmentPath, $attachmentData); | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |         } catch (Exception $e) { | 
					
						
							| 
									
										
										
										
											2021-06-06 07:51:06 +08:00
										 |  |  |             Log::error('Error when attempting file upload:' . $e->getMessage()); | 
					
						
							| 
									
										
										
										
											2021-06-26 23:23:15 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-13 19:11:23 +08:00
										 |  |  |             throw new FileUploadException(trans('errors.path_not_writable', ['filePath' => $attachmentPath])); | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-13 19:11:23 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-12 22:12:26 +08:00
										 |  |  |         return $attachmentPath; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-29 00:58:52 +08:00
										 |  |  | } |