246 lines
7.6 KiB
PHP
246 lines
7.6 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace App\Services\Uploads;
|
|
|
|
use App\DTOs\Uploads\UploadStoredFile;
|
|
use Illuminate\Http\UploadedFile;
|
|
use Illuminate\Support\Facades\File;
|
|
use Illuminate\Support\Facades\Storage;
|
|
use Illuminate\Support\Str;
|
|
use RuntimeException;
|
|
|
|
final class UploadStorageService
|
|
{
|
|
public function localOriginalsRoot(): string
|
|
{
|
|
return rtrim((string) config('uploads.local_originals_root'), DIRECTORY_SEPARATOR);
|
|
}
|
|
|
|
public function sectionPath(string $section): string
|
|
{
|
|
$root = rtrim((string) config('uploads.storage_root'), DIRECTORY_SEPARATOR);
|
|
$paths = (array) config('uploads.paths');
|
|
|
|
if (! array_key_exists($section, $paths)) {
|
|
throw new RuntimeException('Unknown upload storage section: ' . $section);
|
|
}
|
|
|
|
return $root . DIRECTORY_SEPARATOR . trim((string) $paths[$section], DIRECTORY_SEPARATOR);
|
|
}
|
|
|
|
public function ensureSection(string $section): string
|
|
{
|
|
$path = $this->sectionPath($section);
|
|
|
|
if (! File::exists($path)) {
|
|
File::makeDirectory($path, 0755, true);
|
|
}
|
|
|
|
return $path;
|
|
}
|
|
|
|
public function storeUploadedFile(UploadedFile $file, string $section): UploadStoredFile
|
|
{
|
|
$dir = $this->ensureSection($section);
|
|
$extension = $this->safeExtension($file);
|
|
$filename = Str::uuid()->toString() . ($extension !== '' ? '.' . $extension : '');
|
|
|
|
$file->move($dir, $filename);
|
|
|
|
$path = $dir . DIRECTORY_SEPARATOR . $filename;
|
|
|
|
return UploadStoredFile::fromPath($path);
|
|
}
|
|
|
|
public function moveToSection(string $path, string $section): string
|
|
{
|
|
if (! is_file($path)) {
|
|
throw new RuntimeException('Source file not found for move.');
|
|
}
|
|
|
|
$dir = $this->ensureSection($section);
|
|
$extension = (string) pathinfo($path, PATHINFO_EXTENSION);
|
|
$filename = Str::uuid()->toString() . ($extension !== '' ? '.' . $extension : '');
|
|
$target = $dir . DIRECTORY_SEPARATOR . $filename;
|
|
|
|
File::move($path, $target);
|
|
|
|
return $target;
|
|
}
|
|
|
|
public function ensureHashDirectory(string $section, string $hash): string
|
|
{
|
|
$segments = $this->hashSegments($hash);
|
|
$dir = $this->sectionPath($section) . DIRECTORY_SEPARATOR . implode(DIRECTORY_SEPARATOR, $segments);
|
|
|
|
if (! File::exists($dir)) {
|
|
File::makeDirectory($dir, 0755, true);
|
|
}
|
|
|
|
return $dir;
|
|
}
|
|
|
|
public function ensureLocalOriginalHashDirectory(string $hash): string
|
|
{
|
|
$segments = $this->hashSegments($hash);
|
|
$dir = $this->localOriginalsRoot() . DIRECTORY_SEPARATOR . implode(DIRECTORY_SEPARATOR, $segments);
|
|
|
|
if (! File::exists($dir)) {
|
|
File::makeDirectory($dir, 0755, true);
|
|
}
|
|
|
|
return $dir;
|
|
}
|
|
|
|
public function sectionRelativePath(string $section, string $hash, string $filename): string
|
|
{
|
|
$segments = $this->hashSegments($hash);
|
|
$section = trim($section, DIRECTORY_SEPARATOR);
|
|
|
|
return $section . '/' . implode('/', $segments) . '/' . ltrim($filename, '/');
|
|
}
|
|
|
|
public function localOriginalPath(string $hash, string $filename): string
|
|
{
|
|
return $this->ensureLocalOriginalHashDirectory($hash) . DIRECTORY_SEPARATOR . ltrim($filename, DIRECTORY_SEPARATOR);
|
|
}
|
|
|
|
public function objectDiskName(): string
|
|
{
|
|
return (string) config('uploads.object_storage.disk', 's3');
|
|
}
|
|
|
|
public function objectBasePrefix(): string
|
|
{
|
|
return trim((string) config('uploads.object_storage.prefix', 'artworks'), '/');
|
|
}
|
|
|
|
public function objectPathForVariant(string $variant, string $hash, string $filename): string
|
|
{
|
|
$segments = implode('/', $this->hashSegments($hash));
|
|
$basePrefix = $this->objectBasePrefix();
|
|
$normalizedVariant = trim($variant, '/');
|
|
|
|
if ($normalizedVariant === 'original') {
|
|
return sprintf('%s/original/%s/%s', $basePrefix, $segments, ltrim($filename, '/'));
|
|
}
|
|
|
|
return sprintf('%s/%s/%s/%s', $basePrefix, $normalizedVariant, $segments, ltrim($filename, '/'));
|
|
}
|
|
|
|
public function putObjectFromPath(string $sourcePath, string $objectPath, string $contentType, array $extraOptions = []): void
|
|
{
|
|
$stream = fopen($sourcePath, 'rb');
|
|
if ($stream === false) {
|
|
throw new RuntimeException('Unable to open source file for object storage upload.');
|
|
}
|
|
|
|
try {
|
|
$written = Storage::disk($this->objectDiskName())->put($objectPath, $stream, array_merge([
|
|
'visibility' => 'public',
|
|
'CacheControl' => 'public, max-age=31536000, immutable',
|
|
'ContentType' => $contentType,
|
|
], $extraOptions));
|
|
} finally {
|
|
fclose($stream);
|
|
}
|
|
|
|
if ($written !== true) {
|
|
throw new RuntimeException('Object storage upload failed.');
|
|
}
|
|
}
|
|
|
|
public function putObjectContents(string $objectPath, string $contents, string $contentType, array $extraOptions = []): void
|
|
{
|
|
$written = Storage::disk($this->objectDiskName())->put($objectPath, $contents, array_merge([
|
|
'visibility' => 'public',
|
|
'CacheControl' => 'public, max-age=31536000, immutable',
|
|
'ContentType' => $contentType,
|
|
], $extraOptions));
|
|
|
|
if ($written !== true) {
|
|
throw new RuntimeException('Object storage upload failed.');
|
|
}
|
|
}
|
|
|
|
public function deleteObject(string $objectPath): void
|
|
{
|
|
if ($objectPath === '') {
|
|
return;
|
|
}
|
|
|
|
Storage::disk($this->objectDiskName())->delete($objectPath);
|
|
}
|
|
|
|
public function readObject(string $objectPath): ?string
|
|
{
|
|
if ($objectPath === '') {
|
|
return null;
|
|
}
|
|
|
|
$disk = Storage::disk($this->objectDiskName());
|
|
if (! $disk->exists($objectPath)) {
|
|
return null;
|
|
}
|
|
|
|
$contents = $disk->get($objectPath);
|
|
|
|
return is_string($contents) && $contents !== '' ? $contents : null;
|
|
}
|
|
|
|
public function deleteLocalFile(?string $path): void
|
|
{
|
|
if (! is_string($path) || trim($path) === '') {
|
|
return;
|
|
}
|
|
|
|
if (File::exists($path)) {
|
|
File::delete($path);
|
|
}
|
|
}
|
|
|
|
public function originalHashExists(string $hash): bool
|
|
{
|
|
$segments = $this->hashSegments($hash);
|
|
$dir = $this->sectionPath('original') . DIRECTORY_SEPARATOR . implode(DIRECTORY_SEPARATOR, $segments);
|
|
|
|
if (! File::isDirectory($dir)) {
|
|
return false;
|
|
}
|
|
|
|
$normalizedHash = strtolower(preg_replace('/[^a-z0-9]/', '', $hash) ?? '');
|
|
if ($normalizedHash === '') {
|
|
return false;
|
|
}
|
|
|
|
$matches = File::glob($dir . DIRECTORY_SEPARATOR . $normalizedHash . '.*');
|
|
return is_array($matches) && count($matches) > 0;
|
|
}
|
|
|
|
private function safeExtension(UploadedFile $file): string
|
|
{
|
|
$extension = (string) $file->guessExtension();
|
|
$extension = strtolower($extension);
|
|
|
|
return preg_match('/^[a-z0-9]+$/', $extension) ? $extension : '';
|
|
}
|
|
|
|
private function hashSegments(string $hash): array
|
|
{
|
|
$hash = strtolower($hash);
|
|
$hash = preg_replace('/[^a-z0-9]/', '', $hash) ?? '';
|
|
$hash = str_pad($hash, 6, '0');
|
|
|
|
// Use two 2-char segments for directory sharding: first two chars, next two chars.
|
|
// Result: <section>/<aa>/<bb>/<filename>
|
|
$segments = [
|
|
substr($hash, 0, 2),
|
|
substr($hash, 2, 2),
|
|
];
|
|
|
|
return array_map(static fn (string $part): string => $part === '' ? '00' : $part, $segments);
|
|
}
|
|
}
|