koel/app/Services/MediaSyncService.php

186 lines
5.7 KiB
PHP
Raw Normal View History

2015-12-13 04:42:28 +00:00
<?php
namespace App\Services;
2019-06-30 11:20:40 +00:00
use App\Console\Commands\SyncCommand;
2016-02-04 15:48:15 +00:00
use App\Events\LibraryChanged;
use App\Events\MediaSyncCompleted;
2016-02-04 15:04:53 +00:00
use App\Libraries\WatchRecord\WatchRecordInterface;
2018-08-31 13:47:15 +00:00
use App\Models\Setting;
2015-12-13 04:42:28 +00:00
use App\Models\Song;
2018-08-29 06:15:11 +00:00
use App\Repositories\SongRepository;
use App\Values\SyncResult;
use Psr\Log\LoggerInterface;
2018-08-24 15:27:19 +00:00
use SplFileInfo;
2016-02-04 15:48:15 +00:00
use Symfony\Component\Finder\Finder;
2015-12-13 04:42:28 +00:00
2018-08-19 15:26:34 +00:00
class MediaSyncService
2015-12-13 04:42:28 +00:00
{
2018-08-29 06:15:11 +00:00
public function __construct(
2022-06-10 10:47:46 +00:00
private SongRepository $songRepository,
private FileSynchronizer $fileSynchronizer,
private Finder $finder,
private LoggerInterface $logger
2018-08-29 07:07:44 +00:00
) {
2018-08-19 09:05:33 +00:00
}
2016-03-22 08:22:39 +00:00
/**
* @param array<string> $excludes The tags to exclude.
2020-12-22 20:11:22 +00:00
* Only taken into account for existing records.
* New records will have all tags synced in regardless.
* @param bool $force Whether to force syncing even unchanged files
* @param SyncCommand $syncCommand The SyncMedia command object, to log to console if executed by artisan
2015-12-13 04:42:28 +00:00
*/
2018-08-24 15:27:19 +00:00
public function sync(
?string $mediaPath = null,
array $excludes = [],
2018-08-24 15:27:19 +00:00
bool $force = false,
2019-06-30 11:20:40 +00:00
?SyncCommand $syncCommand = null
2018-08-24 15:27:19 +00:00
): void {
$this->setSystemRequirements();
2015-12-13 04:42:28 +00:00
$syncResult = SyncResult::init();
2018-08-31 13:47:15 +00:00
$songPaths = $this->gatherFiles($mediaPath ?: Setting::get('media_path'));
$syncCommand?->createProgressBar(count($songPaths));
2017-06-03 23:21:50 +00:00
foreach ($songPaths as $path) {
$result = $this->fileSynchronizer->setFile($path)->sync($excludes, $force);
2017-06-03 23:21:50 +00:00
switch ($result) {
case FileSynchronizer::SYNC_RESULT_SUCCESS:
$syncResult->success->add($path);
2017-06-03 23:21:50 +00:00
break;
2020-12-22 20:11:22 +00:00
case FileSynchronizer::SYNC_RESULT_UNMODIFIED:
$syncResult->unmodified->add($path);
2017-06-03 23:21:50 +00:00
break;
2020-12-22 20:11:22 +00:00
2017-06-03 23:21:50 +00:00
default:
$syncResult->bad->add($path);
2017-06-03 23:21:50 +00:00
break;
2015-12-13 04:42:28 +00:00
}
if ($syncCommand) {
2018-08-24 15:27:19 +00:00
$syncCommand->advanceProgressBar();
$syncCommand->logSyncStatusToConsole($path, $result, $this->fileSynchronizer->getSyncError());
2015-12-13 04:42:28 +00:00
}
}
event(new MediaSyncCompleted($syncResult));
2015-12-13 04:42:28 +00:00
2021-12-10 15:27:06 +00:00
// Trigger LibraryChanged, so that PruneLibrary handler is fired to prune the lib.
2016-02-02 07:47:00 +00:00
event(new LibraryChanged());
}
2015-12-13 04:42:28 +00:00
2016-02-02 07:47:00 +00:00
/**
* Gather all applicable files in a given directory.
*
* @param string $path The directory's full path
*
2020-12-22 20:11:22 +00:00
* @return array<SplFileInfo>
2016-02-02 07:47:00 +00:00
*/
2018-08-24 15:27:19 +00:00
public function gatherFiles(string $path): array
2016-02-02 07:47:00 +00:00
{
2017-04-23 16:01:02 +00:00
return iterator_to_array(
$this->finder->create()
2017-04-23 16:01:02 +00:00
->ignoreUnreadableDirs()
->ignoreDotFiles((bool) config('koel.ignore_dot_files')) // https://github.com/koel/koel/issues/450
2017-04-23 16:01:02 +00:00
->files()
->followLinks()
->name('/\.(mp3|ogg|m4a|flac)$/i')
->in($path)
);
2015-12-13 04:42:28 +00:00
}
2018-08-24 15:27:19 +00:00
public function syncByWatchRecord(WatchRecordInterface $record): void
2016-02-02 07:47:00 +00:00
{
$this->logger->info("New watch record received: '{$record->getPath()}'");
2017-06-03 23:21:50 +00:00
$record->isFile() ? $this->syncFileRecord($record) : $this->syncDirectoryRecord($record);
}
2016-02-02 07:47:00 +00:00
2018-08-24 15:27:19 +00:00
private function syncFileRecord(WatchRecordInterface $record): void
2017-06-03 23:21:50 +00:00
{
$path = $record->getPath();
2018-08-31 13:47:15 +00:00
$this->logger->info("'$path' is a file.");
2017-01-06 03:04:08 +00:00
2017-06-03 23:21:50 +00:00
if ($record->isDeleted()) {
$this->handleDeletedFileRecord($path);
2020-12-22 20:11:22 +00:00
} elseif ($record->isNewOrModified()) {
$this->handleNewOrModifiedFileRecord($path);
2016-02-02 07:47:00 +00:00
}
2017-06-03 23:21:50 +00:00
}
2016-02-02 07:47:00 +00:00
2018-08-24 15:27:19 +00:00
private function syncDirectoryRecord(WatchRecordInterface $record): void
2017-06-03 23:21:50 +00:00
{
$path = $record->getPath();
2018-08-31 13:47:15 +00:00
$this->logger->info("'$path' is a directory.");
2016-02-02 07:47:00 +00:00
2016-02-04 15:04:53 +00:00
if ($record->isDeleted()) {
$this->handleDeletedDirectoryRecord($path);
2016-02-04 15:04:53 +00:00
} elseif ($record->isNewOrModified()) {
$this->handleNewOrModifiedDirectoryRecord($path);
2016-02-02 07:47:00 +00:00
}
}
private function setSystemRequirements(): void
{
if (!app()->runningInConsole()) {
set_time_limit(config('koel.sync.timeout'));
}
if (config('koel.memory_limit')) {
2020-12-22 20:11:22 +00:00
ini_set('memory_limit', config('koel.memory_limit') . 'M');
}
}
private function handleDeletedFileRecord(string $path): void
{
2020-12-22 20:11:22 +00:00
$song = $this->songRepository->getOneByPath($path);
if ($song) {
$song->delete();
2018-08-31 13:47:15 +00:00
$this->logger->info("$path deleted.");
event(new LibraryChanged());
} else {
2018-08-31 13:47:15 +00:00
$this->logger->info("$path doesn't exist in our database--skipping.");
}
}
private function handleNewOrModifiedFileRecord(string $path): void
{
$result = $this->fileSynchronizer->setFile($path)->sync();
if ($result === FileSynchronizer::SYNC_RESULT_SUCCESS) {
2018-08-31 13:47:15 +00:00
$this->logger->info("Synchronized $path");
} else {
2018-08-31 13:47:15 +00:00
$this->logger->info("Failed to synchronized $path. Maybe an invalid file?");
}
event(new LibraryChanged());
}
private function handleDeletedDirectoryRecord(string $path): void
{
2020-12-22 20:11:22 +00:00
$count = Song::inDirectory($path)->delete();
if ($count) {
2018-08-31 13:47:15 +00:00
$this->logger->info("Deleted $count song(s) under $path");
event(new LibraryChanged());
} else {
2018-08-31 13:47:15 +00:00
$this->logger->info("$path is empty--no action needed.");
}
}
private function handleNewOrModifiedDirectoryRecord(string $path): void
{
foreach ($this->gatherFiles($path) as $file) {
$this->fileSynchronizer->setFile($file)->sync();
}
2018-08-31 13:47:15 +00:00
$this->logger->info("Synced all song(s) under $path");
event(new LibraryChanged());
}
2015-12-13 04:42:28 +00:00
}