From 982de9715625185e1fe6685ec2654a1ddc862796 Mon Sep 17 00:00:00 2001 From: sometimes-coding Date: Thu, 19 Dec 2024 15:25:13 -0500 Subject: [PATCH] Adds cover_url and banner_url to API , fixes blades --- app/Helpers/UrlHelper.php | 84 +++++++++++++ .../Controllers/API/TorrentController.php | 118 ++++++++++++++++++ app/Models/Torrent.php | 8 ++ composer.json | 3 +- ...cover_and_banner_url_to_torrents_table.php | 33 +++++ database/schema/mysql-schema.sql | 1 + .../views/components/torrent/card.blade.php | 47 ++++--- .../views/components/torrent/row.blade.php | 41 +++--- .../views/torrent/download_check.blade.php | 2 +- .../views/torrent/partials/no_meta.blade.php | 29 +++-- 10 files changed, 322 insertions(+), 44 deletions(-) create mode 100644 app/Helpers/UrlHelper.php create mode 100644 database/migrations/2024_12_16_183945_add_cover_and_banner_url_to_torrents_table.php diff --git a/app/Helpers/UrlHelper.php b/app/Helpers/UrlHelper.php new file mode 100644 index 0000000000..c9cac35b44 --- /dev/null +++ b/app/Helpers/UrlHelper.php @@ -0,0 +1,84 @@ +downloadAndStoreImage($url, $type, $torrentId); + } + } + + /** * Store a newly created resource in storage. */ @@ -172,6 +184,24 @@ public function store(Request $request): \Illuminate\Http\JsonResponse $torrent->refundable = $user->group->is_modo || $user->group->is_internal ? ($request->input('refundable') ?? false) : false; $du_until = $request->input('du_until'); + // Check and set if it's in safe external hosts + if ($request->has('cover_url')) { + $coverUrl = $request->input('cover_url'); + if (UrlHelper::isTrustedExternalHost($coverUrl)) { + $torrent->cover_url = $coverUrl; + } + // Note: We don't set it here if not safe external host because we need the torrent ID at save + } + + // Check and set banner_url if it's safe external hosts + if ($request->has('banner_url')) { + $bannerUrl = $request->input('banner_url'); + if (UrlHelper::isTrustedExternalHost($bannerUrl)) { + $torrent->banner_url = $bannerUrl; + } + // Note: We don't set it here if not safe external host because we need the torrent ID at save + } + if (($user->group->is_modo || $user->group->is_internal) && isset($du_until)) { $torrent->du_until = Carbon::now()->addDays($request->integer('du_until')); } @@ -193,6 +223,8 @@ public function store(Request $request): \Illuminate\Http\JsonResponse // Validation $v = validator($torrent->toArray(), [ + 'cover_url' => ['nullable', 'url', 'regex:/\.(avif|bmp|eps|heic|heif|jpe?g|png|svg|tiff?|webp)$/i'], + 'banner_url' => ['nullable', 'url', 'regex:/\.(avif|bmp|eps|heic|heif|jpe?g|png|svg|tiff?|webp)$/i'], 'name' => [ 'required', Rule::unique('torrents')->whereNull('deleted_at'), @@ -320,6 +352,15 @@ public function store(Request $request): \Illuminate\Http\JsonResponse // Save The Torrent $torrent->save(); + // Set cover and/or banner _ if not safe2host externally + if ($request->has('cover_url') && !$torrent->cover_url) { + $torrent->cover_url = $this->handleImage($request->input('cover_url'), 'cover', $torrent->id); + } + + if ($request->has('banner_url') && !$torrent->banner_url) { + $torrent->banner_url = $this->handleImage($request->input('banner_url'), 'banner', $torrent->id); + } + // Populate the status/seeders/leechers/times_completed fields for the external tracker $torrent->refresh(); @@ -448,6 +489,83 @@ public function store(Request $request): \Illuminate\Http\JsonResponse return $this->sendResponse(route('torrent.download.rsskey', ['id' => $torrent->id, 'rsskey' => auth('api')->user()->rsskey]), 'Torrent uploaded successfully.'); } + /** + * Download image from URL and store it locally, returning the URL to stored image. + * If size cannot be checked beforehand, temporarily download, check size (discard if oversized) else compress then rename. + * + * @param string $url + * @param string $type ('cover' or 'banner') + * @param int $torrentId + * @return string|null + */ + private function downloadAndStoreImage($url, $type, int $torrentId) + { + try { + $headers = get_headers($url, true); + $contentLength = isset($headers['Content-Length']) ? (int)$headers['Content-Length'] : null; + + $prefix = ($type === 'cover') ? 'torrent-cover' : 'torrent-banner'; + $tempName = "{$prefix}_{$torrentId}_temp"; + $finalName = "{$prefix}_{$torrentId}.webp"; + $tempPath = public_path('/files/img/') . $tempName; + $finalPath = public_path('/files/img/') . $finalName; + + // Skip if over 12MB -- should be easily configurable. Note: 10MB PNG can be crunched to 200-400Kb in WebP + if ($contentLength !== null && $contentLength > 12 * 1024 * 1024) { + \Log::warning("Image from URL {$url} exceeds the 12MB size limit. Skipping download."); + return null; + } + + $contents = file_get_contents($url); + if ($contents === false) { + throw new \Exception("Failed to fetch image from URL"); + } + + if (!file_put_contents($tempPath, $contents)) { + throw new \Exception("Failed to save temporary image"); + } + + $imageInfo = getimagesize($tempPath); + $mimeType = $imageInfo['mime']; + if ($mimeType === 'image/webp') { // No need to convert if already webP + rename($tempPath, $finalPath); + return asset('files/img/' . $finalName); + } + + // Convert and compress based on size + $actualSize = $contentLength ?? filesize($tempPath); + $image = new \Imagick($tempPath); + + // Preserve Alpha Channels (Transparency) + if ($image->getImageAlphaChannel() !== \Imagick::ALPHACHANNEL_ACTIVATE) { + $image->setImageAlphaChannel(\Imagick::ALPHACHANNEL_ACTIVATE); + } + + // Strip all metadata + $image->stripImage(); + + // Ensure alpha channel remains active after stripping + $image->setImageAlphaChannel(\Imagick::ALPHACHANNEL_ACTIVATE); + + // Convert to WebP with appropriate quality + $quality = $actualSize <= 2 * 1024 * 1024 ? 100 : 80; + $image->setImageFormat('webp'); + $image->setImageCompressionQuality($quality); + $image->setOption('webp:lossless', 'false'); + $image->writeImage($finalPath); + unlink($tempPath); // Remove the temporary file + + return asset('files/img/' . $finalName); + } catch (\Exception $e) { + \Log::error("Failed to download and store image: " . $e->getMessage() . " in " . $e->getFile() . " on line " . $e->getLine()); + if (isset($tempPath) && file_exists($tempPath)) { + unlink($tempPath); + } + return null; + } + } + + /** * Display the specified resource. */ diff --git a/app/Models/Torrent.php b/app/Models/Torrent.php index 0230b3dfab..ffa6c48d66 100644 --- a/app/Models/Torrent.php +++ b/app/Models/Torrent.php @@ -36,6 +36,8 @@ * * @property string $info_hash * @property int $id + * @property string|null $cover_url + * @property string|null $banner_url * @property string $name * @property string $description * @property string|null $mediainfo @@ -140,6 +142,8 @@ protected function casts(): array public const string SEARCHABLE = <<<'SQL' torrents.id, torrents.name, + torrents.cover_url, + torrents.banner_url, torrents.description, torrents.mediainfo, torrents.bdinfo, @@ -832,6 +836,8 @@ public function toSearchableArray(): array $missingRequiredAttributes = array_diff([ 'id', 'name', + 'cover_url', + 'banner_url', 'description', 'mediainfo', 'bdinfo', @@ -904,6 +910,8 @@ public function toSearchableArray(): array return [ 'id' => $torrent->id, 'name' => $torrent->name, + 'cover_url' => $torrent->cover_url, + 'banner_url' => $torrent->banner_url, 'description' => $torrent->description, 'mediainfo' => $torrent->mediainfo, 'bdinfo' => $torrent->bdinfo, diff --git a/composer.json b/composer.json index 90afb6fb64..53e377e2fd 100644 --- a/composer.json +++ b/composer.json @@ -42,7 +42,8 @@ "symfony/dom-crawler": "^6.4.12", "theodorejb/polycast": "dev-master", "voku/anti-xss": "^4.1.42", - "vstelmakh/url-highlight": "^3.1.1" + "vstelmakh/url-highlight": "^3.1.1", + "ext-imagick": "*" }, "require-dev": { "barryvdh/laravel-debugbar": "^3.14.6", diff --git a/database/migrations/2024_12_16_183945_add_cover_and_banner_url_to_torrents_table.php b/database/migrations/2024_12_16_183945_add_cover_and_banner_url_to_torrents_table.php new file mode 100644 index 0000000000..717d44faed --- /dev/null +++ b/database/migrations/2024_12_16_183945_add_cover_and_banner_url_to_torrents_table.php @@ -0,0 +1,33 @@ +string('cover_url', 255)->nullable()->after('folder'); + $table->string('banner_url', 255)->nullable()->after('cover_url'); + }); + } + + /** + * Reverse the migrations. + * + * @return void + */ + public function down() + { + Schema::table('torrents', function (Blueprint $table) { + $table->dropColumn(['cover_url', 'banner_url']); + }); + } +} diff --git a/database/schema/mysql-schema.sql b/database/schema/mysql-schema.sql index f800440790..6350ed3958 100644 --- a/database/schema/mysql-schema.sql +++ b/database/schema/mysql-schema.sql @@ -2835,3 +2835,4 @@ INSERT INTO `migrations` (`id`, `migration`, `batch`) VALUES (324,'2024_10_29_18 INSERT INTO `migrations` (`id`, `migration`, `batch`) VALUES (325,'2024_11_01_013426_add_soft_deletes_to_donation_packages_table',1); INSERT INTO `migrations` (`id`, `migration`, `batch`) VALUES (326,'2024_11_13_044550_create_unregistered_info_hashes_table',1); INSERT INTO `migrations` (`id`, `migration`, `batch`) VALUES (327,'2024_11_26_170256_add_is_torrent_modo_to_groups_table',1); +INSERT INTO `migrations` (`id`, `migration`, `batch`) VALUES (328, '2024_12_16_183945_add_cover_and_banner_url_to_torrents_table.php', 1); diff --git a/resources/views/components/torrent/card.blade.php b/resources/views/components/torrent/card.blade.php index 515c3f488e..d3d229764d 100644 --- a/resources/views/components/torrent/card.blade.php +++ b/resources/views/components/torrent/card.blade.php @@ -44,23 +44,40 @@ class="torrent-card__similar-link"
category->movie_meta || $torrent->category->tv_meta) src="{{ isset($meta->poster) ? tmdb_image('poster_mid', $meta->poster) : 'https://via.placeholder.com/160x240' }}" - - @break - @case($torrent->category->game_meta && isset($torrent->meta) && $meta->cover->image_id && $meta->name) - src="https://images.igdb.com/igdb/image/upload/t_cover_big/{{ $torrent->meta->cover->image_id }}.jpg" - - @break - @case($torrent->category->music_meta) - src="https://via.placeholder.com/160x240" - - @break - @case($torrent->category->no_meta && file_exists(public_path() . '/files/img/torrent-cover_' . $torrent->id . '.jpg')) - src="{{ url('files/img/torrent-cover_' . $torrent->id . '.jpg') }}" - - @break + + @break + @case($torrent->category->game_meta && isset($torrent->meta) && $meta->cover->image_id && $meta->name) + src="https://images.igdb.com/igdb/image/upload/t_cover_big/{{ $torrent->meta->cover->image_id }}.jpg" + + @break + @case($torrent->category->music_meta) {{--Could combine music_meta||no_meta , left seperate to ease special handling if needed--}} + @php + $coverPath = 'files/img/torrent-cover_' . $torrent->id; + $coverFile = collect(glob(public_path($coverPath . '.*')))->first(); + $imageSrc = $coverFile !== null + ? asset(str_replace(public_path(), '', $coverFile)) + : ($torrent->cover_url && filter_var($torrent->cover_url, FILTER_VALIDATE_URL) + ? $torrent->cover_url + : 'https://via.placeholder.com/160x240'); + @endphp + src="{{ $imageSrc }}" + + @break + @case($torrent->category->no_meta) + @php + $coverPath = 'files/img/torrent-cover_' . $torrent->id; + $coverFile = collect(glob(public_path($coverPath . '.*')))->first(); + $imageSrc = $coverFile !== null + ? asset(str_replace(public_path(), '', $coverFile)) + : ($torrent->cover_url && filter_var($torrent->cover_url, FILTER_VALIDATE_URL) + ? $torrent->cover_url + : 'https://via.placeholder.com/160x240'); + @endphp + src="{{ $imageSrc }}" + @break @endswitch alt="{{ __('torrent.similar') }}" diff --git a/resources/views/components/torrent/row.blade.php b/resources/views/components/torrent/row.blade.php index 1dcca7632f..f225b0bae4 100644 --- a/resources/views/components/torrent/row.blade.php +++ b/resources/views/components/torrent/row.blade.php @@ -46,8 +46,17 @@ class="torrent-search--list__poster-img" @endif @if ($torrent->category->music_meta) + @php + $coverPath = 'files/img/torrent-cover_' . $torrent->id; + $coverFile = collect(glob(public_path($coverPath . '.*')))->first(); + $imageSrc = $coverFile !== null + ? asset(str_replace(public_path(), '', $coverFile)) + : ($torrent->cover_url && filter_var($torrent->cover_url, FILTER_VALIDATE_URL) + ? $torrent->cover_url + : 'https://via.placeholder.com/90x135'); + @endphp {{ __('torrent.similar') }}category->no_meta) - @if (file_exists(public_path() . '/files/img/torrent-cover_' . $torrent->id . '.jpg')) - {{ __('torrent.similar') }} - @else - {{ __('torrent.similar') }} - @endif + @php + $coverPath = 'files/img/torrent-cover_' . $torrent->id; + $coverFile = collect(glob(public_path($coverPath . '.*')))->first(); + $imageSrc = $coverFile !== null + ? asset(str_replace(public_path(), '', $coverFile)) + : ($torrent->cover_url && filter_var($torrent->cover_url, FILTER_VALIDATE_URL) + ? $torrent->cover_url + : 'https://via.placeholder.com/400x600'); + @endphp + {{ __('torrent.similar') }} @endif diff --git a/resources/views/torrent/download_check.blade.php b/resources/views/torrent/download_check.blade.php index 0763d25bac..5717282190 100644 --- a/resources/views/torrent/download_check.blade.php +++ b/resources/views/torrent/download_check.blade.php @@ -1,4 +1,4 @@ -@extends('layout.default') +Ï@extends('layout.default') @section('title') {{ __('torrent.download-check') }} - {{ config('other.title') }} diff --git a/resources/views/torrent/partials/no_meta.blade.php b/resources/views/torrent/partials/no_meta.blade.php index da6728ed34..9d265a3cda 100644 --- a/resources/views/torrent/partials/no_meta.blade.php +++ b/resources/views/torrent/partials/no_meta.blade.php @@ -1,17 +1,24 @@ +
- @if (file_exists(public_path() . '/files/img/torrent-banner_' . $torrent->id . '.jpg')) - - @endif + @php + $bannerPath = 'files/img/torrent-banner_' . $torrent->id; + $coverPath = 'files/img/torrent-cover_' . $torrent->id; + + function setImageSrc($path, $urlField, $placeholder) { + $file = collect(glob(public_path($path . '.*')))->first(); + return $file + ? asset(str_replace(public_path(), '', $file)) + : ($urlField && filter_var($urlField, FILTER_VALIDATE_URL) ? $urlField : $placeholder); + } + + $bannerSrc = setImageSrc($bannerPath, $torrent->banner_url, 'https://via.placeholder.com/1280x720'); + $coverSrc = setImageSrc($coverPath, $torrent->cover_url, 'https://via.placeholder.com/400x600'); + @endphp + + Banner Image - + Cover Image