Switch Requests to use Meilisearch.
This commit is contained in:
parent
81d16d619a
commit
2586277e9f
|
@ -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
|
||||||
|
|
|
@ -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]
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -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,
|
||||||
|
|
Loading…
Reference in New Issue