Switch Requests to use Meilisearch.

This commit is contained in:
Buster Neece 2023-01-29 08:38:59 -06:00
parent 81d16d619a
commit 2586277e9f
No known key found for this signature in database
GPG Key ID: F1D2E64A0005E80E
3 changed files with 64 additions and 79 deletions

View File

@ -32,7 +32,7 @@
</div> </div>
<data-table <data-table
id="station_on_demand_table" id="public_on_demand"
ref="datatable" ref="datatable"
paginated paginated
select-fields select-fields

View File

@ -1,7 +1,7 @@
<template> <template>
<div style="overflow-x: hidden"> <div style="overflow-x: hidden">
<data-table <data-table
id="song_requests" id="public_requests"
ref="datatable" ref="datatable"
paginated paginated
select-fields select-fields
@ -74,46 +74,51 @@ const fields = computed(() => {
key: 'name', key: 'name',
isRowHeader: true, isRowHeader: true,
label: $gettext('Name'), label: $gettext('Name'),
sortable: true, sortable: false,
selectable: true selectable: true
}, },
{ {
key: 'song.title', key: 'title',
label: $gettext('Title'), label: $gettext('Title'),
sortable: true, sortable: true,
selectable: true, selectable: true,
visible: false, visible: false,
formatter: (value, key, item) => item.song.title
}, },
{ {
key: 'song.artist', key: 'artist',
label: $gettext('Artist'), label: $gettext('Artist'),
sortable: true, sortable: true,
selectable: true, selectable: true,
visible: false, visible: false,
formatter: (value, key, item) => item.song.artist
}, },
{ {
key: 'song.album', key: 'album',
label: $gettext('Album'), label: $gettext('Album'),
sortable: true, sortable: true,
selectable: true, selectable: true,
visible: false visible: false,
formatter: (value, key, item) => item.song.album
}, },
{ {
key: 'song.genre', key: 'genre',
label: $gettext('Genre'), label: $gettext('Genre'),
sortable: true, sortable: true,
selectable: true, selectable: true,
visible: false visible: false,
formatter: (value, key, item) => item.song.genre
} }
]; ];
forEach({...props.customFields}, (field) => { forEach({...props.customFields}, (field) => {
fields.push({ fields.push({
key: 'song.custom_fields.' + field.short_name, key: 'custom_field_' + field.id,
label: field.name, label: field.name,
sortable: false, sortable: false,
selectable: true, selectable: true,
visible: false visible: false,
formatter: (value, key, item) => item.song.custom_fields[field.short_name]
}); });
}); });

View File

