Files
UNIT3D/app/Http/Controllers/API/QuickSearchController.php
Roardom 26e27f7300 fix: quicksearch searching by null throws error
The middleware automatically converts empty strings to null. We want to accept empty strings here for the search to work. Currently, the regex throws an error when a null is passed to it.
2025-07-22 09:26:21 +00:00

147 lines
5.2 KiB
PHP

<?php
declare(strict_types=1);
/**
* NOTICE OF LICENSE.
*
* UNIT3D Community Edition is open-sourced software licensed under the GNU Affero General Public License v3.0
* The details is bundled with this project in the file LICENSE.txt.
*
* @project UNIT3D Community Edition
*
* @author HDVinnie <hdinnovations@protonmail.com>
* @license https://www.gnu.org/licenses/agpl-3.0.en.html/ GNU Affero General Public License v3.0
*/
namespace App\Http\Controllers\API;
use App\Enums\ModerationStatus;
use App\Http\Controllers\Controller;
use Illuminate\Http\Request;
use Meilisearch\Client;
use Meilisearch\Contracts\MultiSearchFederation;
use Meilisearch\Contracts\SearchQuery;
class QuickSearchController extends Controller
{
public function index(Request $request): \Illuminate\Http\JsonResponse
{
$query = $request->input('query', '');
$filters = [
'deleted_at IS NULL',
'status = '.ModerationStatus::APPROVED->value,
[
'category.movie_meta = true',
'category.tv_meta = true',
],
[
'tmdb_movie.name EXISTS',
'tmdb_tv.name EXISTS',
],
[
'tmdb_movie_id IS NOT NULL AND tmdb_movie_id != 0',
'tmdb_tv_id IS NOT NULL AND tmdb_tv_id != 0',
]
];
// Check if the query is an IMDb or TMDB ID
$searchById = false;
if (preg_match('/^(\d+)$/', $query, $matches)) {
$filters[] = [
'tmdb_movie_id = '.$matches[1],
'tmdb_movie.name = '.$matches[1],
'tmdb_tv_id = '.$matches[1],
'tmdb_tv.name = '.$matches[1],
];
$searchById = true;
}
if (preg_match('/tt0*(\d{7,})/', $query, $matches)) {
$filters[] = 'imdb = '.$matches[1];
$searchById = true;
}
$client = new Client(config('scout.meilisearch.host'), config('scout.meilisearch.key'));
// Prepare the search queries
$searchQueries = [
(new SearchQuery())
->setIndexUid(config('scout.prefix').'torrents')
->setQuery($searchById ? '' : $query)
->setFilter($filters)
->setAttributesToRetrieve([
'id',
'name',
'tmdb_movie_id',
'tmdb_tv_id',
'category.id',
'category.name',
'category.movie_meta',
'category.tv_meta',
'tmdb_movie.name',
'tmdb_movie.year',
'tmdb_movie.poster',
'tmdb_tv.name',
'tmdb_tv.year',
'tmdb_tv.poster',
])
->setDistinct('imdb')
];
// Add the people search query only if it's not an ID search
if (!$searchById) {
$searchQueries[] = (new SearchQuery())
->setIndexUid(config('scout.prefix').'people')
->setQuery($query)
->setAttributesToRetrieve([
'id',
'name',
'birthday',
'still',
]);
}
// Perform multi-search with MultiSearchFederation
$searchQuery = fn () => $client->multiSearch($searchQueries, ((new MultiSearchFederation()))->setLimit(20));
if (preg_match("/^[a-zA-Z0-9-_ .'@:\\[\\]+&\\/,!#()?\"]{1,2}$/", $query)) {
$multiSearchResults = cache()->remember('quick-search:'.strtolower($query), 3600 * 24, $searchQuery);
} else {
$multiSearchResults = $searchQuery();
}
$results = [];
// Process the hits from the multiSearchResults
foreach ($multiSearchResults['hits'] as $hit) {
if ($hit['_federation']['indexUid'] === config('scout.prefix').'torrents') {
$type = $hit['category']['movie_meta'] === true ? 'tmdb_movie' : 'tmdb_tv';
$results[] = [
'id' => $hit['id'],
'name' => $hit[$type]['name'],
'year' => $hit[$type]['year'],
'image' => $hit[$type]['poster'] ? tmdb_image('poster_small', $hit[$type]['poster']) : mb_substr($hit['name'], 0, 2),
'url' => route('torrents.similar', ['category_id' => $hit['category']['id'], 'tmdb' => $hit["{$type}_id"]]),
'type' => $hit['category']['name'],
];
} elseif ($hit['_federation']['indexUid'] === config('scout.prefix').'people') {
$results[] = [
'id' => $hit['id'],
'name' => $hit['name'],
'year' => $hit['birthday'],
'image' => $hit['still'] ? tmdb_image('poster_small', $hit['still']) : mb_substr($hit['name'], 0, 1).mb_substr(str($hit['name'])->explode(' ')->last() ?? '', 0, 1),
'url' => route('mediahub.persons.show', ['id' => $hit['id']]),
'type' => 'Person',
];
}
}
return response()->json(['results' => $results]);
}
}