mirror of
https://github.com/koel/koel
synced 2025-01-02 07:48:44 +00:00
172 lines
4.5 KiB
PHP
172 lines
4.5 KiB
PHP
<?php
|
|
|
|
namespace App\Models;
|
|
|
|
use App\Builders\SongBuilder;
|
|
use Carbon\Carbon;
|
|
use Illuminate\Database\Eloquent\Casts\Attribute;
|
|
use Illuminate\Database\Eloquent\Factories\HasFactory;
|
|
use Illuminate\Database\Eloquent\Model;
|
|
use Illuminate\Database\Eloquent\Relations\BelongsTo;
|
|
use Illuminate\Database\Eloquent\Relations\BelongsToMany;
|
|
use Illuminate\Database\Eloquent\Relations\HasMany;
|
|
use Illuminate\Support\Str;
|
|
use Laravel\Scout\Searchable;
|
|
|
|
/**
|
|
* @property string $path
|
|
* @property string $title
|
|
* @property Album $album
|
|
* @property User $uploader
|
|
* @property Artist $artist
|
|
* @property Artist $album_artist
|
|
* @property float $length
|
|
* @property string $lyrics
|
|
* @property int $track
|
|
* @property int $disc
|
|
* @property int $album_id
|
|
* @property int|null $year
|
|
* @property string $genre
|
|
* @property string $id
|
|
* @property int $artist_id
|
|
* @property int $mtime
|
|
* @property ?bool $liked Whether the song is liked by the current user (dynamically calculated)
|
|
* @property ?int $play_count The number of times the song has been played by the current user (dynamically calculated)
|
|
* @property Carbon $created_at
|
|
* @property array<mixed> $s3_params
|
|
* @property int $owner_id
|
|
* @property bool $is_public
|
|
* @property User $owner
|
|
*/
|
|
class Song extends Model
|
|
{
|
|
use HasFactory;
|
|
use Searchable;
|
|
use SupportsDeleteWhereValueNotIn;
|
|
|
|
public const ID_REGEX = '[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}';
|
|
|
|
public $incrementing = false;
|
|
protected $guarded = [];
|
|
|
|
protected $hidden = ['updated_at', 'path', 'mtime'];
|
|
|
|
protected $casts = [
|
|
'length' => 'float',
|
|
'mtime' => 'int',
|
|
'track' => 'int',
|
|
'disc' => 'int',
|
|
'is_public' => 'bool',
|
|
];
|
|
|
|
protected $keyType = 'string';
|
|
|
|
protected static function booted(): void
|
|
{
|
|
static::creating(static fn (self $song) => $song->id = Str::uuid()->toString());
|
|
}
|
|
|
|
public static function query(): SongBuilder
|
|
{
|
|
return parent::query();
|
|
}
|
|
|
|
public function owner(): BelongsTo
|
|
{
|
|
return $this->belongsTo(User::class);
|
|
}
|
|
|
|
public function newEloquentBuilder($query): SongBuilder
|
|
{
|
|
return new SongBuilder($query);
|
|
}
|
|
|
|
public function artist(): BelongsTo
|
|
{
|
|
return $this->belongsTo(Artist::class);
|
|
}
|
|
|
|
public function album(): BelongsTo
|
|
{
|
|
return $this->belongsTo(Album::class);
|
|
}
|
|
|
|
public function album_artist(): BelongsTo // @phpcs:ignore
|
|
{
|
|
return $this->album->artist();
|
|
}
|
|
|
|
public function playlists(): BelongsToMany
|
|
{
|
|
return $this->belongsToMany(Playlist::class);
|
|
}
|
|
|
|
public function interactions(): HasMany
|
|
{
|
|
return $this->hasMany(Interaction::class);
|
|
}
|
|
|
|
protected function title(): Attribute
|
|
{
|
|
return new Attribute(
|
|
get: fn (?string $value) => $value ?: pathinfo($this->path, PATHINFO_FILENAME),
|
|
set: static fn (string $value) => html_entity_decode($value)
|
|
);
|
|
}
|
|
|
|
public function accessibleBy(User $user): bool
|
|
{
|
|
return $this->is_public || $this->owner_id === $user->id;
|
|
}
|
|
|
|
protected function lyrics(): Attribute
|
|
{
|
|
$normalizer = static function (?string $value): string {
|
|
// Since we're displaying the lyrics using <pre>, replace breaks with newlines and strip all tags.
|
|
$value = strip_tags(preg_replace('#<br\s*/?>#i', PHP_EOL, $value));
|
|
|
|
// also remove the timestamps that often come with LRC files
|
|
return preg_replace('/\[\d{2}:\d{2}.\d{2}]\s*/m', '', $value);
|
|
};
|
|
|
|
return new Attribute(get: $normalizer, set: $normalizer);
|
|
}
|
|
|
|
protected function s3Params(): Attribute
|
|
{
|
|
return Attribute::get(function (): ?array {
|
|
if (!preg_match('/^s3:\\/\\/(.*)/', $this->path, $matches)) {
|
|
return null;
|
|
}
|
|
|
|
[$bucket, $key] = explode('/', $matches[1], 2);
|
|
|
|
return compact('bucket', 'key');
|
|
});
|
|
}
|
|
|
|
public static function getPathFromS3BucketAndKey(string $bucket, string $key): string
|
|
{
|
|
return "s3://$bucket/$key";
|
|
}
|
|
|
|
/** @return array<mixed> */
|
|
public function toSearchableArray(): array
|
|
{
|
|
$array = [
|
|
'id' => $this->id,
|
|
'title' => $this->title,
|
|
];
|
|
|
|
if (!$this->artist->is_unknown && !$this->artist->is_various) {
|
|
$array['artist'] = $this->artist->name;
|
|
}
|
|
|
|
return $array;
|
|
}
|
|
|
|
public function __toString(): string
|
|
{
|
|
return $this->id;
|
|
}
|
|
}
|