@ -10,8 +10,7 @@ use App\Http\Response;
use App\Http\ServerRequest; use App\Http\ServerRequest;
use App\OpenApi; use App\OpenApi;
use App\Paginator; use App\Paginator;
use App\Radio\AutoDJ\Scheduler; use App\Service\Meilisearch;
use Carbon\CarbonImmutable;
use Doctrine\ORM\EntityManagerInterface; use Doctrine\ORM\EntityManagerInterface;
use OpenApi\Attributes as OA; use OpenApi\Attributes as OA;
use Psr\Http\Message\ResponseInterface; use Psr\Http\Message\ResponseInterface;
@ -62,13 +61,13 @@ use Psr\Http\Message\ResponseInterface;
] ]
) )
] ]
final class RequestsController final readonly class RequestsController
{ {
public function __construct( public function __construct(
private readonly EntityManagerInterface $em, private EntityManagerInterface $em,
private readonly Entity\Repository\StationRequestRepository $requestRepo, private Entity\Repository\StationRequestRepository $requestRepo,
private readonly Entity\ApiGenerator\SongApiGenerator $songApiGenerator, private Entity\ApiGenerator\SongApiGenerator $songApiGenerator,
private readonly Scheduler $scheduler private Meilisearch $meilisearch
) { ) {
} }
@ -79,60 +78,68 @@ final class RequestsController
): ResponseInterface { ): ResponseInterface {
$station = $request->getStation(); $station = $request->getStation();
$playlistIds = $this->getRequestablePlaylists($station); // Verify that the station supports on-demand streaming.
if (!$station->getEnableRequests()) {
return $response->withStatus(403)
->withJson(new Entity\Api\Error(403, __('This station does not support requests.')));
}
$qb = $this->em->createQueryBuilder(); if (!$this->meilisearch->isSupported()) {
$qb->select('sm, spm, sp') return $response->withStatus(403)
->from(Entity\StationMedia::class, 'sm') ->withJson(new Entity\Api\Error(403, __('This feature is not supported on this installation.')));
->leftJoin('sm.playlists', 'spm') }
->leftJoin('spm.playlist', 'sp')
->where('sm.storage_location = :storageLocation') $index = $this->meilisearch->getIndex($station->getMediaStorageLocation());
->andWhere('sp.id IN (:playlistIds)')
->setParameter('storageLocation', $station->getMediaStorageLocation())
->setParameter('playlistIds', $playlistIds);
$queryParams = $request->getQueryParams(); $queryParams = $request->getQueryParams();
$searchPhrase = trim($queryParams['searchPhrase'] ?? '');
$searchParams = [];
if (!empty($queryParams['sort'])) { if (!empty($queryParams['sort'])) {
$sortDirection = (($queryParams['sortOrder'] ?? 'ASC') === 'ASC') ? 'ASC' : 'DESC'; $sortField = (string)$queryParams['sort'];
$sortDirection = strtolower($queryParams['sortOrder'] ?? 'asc');
match ($queryParams['sort']) { $searchParams['sort'] = [$sortField . ':' . $sortDirection];
'name', 'song_title' => $qb->addOrderBy('sm.title', $sortDirection),
'song_artist' => $qb->addOrderBy('sm.artist', $sortDirection),
'song_album' => $qb->addOrderBy('sm.album', $sortDirection),
'song_genre' => $qb->addOrderBy('sm.genre', $sortDirection),
default => null,
};
} else {
$qb->orderBy('sm.artist', 'ASC')
->addOrderBy('sm.title', 'ASC');
} }
$search_phrase = trim($queryParams['searchPhrase'] ?? ''); $hydrateCallback = function (array $results) {
if (!empty($search_phrase)) { $ids = array_column($results, 'id');
$qb->andWhere('(sm.title LIKE :query OR sm.artist LIKE :query OR sm.album LIKE :query)')
->setParameter('query', '%' . $search_phrase . '%');
}
$paginator = Paginator::fromQueryBuilder($qb, $request); return $this->em->createQuery(
<<<'DQL'
SELECT sm
FROM App\Entity\StationMedia sm
WHERE sm.id IN (:ids)
ORDER BY FIELD(sm.id, :ids)
DQL
)->setParameter('ids', $ids)
->toIterable();
};
$paginatorAdapter = $index->getOnDemandSearchPaginator(
$station,
$hydrateCallback,
$searchPhrase,
$searchParams,
);
$paginator = Paginator::fromAdapter($paginatorAdapter, $request);
$router = $request->getRouter(); $router = $request->getRouter();
$baseUrl = $router->getBaseUrl();
$paginator->setPostprocessor( $paginator->setPostprocessor(
function (Entity\StationMedia $media_row) use ($station, $baseUrl, $router) { function (Entity\StationMedia $media) use ($station, $router) {
$row = new Entity\Api\StationRequest(); $row = new Entity\Api\StationRequest();
$row->song = ($this->songApiGenerator)($media_row, $station, $baseUrl); $row->song = ($this->songApiGenerator)($media, $station, $router->getBaseUrl());
$row->request_id = $media_row->getUniqueId(); $row->request_id = $media->getUniqueId();
$row->request_url = $router->named( $row->request_url = $router->named(
'api:requests:submit', 'api:requests:submit',
[ [
'station_id' => $station->getId(), 'station_id' => $station->getId(),
'media_id' => $media_row->getUniqueId(), 'media_id' => $media->getUniqueId(),
] ]
); );
$row->resolveUrls($baseUrl); $row->resolveUrls($router->getBaseUrl());
return $row; return $row;
} }
@ -141,33 +148,6 @@ final class RequestsController
return $paginator->write($response); return $paginator->write($response);
} }
/**
* @param Entity\Station $station
*/
private function getRequestablePlaylists(Entity\Station $station): array
{
$playlists = $this->em->createQuery(
<<<DQL
SELECT sp FROM App\Entity\StationPlaylist sp
WHERE sp.station = :station
AND sp.is_enabled = 1 AND sp.include_in_requests = 1
DQL
)->setParameter('station', $station)
->toIterable();
$ids = [];
$now = CarbonImmutable::now($station->getTimezoneObject());
/** @var Entity\StationPlaylist $playlist */
foreach ($playlists as $playlist) {
if ($this->scheduler->isPlaylistScheduledToPlayNow($playlist, $now)) {
$ids[] = $playlist->getIdRequired();
}
}
return $ids;
}
public function submitAction( public function submitAction(
ServerRequest $request, ServerRequest $request,
Response $response, Response $response,