koel/app/Models/File.php

307 lines
8.9 KiB
PHP
Raw Normal View History

2016-03-22 08:22:39 +00:00
<?php
namespace App\Models;
use Exception;
use getID3;
use getid3_lib;
use Illuminate\Support\Facades\Log;
use SplFileInfo;
class File
{
/**
* A MD5 hash of the file's path.
* This value is unique, and can be used to query a Song record.
*
* @var string
*/
protected $hash;
/**
* The file's last modified time.
*
* @var int
*/
protected $mtime;
/**
* The file's path.
*
* @var string
*/
protected $path;
/**
* The getID3 object, for ID3 tag reading.
*
* @var getID3
*/
protected $getID3;
/**
* The SplFileInfo object of the file.
*
* @var SplFileInfo
*/
protected $splFileInfo;
/**
* The song model that's associated with this file.
*
* @var Song
*/
protected $song;
/**
* Construct our File object.
2016-03-27 02:54:22 +00:00
* Upon construction, we'll set the path, hash, and associated Song object (if any).
2016-03-22 08:22:39 +00:00
*
* @param string|SplFileInfo $path Either the file's path, or a SplFileInfo object
* @param getID3 $getID3 A getID3 object for DI (and better performance)
*/
public function __construct($path, $getID3 = null)
{
2016-04-05 07:38:10 +00:00
$this->splFileInfo = $path instanceof SplFileInfo ? $path : new SplFileInfo($path);
2016-03-22 08:22:39 +00:00
$this->setGetID3($getID3);
$this->mtime = $this->splFileInfo->getMTime();
$this->path = $this->splFileInfo->getPathname();
$this->hash = self::getHash($this->path);
$this->song = Song::find($this->hash);
}
/**
* Get all applicable ID3 info from the file.
*
* @return array|void
*/
public function getInfo()
{
$info = $this->getID3->analyze($this->path);
2016-04-17 15:38:06 +00:00
if (isset($info['error']) || !isset($info['playtime_seconds'])) {
2016-03-22 08:22:39 +00:00
return;
}
// Copy the available tags over to comment.
// This is a helper from getID3, though it doesn't really work well.
// We'll still prefer getting ID3v2 tags directly later.
// Read on.
getid3_lib::CopyTagsToComments($info);
$track = 0;
2016-03-24 03:06:28 +00:00
// Apparently track number can be stored with different indices as the following.
$trackIndices = [
'comments.track',
'comments.tracknumber',
'comments.track_number',
];
2016-03-24 03:06:28 +00:00
for ($i = 0; $i < count($trackIndices) && $track === 0; $i++) {
$track = array_get($info, $trackIndices[$i], [0])[0];
}
2016-03-22 08:22:39 +00:00
$props = [
'artist' => '',
'album' => '',
2016-04-17 15:38:06 +00:00
'part_of_a_compilation' => false,
2016-03-22 08:22:39 +00:00
'title' => '',
'length' => $info['playtime_seconds'],
2016-03-24 03:06:28 +00:00
'track' => (int) $track,
2016-03-22 08:22:39 +00:00
'lyrics' => '',
'cover' => array_get($info, 'comments.picture', [null])[0],
'path' => $this->path,
'mtime' => $this->mtime,
];
if (!$comments = array_get($info, 'comments_html')) {
return $props;
}
2016-04-17 15:38:06 +00:00
// getID3's 'part_of_a_compilation' value can either be null, ['0'], or ['1']
// We convert it into a boolean here.
2016-05-21 11:32:24 +00:00
$props['part_of_a_compilation'] = (bool) array_get($comments, 'part_of_a_compilation', [false])[0];
2016-04-17 15:38:06 +00:00
2016-03-22 08:22:39 +00:00
// We prefer id3v2 tags over others.
if (!$artist = array_get($info, 'tags.id3v2.artist', [null])[0]) {
$artist = array_get($comments, 'artist', [''])[0];
}
if (!$album = array_get($info, 'tags.id3v2.album', [null])[0]) {
$album = array_get($comments, 'album', [''])[0];
}
if (!$title = array_get($info, 'tags.id3v2.title', [null])[0]) {
$title = array_get($comments, 'title', [''])[0];
}
if (!$lyrics = array_get($info, 'tags.id3v2.unsynchronised_lyric', [null])[0]) {
$lyrics = array_get($comments, 'unsynchronised_lyric', [''])[0];
}
2016-05-05 15:03:30 +00:00
// Fixes #323, where tag names can be htmlentities()'ed
$props['artist'] = html_entity_decode(trim($artist));
$props['album'] = html_entity_decode(trim($album));
$props['title'] = html_entity_decode(trim($title));
$props['lyrics'] = html_entity_decode(trim($lyrics));
2016-03-22 08:22:39 +00:00
return $this->info = $props;
}
/**
* Sync the song with all available media info against the database.
*
* @param array $tags The (selective) tags to sync (if the song exists)
2016-06-04 14:17:24 +00:00
* @param bool $force Whether to force syncing, even if the file is unchanged
2016-03-22 08:22:39 +00:00
*
* @return bool|Song A Song object on success,
* true if file exists but is unmodified,
* or false on an error.
*/
public function sync($tags, $force = false)
{
// If the file is not new or changed and we're not forcing update, don't do anything.
if (!$this->isNewOrChanged() && !$force) {
return true;
}
// If the file is invalid, don't do anything.
if (!$info = $this->getInfo()) {
return false;
}
$artist = null;
2016-03-22 08:22:39 +00:00
if ($this->isChanged() || $force) {
// This is a changed file, or the user is forcing updates.
// In such a case, the user must have specified a list of tags to sync.
// A sample command could be: ./artisan koel:sync --force --tags=artist,album,lyrics
// We cater for these tags by removing those not specified.
2016-06-04 18:17:27 +00:00
// There's a special case with 'album though'.
// If 'part_of_a_compilation' tag is specified, 'album' must be counted in as well.
// But if 'album' isn't specified, we don't want to update normal albums.
// This variable is to keep track of this state.
$changeCompilationAlbumOnly = false;
if (in_array('part_of_a_compilation', $tags) && !in_array('album', $tags)) {
$tags[] = 'album';
2016-06-04 18:17:27 +00:00
$changeCompilationAlbumOnly = true;
}
2016-03-22 08:22:39 +00:00
$info = array_intersect_key($info, array_flip($tags));
// If the "artist" tag is specified, use it.
// Otherwise, re-use the existing model value.
2016-03-22 08:22:39 +00:00
$artist = isset($info['artist']) ? Artist::get($info['artist']) : $this->song->album->artist;
2016-05-21 11:32:24 +00:00
$isCompilation = (bool) array_get($info, 'part_of_a_compilation');
// If the "album" tag is specified, use it.
// Otherwise, re-use the existing model value.
2016-06-04 18:17:27 +00:00
if (isset($info['album'])) {
$album = $changeCompilationAlbumOnly
? $this->song->album
: Album::get($artist, $info['album'], $isCompilation);
} else {
$album = $this->song->album;
}
2016-03-22 08:22:39 +00:00
} else {
// The file is newly added.
2016-05-21 11:32:24 +00:00
$isCompilation = (bool) array_get($info, 'part_of_a_compilation');
$artist = Artist::get($info['artist']);
$album = Album::get($artist, $info['album'], $isCompilation);
2016-03-22 08:22:39 +00:00
}
if (!empty($info['cover']) && !$album->has_cover) {
try {
$album->generateCover($info['cover']);
} catch (Exception $e) {
Log::error($e);
}
}
$info['album_id'] = $album->id;
// If the song is part of a compilation, make sure we properly set its
// artist and contributing artist attributes.
if ($isCompilation) {
$info['contributing_artist_id'] = $artist->id;
2016-04-17 15:38:06 +00:00
}
2016-03-24 03:26:52 +00:00
// Remove these values from the info array, so that we can just use the array as model's input data.
2016-04-17 15:38:06 +00:00
array_forget($info, ['artist', 'album', 'cover', 'part_of_a_compilation']);
2016-03-22 08:22:39 +00:00
$song = Song::updateOrCreate(['id' => $this->hash], $info);
$song->save();
return $song;
}
/**
* Determine if the file is new (its Song record can't be found in the database).
*
* @return bool
*/
public function isNew()
{
return !$this->song;
}
/**
* Determine if the file is changed (its Song record is found, but the timestamp is different).
*
* @return bool
*/
public function isChanged()
{
return !$this->isNew() && $this->song->mtime !== $this->mtime;
}
/**
* Determine if the file is new or changed.
*
* @return bool
*/
public function isNewOrChanged()
{
return $this->isNew() || $this->isChanged();
}
/**
* @return getID3
*/
public function getGetID3()
{
return $this->getID3;
}
/**
* @param getID3 $getID3
*/
public function setGetID3($getID3 = null)
{
$this->getID3 = $getID3 ?: new getID3();
}
/**
* @return string
*/
public function getPath()
{
return $this->path;
}
/**
* Get a unique hash from a file path.
*
* @param string $path
*
* @return string
*/
public static function getHash($path)
{
return md5(config('app.key').$path);
}
}