Forum: - TipTap WYSIWYG editor with full toolbar - @emoji-mart/react emoji picker (consistent with tweets) - @mention autocomplete with user search API - Fix PHP 8.4 parse errors in Blade templates - Fix thread data display (paginator items) - Align forum page widths to max-w-5xl Discover: - Extract shared _nav.blade.php partial - Add missing nav links to for-you page - Add Following link for authenticated users Feed/Posts: - Post model, controllers, policies, migrations - Feed page components (PostComposer, FeedCard, etc) - Post reactions, comments, saves, reports, sharing - Scheduled publishing support - Link preview controller Profile: - Profile page components (ProfileHero, ProfileTabs) - Profile API controller Uploads: - Upload wizard enhancements - Scheduled publish picker - Studio status bar and readiness checklist
129 lines
3.9 KiB
PHP
129 lines
3.9 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\Str;
|
|
use RuntimeException;
|
|
|
|
final class UploadStorageService
|
|
{
|
|
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 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 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);
|
|
}
|
|
}
|