filePath = $file->getRealPath(); $this->song = $this->songRepository->getOneByPath($this->filePath); $this->fileModifiedTime = Helper::getModifiedTime($file); return $this; } public function getFileScanInformation(): ?SongScanInformation { $raw = $this->getID3->analyze($this->filePath); $this->syncError = Arr::get($raw, 'error.0') ?: (Arr::get($raw, 'playtime_seconds') ? null : 'Empty file'); if ($this->syncError) { return null; } $this->getID3->CopyTagsToComments($raw); $info = SongScanInformation::fromGetId3Info($raw); $info->lyrics = $info->lyrics ?: $this->lrcReader->tryReadForMediaFile($this->filePath); return $info; } /** * Sync the song with all available media info into the database. * * @param array $ignores The tags to ignore/exclude (only taken into account if the song already exists) * @param bool $force Whether to force syncing, even if the file is unchanged */ public function sync(array $ignores = [], bool $force = false): SyncResult { if (!$this->isFileNewOrChanged() && !$force) { return SyncResult::skipped($this->filePath); } $info = $this->getFileScanInformation()?->toArray(); if (!$info) { return SyncResult::error($this->filePath, $this->syncError); } if (!$this->isFileNew()) { Arr::forget($info, $ignores); } $artist = Arr::get($info, 'artist') ? Artist::getOrCreate($info['artist']) : $this->song->artist; $albumArtist = Arr::get($info, 'albumartist') ? Artist::getOrCreate($info['albumartist']) : $artist; $album = Arr::get($info, 'album') ? Album::getOrCreate($albumArtist, $info['album']) : $this->song->album; if (!$album->has_cover) { $this->tryGenerateAlbumCover($album, Arr::get($info, 'cover', [])); } $data = Arr::except($info, ['album', 'artist', 'albumartist', 'cover']); $data['album_id'] = $album->id; $data['artist_id'] = $artist->id; $this->song = Song::query()->updateOrCreate(['path' => $this->filePath], $data); // @phpstan-ignore-line return SyncResult::success($this->filePath); } /** * Try to generate a cover for an album based on extracted data, or use the cover file under the directory. * * @param array|null $coverData */ private function tryGenerateAlbumCover(Album $album, ?array $coverData): void { attempt(function () use ($album, $coverData): void { // If the album has no cover, we try to get the cover image from existing tag data if ($coverData) { $extension = explode('/', $coverData['image_mime']); $extension = $extension[1] ?? 'png'; $this->mediaMetadataService->writeAlbumCover($album, $coverData['data'], $extension); return; } // Or, if there's a cover image under the same directory, use it. $cover = $this->getCoverFileUnderSameDirectory(); if ($cover) { $extension = pathinfo($cover, PATHINFO_EXTENSION); $this->mediaMetadataService->writeAlbumCover($album, $cover, $extension); } }, false); } /** * Issue #380. * Some albums have its own cover image under the same directory as cover|folder.jpg/png. * We'll check if such a cover file is found, and use it if positive. */ private function getCoverFileUnderSameDirectory(): ?string { // As directory scanning can be expensive, we cache and reuse the result. return $this->cache->remember(md5($this->filePath . '_cover'), now()->addDay(), function (): ?string { $matches = array_keys( iterator_to_array( $this->finder->create() ->depth(0) ->ignoreUnreadableDirs() ->files() ->followLinks() ->name('/(cov|fold)er\.(jpe?g|png)$/i') ->in(dirname($this->filePath)) ) ); $cover = $matches ? $matches[0] : null; return $cover && self::isImage($cover) ? $cover : null; }); } private static function isImage(string $path): bool { return attempt(static fn () => (bool) exif_imagetype($path)) ?? false; } /** * Determine if the file is new (its Song record can't be found in the database). */ public function isFileNew(): bool { return !$this->song; } /** * Determine if the file is changed (its Song record is found, but the timestamp is different). */ public function isFileChanged(): bool { return !$this->isFileNew() && $this->song->mtime !== $this->fileModifiedTime; } public function isFileNewOrChanged(): bool { return $this->isFileNew() || $this->isFileChanged(); } public function getSong(): ?Song { return $this->song; } }