AzuraCast/src/Entity/StationRemote.php

426 lines
10 KiB
PHP

<?php
declare(strict_types=1);
namespace App\Entity;
use App\Radio\Adapters;
use App\Radio\Remote\AbstractRemote;
use App\Utilities;
use Doctrine\ORM\Mapping as ORM;
use GuzzleHttp\Psr7\Uri;
use Psr\Http\Message\UriInterface;
use Stringable;
use Symfony\Component\Validator\Constraints as Assert;
#[
ORM\Entity,
ORM\Table(name: 'station_remotes'),
Attributes\Auditable
]
class StationRemote implements
Stringable,
Interfaces\StationMountInterface,
Interfaces\StationCloneAwareInterface,
Interfaces\IdentifiableEntityInterface
{
use Traits\HasAutoIncrementId;
use Traits\TruncateStrings;
#[ORM\Column(nullable: false)]
protected int $station_id;
#[ORM\ManyToOne(inversedBy: 'remotes')]
#[ORM\JoinColumn(name: 'station_id', referencedColumnName: 'id', nullable: false, onDelete: 'CASCADE')]
protected Station $station;
#[ORM\Column(nullable: true)]
protected ?int $relay_id = null;
#[ORM\ManyToOne(inversedBy: 'remotes')]
#[ORM\JoinColumn(name: 'relay_id', referencedColumnName: 'id', nullable: true, onDelete: 'CASCADE')]
protected ?Relay $relay = null;
#[ORM\Column(length: 255, nullable: true)]
protected ?string $display_name = null;
#[ORM\Column]
protected bool $is_visible_on_public_pages = true;
#[ORM\Column(length: 50)]
#[Assert\Choice(choices: [Adapters::REMOTE_ICECAST, Adapters::REMOTE_SHOUTCAST1, Adapters::REMOTE_SHOUTCAST2])]
protected string $type;
#[ORM\Column]
protected bool $enable_autodj = false;
#[ORM\Column(length: 10, nullable: true)]
protected ?string $autodj_format = null;
#[ORM\Column(type: 'smallint', nullable: true)]
protected ?int $autodj_bitrate = null;
#[ORM\Column(length: 255, nullable: true)]
protected ?string $custom_listen_url = null;
#[ORM\Column(length: 255, nullable: false)]
protected string $url = '';
#[ORM\Column(length: 150, nullable: true)]
protected ?string $mount = null;
#[ORM\Column(length: 100, nullable: true)]
protected ?string $admin_password = null;
#[ORM\Column(type: 'smallint', nullable: true, options: ['unsigned' => true])]
protected ?int $source_port = null;
#[ORM\Column(length: 150, nullable: true)]
protected ?string $source_mount = null;
#[ORM\Column(length: 100, nullable: true)]
protected ?string $source_username = null;
#[ORM\Column(length: 100, nullable: true)]
protected ?string $source_password = null;
#[ORM\Column]
protected bool $is_public = false;
#[ORM\Column]
#[Attributes\AuditIgnore]
protected int $listeners_unique = 0;
#[ORM\Column]
#[Attributes\AuditIgnore]
protected int $listeners_total = 0;
public function __construct(Station $station)
{
$this->station = $station;
}
public function getStation(): Station
{
return $this->station;
}
public function setStation(Station $station): void
{
$this->station = $station;
}
public function getRelay(): ?Relay
{
return $this->relay;
}
public function setRelay(?Relay $relay): void
{
$this->relay = $relay;
}
public function getIsVisibleOnPublicPages(): bool
{
return $this->is_visible_on_public_pages;
}
public function setIsVisibleOnPublicPages(bool $is_visible_on_public_pages): void
{
$this->is_visible_on_public_pages = $is_visible_on_public_pages;
}
public function getEnableAutodj(): bool
{
return $this->enable_autodj;
}
public function setEnableAutodj(bool $enable_autodj): void
{
$this->enable_autodj = $enable_autodj;
}
public function getAutodjFormat(): ?string
{
return $this->autodj_format;
}
public function setAutodjFormat(string $autodj_format = null): void
{
$this->autodj_format = $autodj_format;
}
public function getAutodjBitrate(): ?int
{
return $this->autodj_bitrate;
}
public function setAutodjBitrate(int $autodj_bitrate = null): void
{
$this->autodj_bitrate = $autodj_bitrate;
}
public function getCustomListenUrl(): ?string
{
return $this->custom_listen_url;
}
public function setCustomListenUrl(?string $custom_listen_url = null): void
{
$this->custom_listen_url = $this->truncateNullableString($custom_listen_url);
}
public function getAutodjUsername(): ?string
{
return $this->getSourceUsername();
}
public function getSourceUsername(): ?string
{
return $this->source_username;
}
public function setSourceUsername(?string $source_username): void
{
$this->source_username = $this->truncateNullableString($source_username, 100);
}
public function getAutodjPassword(): ?string
{
$password = $this->getSourcePassword();
if (Adapters::REMOTE_SHOUTCAST2 === $this->getType()) {
$mount = $this->getSourceMount();
if (empty($mount)) {
$mount = $this->getMount();
}
if (!empty($mount)) {
$password .= ':#' . $mount;
}
}
return $password;
}
public function getSourcePassword(): ?string
{
return $this->source_password;
}
public function setSourcePassword(?string $source_password): void
{
$this->source_password = $this->truncateNullableString($source_password, 100);
}
public function getType(): string
{
return $this->type;
}
public function setType(string $type): void
{
$this->type = $type;
}
public function getSourceMount(): ?string
{
return $this->source_mount;
}
public function setSourceMount(?string $source_mount): void
{
$this->source_mount = $this->truncateNullableString($source_mount, 150);
}
public function getMount(): ?string
{
return $this->mount;
}
public function setMount(?string $mount): void
{
$this->mount = $this->truncateNullableString($mount, 150);
}
public function getAdminPassword(): ?string
{
return $this->admin_password;
}
public function setAdminPassword(?string $admin_password): void
{
$this->admin_password = $admin_password;
}
/** @inheritdoc */
public function getAutodjMount(): ?string
{
if (Adapters::REMOTE_ICECAST !== $this->getType()) {
return null;
}
$mount = $this->getSourceMount();
if (!empty($mount)) {
return $mount;
}
return $this->getMount();
}
public function getAutodjHost(): ?string
{
return $this->getUrlAsUri()->getHost();
}
public function getUrl(): string
{
return $this->url;
}
public function getUrlAsUri(): UriInterface
{
return new Uri($this->url);
}
public function setUrl(string $url): void
{
if (!empty($url) && !str_starts_with($url, 'http')) {
/** @noinspection HttpUrlsUsage */
$url = 'http://' . $url;
}
$this->url = $this->truncateString($url);
}
/*
* StationMountInterface compliance methods
*/
/** @inheritdoc */
public function getAutodjPort(): ?int
{
return $this->getSourcePort() ?? $this->getUrlAsUri()->getPort();
}
public function getSourcePort(): ?int
{
return $this->source_port;
}
public function setSourcePort(?int $source_port): void
{
if ((int)$source_port === 0) {
$source_port = null;
}
$this->source_port = $source_port;
}
public function getAutodjProtocol(): ?string
{
$urlScheme = $this->getUrlAsUri()->getScheme();
return match ($this->getAutodjAdapterType()) {
Adapters::REMOTE_SHOUTCAST1, Adapters::REMOTE_SHOUTCAST2 => self::PROTOCOL_ICY,
default => ('https' === $urlScheme) ? self::PROTOCOL_HTTPS : self::PROTOCOL_HTTP
};
}
public function getAutodjAdapterType(): string
{
return $this->getType();
}
public function getIsPublic(): bool
{
return $this->is_public;
}
public function setIsPublic(bool $is_public): void
{
$this->is_public = $is_public;
}
public function getListenersUnique(): int
{
return $this->listeners_unique;
}
public function setListenersUnique(int $listeners_unique): void
{
$this->listeners_unique = $listeners_unique;
}
public function getListenersTotal(): int
{
return $this->listeners_total;
}
public function setListenersTotal(int $listeners_total): void
{
$this->listeners_total = $listeners_total;
}
/**
* @return bool Whether this remote relay can be hand-edited.
*/
public function isEditable(): bool
{
return (Adapters::REMOTE_AZURARELAY !== $this->type);
}
/**
* Retrieve the API version of the object/array.
*
* @param AbstractRemote $adapter
*/
public function api(
AbstractRemote $adapter
): Api\NowPlaying\StationRemote {
$response = new Api\NowPlaying\StationRemote();
$response->id = $this->getIdRequired();
$response->name = $this->getDisplayName();
$response->url = $adapter->getPublicUrl($this);
$response->listeners = new Api\NowPlaying\Listeners(
total: $this->listeners_total,
unique: $this->listeners_unique
);
if ($this->enable_autodj || (Adapters::REMOTE_AZURARELAY === $this->type)) {
$response->bitrate = (int)$this->autodj_bitrate;
$response->format = (string)$this->autodj_format;
}
return $response;
}
public function getDisplayName(): string
{
if (!empty($this->display_name)) {
return $this->display_name;
}
if ($this->enable_autodj) {
return $this->autodj_bitrate . 'kbps ' . strtoupper($this->autodj_format ?? '');
}
return Utilities\Strings::truncateUrl($this->url);
}
/**
* @param string|null $display_name
*/
public function setDisplayName(?string $display_name): void
{
$this->display_name = $this->truncateNullableString($display_name);
}
public function __toString(): string
{
return $this->getStation() . ' Relay: ' . $this->getDisplayName();
}
}