| 
									
										
										
										
											2023-10-01 02:12:22 +08:00
										 |  |  | <?php | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace BookStack\Uploads; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-13 22:30:53 +08:00
										 |  |  | use BookStack\Util\FilePathNormalizer; | 
					
						
							| 
									
										
										
										
											2023-10-01 02:12:22 +08:00
										 |  |  | use Illuminate\Contracts\Filesystem\Filesystem; | 
					
						
							|  |  |  | use Illuminate\Filesystem\FilesystemAdapter; | 
					
						
							| 
									
										
										
										
											2025-05-04 03:30:50 +08:00
										 |  |  | use Illuminate\Support\Facades\Log; | 
					
						
							|  |  |  | use League\Flysystem\UnableToSetVisibility; | 
					
						
							| 
									
										
										
										
											2025-05-15 01:15:20 +08:00
										 |  |  | use League\Flysystem\Visibility; | 
					
						
							| 
									
										
										
										
											2023-10-01 02:12:22 +08:00
										 |  |  | use Symfony\Component\HttpFoundation\StreamedResponse; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ImageStorageDisk | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     public function __construct( | 
					
						
							|  |  |  |         protected string $diskName, | 
					
						
							|  |  |  |         protected Filesystem $filesystem, | 
					
						
							|  |  |  |     ) { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Check if local secure image storage (Fetched behind authentication) | 
					
						
							|  |  |  |      * is currently active in the instance. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function usingSecureImages(): bool | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return $this->diskName === 'local_secure_images'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Change the originally provided path to fit any disk-specific requirements. | 
					
						
							|  |  |  |      * This also ensures the path is kept to the expected root folders. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function adjustPathForDisk(string $path): string | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-01-13 22:30:53 +08:00
										 |  |  |         $trimmed = str_replace('uploads/images/', '', $path); | 
					
						
							|  |  |  |         $normalized = FilePathNormalizer::normalize($trimmed); | 
					
						
							| 
									
										
										
										
											2023-10-01 02:12:22 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if ($this->usingSecureImages()) { | 
					
						
							| 
									
										
										
										
											2025-01-13 22:30:53 +08:00
										 |  |  |             return $normalized; | 
					
						
							| 
									
										
										
										
											2023-10-01 02:12:22 +08:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-13 22:30:53 +08:00
										 |  |  |         return 'uploads/images/' . $normalized; | 
					
						
							| 
									
										
										
										
											2023-10-01 02:12:22 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Check if a file at the given path exists. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function exists(string $path): bool | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return $this->filesystem->exists($this->adjustPathForDisk($path)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Get the file at the given path. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2023-10-01 03:00:48 +08:00
										 |  |  |     public function get(string $path): ?string | 
					
						
							| 
									
										
										
										
											2023-10-01 02:12:22 +08:00
										 |  |  |     { | 
					
						
							|  |  |  |         return $this->filesystem->get($this->adjustPathForDisk($path)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-21 20:59:15 +08:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Get a stream to the file at the given path. | 
					
						
							|  |  |  |      * @returns ?resource | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function stream(string $path): mixed | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return $this->filesystem->readStream($this->adjustPathForDisk($path)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-01 02:12:22 +08:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Save the given image data at the given path. Can choose to set | 
					
						
							|  |  |  |      * the image as public which will update its visibility after saving. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function put(string $path, string $data, bool $makePublic = false): void | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $path = $this->adjustPathForDisk($path); | 
					
						
							|  |  |  |         $this->filesystem->put($path, $data); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-04 03:30:50 +08:00
										 |  |  |         // Set public visibility to ensure public access on S3, or that the file is accessible
 | 
					
						
							|  |  |  |         // to other processes (like web-servers) for local file storage options.
 | 
					
						
							|  |  |  |         // We avoid attempting this for (non-AWS) s3-like systems (even in a try-catch) as
 | 
					
						
							|  |  |  |         // we've always avoided setting permissions for s3-like due to potential issues,
 | 
					
						
							|  |  |  |         // with docs advising setting pre-configured permissions instead.
 | 
					
						
							|  |  |  |         // We also don't do this as the default filesystem/driver level as that can technically
 | 
					
						
							|  |  |  |         // require different ACLs for S3, and this provides us more logical control.
 | 
					
						
							| 
									
										
										
										
											2023-10-01 02:12:22 +08:00
										 |  |  |         if ($makePublic && !$this->isS3Like()) { | 
					
						
							| 
									
										
										
										
											2025-05-04 03:30:50 +08:00
										 |  |  |             try { | 
					
						
							| 
									
										
										
										
											2025-05-15 01:15:20 +08:00
										 |  |  |                 $this->filesystem->setVisibility($path, Visibility::PUBLIC); | 
					
						
							| 
									
										
										
										
											2025-05-04 03:30:50 +08:00
										 |  |  |             } catch (UnableToSetVisibility $e) { | 
					
						
							|  |  |  |                 Log::warning("Unable to set visibility for image upload with relative path: {$path}"); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2023-10-01 02:12:22 +08:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Destroys an image at the given path. | 
					
						
							|  |  |  |      * Searches for image thumbnails in addition to main provided path. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function destroyAllMatchingNameFromPath(string $path): void | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $path = $this->adjustPathForDisk($path); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $imageFolder = dirname($path); | 
					
						
							|  |  |  |         $imageFileName = basename($path); | 
					
						
							|  |  |  |         $allImages = collect($this->filesystem->allFiles($imageFolder)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Delete image files
 | 
					
						
							|  |  |  |         $imagesToDelete = $allImages->filter(function ($imagePath) use ($imageFileName) { | 
					
						
							|  |  |  |             return basename($imagePath) === $imageFileName; | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |         $this->filesystem->delete($imagesToDelete->all()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Cleanup of empty folders
 | 
					
						
							|  |  |  |         $foldersInvolved = array_merge([$imageFolder], $this->filesystem->directories($imageFolder)); | 
					
						
							|  |  |  |         foreach ($foldersInvolved as $directory) { | 
					
						
							|  |  |  |             if ($this->isFolderEmpty($directory)) { | 
					
						
							|  |  |  |                 $this->filesystem->deleteDirectory($directory); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Get the mime type of the file at the given path. | 
					
						
							|  |  |  |      * Only works for local filesystem adapters. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function mimeType(string $path): string | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-10-01 03:00:48 +08:00
										 |  |  |         $path = $this->adjustPathForDisk($path); | 
					
						
							| 
									
										
										
										
											2023-10-01 02:12:22 +08:00
										 |  |  |         return $this->filesystem instanceof FilesystemAdapter ? $this->filesystem->mimeType($path) : ''; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Get a stream response for the image at the given path. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function response(string $path): StreamedResponse | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-10-01 03:00:48 +08:00
										 |  |  |         return $this->filesystem->response($this->adjustPathForDisk($path)); | 
					
						
							| 
									
										
										
										
											2023-10-01 02:12:22 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Check if the image storage in use is an S3-like (but not likely S3) external system. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function isS3Like(): bool | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $usingS3 = $this->diskName === 's3'; | 
					
						
							|  |  |  |         return $usingS3 && !is_null(config('filesystems.disks.s3.endpoint')); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Check whether a folder is empty. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function isFolderEmpty(string $path): bool | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $files = $this->filesystem->files($path); | 
					
						
							|  |  |  |         $folders = $this->filesystem->directories($path); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return count($files) === 0 && count($folders) === 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |