Newer
Older
<?php

Yassine Doghri
committed

Yassine Doghri
committed
declare(strict_types=1);
* @copyright 2020 Ad Aures
* @license https://www.gnu.org/licenses/agpl-3.0.en.html AGPL3
* @link https://castopod.org/
*/
namespace App\Entities;

Yassine Doghri
committed
use App\Entities\Clip\Soundbite;

Yassine Doghri
committed
use App\Libraries\SimpleRSSElement;

Yassine Doghri
committed
use App\Models\ClipModel;
use App\Models\EpisodeCommentModel;

Yassine Doghri
committed
use App\Models\EpisodeModel;
use App\Models\PersonModel;
use App\Models\PodcastModel;
use App\Models\PostModel;

Yassine Doghri
committed
use CodeIgniter\Entity\Entity;

Yassine Doghri
committed
use CodeIgniter\Files\File;
use CodeIgniter\HTTP\Files\UploadedFile;

Yassine Doghri
committed
use CodeIgniter\I18n\Time;

Yassine Doghri
committed
use Config\Images;

Yassine Doghri
committed
use League\CommonMark\Environment\Environment;
use League\CommonMark\Extension\Autolink\AutolinkExtension;
use League\CommonMark\Extension\CommonMark\CommonMarkCoreExtension;
use League\CommonMark\Extension\DisallowedRawHtml\DisallowedRawHtmlExtension;
use League\CommonMark\Extension\SmartPunct\SmartPunctExtension;
use League\CommonMark\MarkdownConverter;
use Modules\Analytics\OP3;
use Modules\Media\Entities\Audio;
use Modules\Media\Entities\Chapters;
use Modules\Media\Entities\Image;
use Modules\Media\Entities\Transcript;
use Modules\Media\Models\MediaModel;

Yassine Doghri
committed
use RuntimeException;
/**
* @property int $id
* @property int $podcast_id
* @property Podcast $podcast

Yassine Doghri
committed
* @property ?string $preview_id
* @property string $preview_link
* @property string $link
* @property string $guid
* @property string $slug
* @property string $title
* @property int $audio_id
* @property ?Audio $audio

Yassine Doghri
committed
* @property string $audio_url
* @property string $audio_web_url
* @property string $audio_opengraph_url
* @property string|null $description Holds text only description, striped of any markdown or html special characters
* @property string $description_markdown
* @property string $description_html
* @property ?int $cover_id
* @property ?Image $cover
* @property int|null $transcript_id
* @property Transcript|null $transcript
* @property string|null $transcript_remote_url
* @property int|null $chapters_id
* @property Chapters|null $chapters
* @property string|null $chapters_remote_url
* @property string|null $parental_advisory
* @property int $number
* @property int $season_number
* @property string $type
* @property bool $is_blocked

Yassine Doghri
committed
* @property Location|null $location
* @property string|null $location_name
* @property string|null $location_geo

Yassine Doghri
committed
* @property string|null $location_osm
* @property array|null $custom_rss
* @property string $custom_rss_string
* @property bool $is_published_on_hubs
* @property int $posts_count
* @property int $comments_count
* @property EpisodeComment[]|null $comments
* @property bool $is_premium
* @property int $created_by
* @property int $updated_by
* @property string $publication_status
* @property Time|null $published_at
* @property Time $created_at
* @property Time $updated_at
* @property Person[] $persons
* @property Soundbite[] $soundbites
* @property string $embed_url
class Episode extends Entity
{
protected Podcast $podcast;
protected string $link;
protected ?Audio $audio = null;

Yassine Doghri
committed
protected string $audio_url;
protected string $audio_web_url;
protected string $audio_opengraph_url;
protected string $embed_url;
protected ?Image $cover = null;

Yassine Doghri
committed
protected ?string $description = null;
protected ?Transcript $transcript = null;
protected ?Chapters $chapters = null;
protected int $downloads = 0;
* @var Person[]|null
protected ?array $persons = null;

Yassine Doghri
committed
* @var Soundbite[]|null

Yassine Doghri
committed
protected ?array $soundbites = null;
/**
* @var Post[]|null
*/
protected ?array $posts = null;

Yassine Doghri
committed
/**
* @var EpisodeComment[]|null

Yassine Doghri
committed
*/
protected ?array $comments = null;

Yassine Doghri
committed
protected ?Location $location = null;
protected string $custom_rss_string;

Yassine Doghri
committed
protected ?string $publication_status = null;

Yassine Doghri
committed

Benjamin Bellamy
committed
/**

Yassine Doghri
committed
* @var array<int, string>
* @phpstan-var list<string>

Benjamin Bellamy
committed
*/
protected $dates = ['published_at', 'created_at', 'updated_at'];
/**
* @var array<string, string>
*/
protected $casts = [
'id' => 'integer',
'podcast_id' => 'integer',

Yassine Doghri
committed
'preview_id' => '?string',
'guid' => 'string',
'slug' => 'string',
'title' => 'string',
'audio_id' => 'integer',
'description_markdown' => 'string',
'description_html' => 'string',
'cover_id' => '?integer',
'transcript_id' => '?integer',
'transcript_remote_url' => '?string',
'chapters_id' => '?integer',
'chapters_remote_url' => '?string',
'parental_advisory' => '?string',
'number' => '?integer',
'season_number' => '?integer',
'type' => 'string',
'is_blocked' => 'boolean',
'location_name' => '?string',
'location_geo' => '?string',
'location_osm' => '?string',
'custom_rss' => '?json-array',
'is_published_on_hubs' => 'boolean',
'posts_count' => 'integer',
'comments_count' => 'integer',
'is_premium' => 'boolean',
'created_by' => 'integer',
'updated_by' => 'integer',

Yassine Doghri
committed
public function setCover(UploadedFile | File $file = null): self
if (! $file instanceof File || ($file instanceof UploadedFile && ! $file->isValid())) {
return $this;
if (array_key_exists('cover_id', $this->attributes) && $this->attributes['cover_id'] !== null) {
$this->getCover()
->setFile($file);
$this->getCover()
->updated_by = $this->attributes['updated_by'];
(new MediaModel('image'))->updateMedia($this->getCover());
} else {
$cover = new Image([
'file_key' => 'podcasts/' . $this->getPodcast()->handle . '/' . $this->attributes['slug'] . '.' . $file->getExtension(),

Yassine Doghri
committed
'sizes' => config(Images::class)
'uploaded_by' => $this->attributes['updated_by'],
'updated_by' => $this->attributes['updated_by'],
]);
$cover->setFile($file);
$this->attributes['cover_id'] = (new MediaModel('image'))->saveMedia($cover);
}

Yassine Doghri
committed
public function getCover(): Image

Yassine Doghri
committed
if ($this->cover instanceof Image) {
return $this->cover;

Yassine Doghri
committed
if ($this->cover_id === null) {
$this->cover = $this->getPodcast()
->getCover();
return $this->cover;
}
$this->cover = (new MediaModel('image'))->getMediaById($this->cover_id);
return $this->cover;

Yassine Doghri
committed
public function setAudio(UploadedFile | File $file = null): self
if (! $file instanceof File || ($file instanceof UploadedFile && ! $file->isValid())) {
return $this;
}

Yassine Doghri
committed
if ($this->audio_id !== 0) {
$this->getAudio()
->setFile($file);
$this->getAudio()
->updated_by = $this->attributes['updated_by'];
(new MediaModel('audio'))->updateMedia($this->getAudio());
} else {

Yassine Doghri
committed
$audio = new Audio([
'file_key' => 'podcasts/' . $this->getPodcast()->handle . '/' . $file->getRandomName(),

Yassine Doghri
committed
'language_code' => $this->getPodcast()
->language_code,
'uploaded_by' => $this->attributes['updated_by'],
'updated_by' => $this->attributes['updated_by'],

Yassine Doghri
committed
$audio->setFile($file);

Yassine Doghri
committed
$this->attributes['audio_id'] = (new MediaModel())->saveMedia($audio);
}
return $this;
}
public function getAudio(): Audio
if (! $this->audio instanceof Audio) {
$this->audio = (new MediaModel('audio'))->getMediaById($this->audio_id);
}
return $this->audio;

Yassine Doghri
committed
public function setTranscript(UploadedFile | File $file = null): self
if (! $file instanceof File || ($file instanceof UploadedFile && ! $file->isValid())) {
return $this;
}
if ($this->getTranscript() instanceof Transcript) {
$this->getTranscript()
->setFile($file);
$this->getTranscript()
->updated_by = $this->attributes['updated_by'];
(new MediaModel('transcript'))->updateMedia($this->getTranscript());
} else {
$transcript = new Transcript([
'file_key' => 'podcasts/' . $this->getPodcast()->handle . '/' . $this->attributes['slug'] . '-transcript.' . $file->getExtension(),

Yassine Doghri
committed
'language_code' => $this->getPodcast()
'uploaded_by' => $this->attributes['updated_by'],
'updated_by' => $this->attributes['updated_by'],
]);
$transcript->setFile($file);

Yassine Doghri
committed
$this->attributes['transcript_id'] = (new MediaModel('transcript'))->saveMedia($transcript);
}
return $this;
}
public function getTranscript(): ?Transcript
if ($this->transcript_id !== null && ! $this->transcript instanceof Transcript) {
$this->transcript = (new MediaModel('transcript'))->getMediaById($this->transcript_id);
return $this->transcript;

Yassine Doghri
committed
public function setChapters(UploadedFile | File $file = null): self
if (! $file instanceof File || ($file instanceof UploadedFile && ! $file->isValid())) {
return $this;
}
if ($this->getChapters() instanceof Chapters) {
$this->getChapters()
->setFile($file);
$this->getChapters()
->updated_by = $this->attributes['updated_by'];
(new MediaModel('chapters'))->updateMedia($this->getChapters());
} else {
$chapters = new Chapters([
'file_key' => 'podcasts/' . $this->getPodcast()->handle . '/' . $this->attributes['slug'] . '-chapters' . '.' . $file->getExtension(),

Yassine Doghri
committed
'language_code' => $this->getPodcast()
'uploaded_by' => $this->attributes['updated_by'],
'updated_by' => $this->attributes['updated_by'],
]);
$chapters->setFile($file);

Yassine Doghri
committed
$this->attributes['chapters_id'] = (new MediaModel('chapters'))->saveMedia($chapters);
}
return $this;
}
public function getChapters(): ?Chapters
if ($this->chapters_id !== null && ! $this->chapters instanceof Chapters) {

Yassine Doghri
committed
$this->chapters = (new MediaModel('chapters'))->getMediaById($this->chapters_id);
}
return $this->chapters;

Yassine Doghri
committed
public function getAudioUrl(): string
$audioURL = url_to(
'episode-audio',
$this->getPodcast()
->handle,
$this->slug,
$this->getAudio()
->file_extension
);
// Wrap episode url with OP3 if episode is public and OP3 is enabled on this podcast
if (! $this->is_premium && service('settings')->get(
'Analytics.enableOP3',
'podcast:' . $this->podcast_id
)) {
$op3 = new OP3(config('Analytics')->OP3);
return $op3->wrap($audioURL, $this);
}
return $audioURL;
public function getAudioWebUrl(): string

Yassine Doghri
committed
return $this->getAudioUrl() . '?_from=-+Website+-';
public function getAudioOpengraphUrl(): string

Yassine Doghri
committed
return $this->getAudioUrl() . '?_from=-+Open+Graph+-';
* Gets transcript url from transcript file uri if it exists or returns the transcript_remote_url which can be null.
public function getTranscriptUrl(): ?string
if ($this->transcript instanceof Transcript) {
return $this->transcript->file_url;

Yassine Doghri
committed
return $this->transcript_remote_url;
* Gets chapters file url from chapters file uri if it exists or returns the chapters_remote_url which can be null.

Yassine Doghri
committed
public function getChaptersFileUrl(): ?string
if ($this->chapters instanceof Chapters) {
return $this->chapters->file_url;

Yassine Doghri
committed
return $this->chapters_remote_url;
/**
* Returns the episode's persons
*
* @return Person[]
public function getPersons(): array
if ($this->id === null) {

Yassine Doghri
committed
throw new RuntimeException('Episode must be created before getting persons.');
if ($this->persons === null) {

Yassine Doghri
committed
$this->persons = (new PersonModel())->getEpisodePersons($this->podcast_id, $this->id);
return $this->persons;
* Returns the episode’s clips

Yassine Doghri
committed
* @return Soundbite[]

Yassine Doghri
committed
public function getSoundbites(): array
if ($this->id === null) {

Yassine Doghri
committed
throw new RuntimeException('Episode must be created before getting soundbites.');

Yassine Doghri
committed
if ($this->soundbites === null) {
$this->soundbites = (new ClipModel())->getEpisodeSoundbites($this->getPodcast()->id, $this->id);

Yassine Doghri
committed
return $this->soundbites;
* @return Post[]
public function getPosts(): array
{
if ($this->id === null) {
throw new RuntimeException('Episode must be created before getting posts.');
}
if ($this->posts === null) {
$this->posts = (new PostModel())->getEpisodePosts($this->id);
}
return $this->posts;
}

Yassine Doghri
committed
/**
* @return EpisodeComment[]

Yassine Doghri
committed
*/
public function getComments(): array
{
if ($this->id === null) {
throw new RuntimeException('Episode must be created before getting comments.');
}
if ($this->comments === null) {
$this->comments = (new EpisodeCommentModel())->getEpisodeComments($this->id);

Yassine Doghri
committed
}
return $this->comments;
}
public function getLink(): string

Yassine Doghri
committed
return url_to('episode', esc($this->getPodcast()->handle), esc($this->attributes['slug']));
public function getEmbedUrl(string $theme = null): string

Yassine Doghri
committed
return $theme
? url_to('embed-theme', esc($this->getPodcast()->handle), esc($this->attributes['slug']), $theme)

Yassine Doghri
committed
: url_to('embed', esc($this->getPodcast()->handle), esc($this->attributes['slug']));
public function setGuid(?string $guid = null): static
$this->attributes['guid'] = $guid === null ? $this->getLink() : $guid;
return $this;
public function getPodcast(): ?Podcast

Yassine Doghri
committed
return (new PodcastModel())->getPodcastById($this->podcast_id);
public function setDescriptionMarkdown(string $descriptionMarkdown): static

Yassine Doghri
committed
$config = [
'html_input' => 'escape',

Yassine Doghri
committed
'allow_unsafe_links' => false,

Yassine Doghri
committed
];
$environment = new Environment($config);
$environment->addExtension(new CommonMarkCoreExtension());
$environment->addExtension(new AutolinkExtension());
$environment->addExtension(new SmartPunctExtension());
$environment->addExtension(new DisallowedRawHtmlExtension());
$converter = new MarkdownConverter($environment);
$this->attributes['description_markdown'] = $descriptionMarkdown;

Yassine Doghri
committed
$this->attributes['description_html'] = $converter->convert($descriptionMarkdown);
return $this;
}
public function getDescriptionHtml(?string $serviceSlug = null): string
$descriptionHtml = '';
if (
$this->getPodcast()
->partner_id !== null &&
$this->getPodcast()
->partner_link_url !== null &&
$this->getPodcast()
->partner_image_url !== null
$descriptionHtml .= "<div><a href=\"{$this->getPartnerLink(
$serviceSlug,
)}\" rel=\"sponsored noopener noreferrer\" target=\"_blank\"><img src=\"{$this->getPartnerImageUrl(
$serviceSlug,
)}\" alt=\"Partner image\" /></a></div>";
}
$descriptionHtml .= $this->attributes['description_html'];
if ($this->getPodcast()->episode_description_footer_html) {
$descriptionHtml .= "<footer>{$this->getPodcast()

Yassine Doghri
committed
->episode_description_footer_html}</footer>";
}
return $descriptionHtml;
public function getDescription(): string
if ($this->description === null) {
$this->description = trim(
preg_replace('~\s+~', ' ', strip_tags((string) $this->attributes['description_html'])),
return $this->description;
public function getPublicationStatus(): string

Yassine Doghri
committed
{

Yassine Doghri
committed
if ($this->publication_status === null) {
if (! $this->published_at instanceof Time) {

Yassine Doghri
committed
$this->publication_status = 'not_published';
} elseif ($this->getPodcast()->publication_status !== 'published') {
$this->publication_status = 'with_podcast';

Yassine Doghri
committed
} elseif ($this->published_at->isBefore(Time::now())) {
$this->publication_status = 'published';
} else {
$this->publication_status = 'scheduled';
}

Yassine Doghri
committed
}

Yassine Doghri
committed

Yassine Doghri
committed
return $this->publication_status;

Yassine Doghri
committed
}
/**
* Saves the location name and fetches OpenStreetMap info
*/

Yassine Doghri
committed
public function setLocation(?Location $location = null): static
if (! $location instanceof Location) {
$this->attributes['location_name'] = null;
$this->attributes['location_geo'] = null;

Yassine Doghri
committed
$this->attributes['location_osm'] = null;

Yassine Doghri
committed
return $this;
}
! isset($this->attributes['location_name']) ||

Yassine Doghri
committed
$this->attributes['location_name'] !== $location->name

Yassine Doghri
committed
$location->fetchOsmLocation();

Yassine Doghri
committed
$this->attributes['location_name'] = $location->name;
$this->attributes['location_geo'] = $location->geo;
$this->attributes['location_osm'] = $location->osm;

Benjamin Bellamy
committed
public function getLocation(): ?Location
{
if ($this->location_name === null) {
return null;
}
if (! $this->location instanceof Location) {

Yassine Doghri
committed
$this->location = new Location($this->location_name, $this->location_geo, $this->location_osm);
}
return $this->location;
}

Benjamin Bellamy
committed
/**
* Get custom rss tag as XML String
*/
public function getCustomRssString(): string

Benjamin Bellamy
committed
{
if ($this->custom_rss === null) {

Benjamin Bellamy
committed
return '';
}

Yassine Doghri
committed
helper('rss');
$xmlNode = (new SimpleRSSElement(
'<?xml version="1.0" encoding="utf-8"?><rss xmlns:itunes="http://www.itunes.com/dtds/podcast-1.0.dtd" xmlns:podcast="https://podcastindex.org/namespace/1.0" xmlns:content="http://purl.org/rss/1.0/modules/content/" version="2.0"/>',

Yassine Doghri
committed
))
->addChild('channel')
->addChild('item');
array_to_rss([
'elements' => $this->custom_rss,

Yassine Doghri
committed
], $xmlNode);
return str_replace(['<item>', '</item>'], '', (string) $xmlNode->asXML());

Benjamin Bellamy
committed
}
/**
* Saves custom rss tag into json
*/
public function setCustomRssString(?string $customRssString = null): static

Benjamin Bellamy
committed
{
if ($customRssString === '') {
$this->attributes['custom_rss'] = null;

Yassine Doghri
committed
return $this;
}

Benjamin Bellamy
committed
helper('rss');
$customXML = simplexml_load_string(
'<?xml version="1.0" encoding="utf-8"?><rss xmlns:itunes="http://www.itunes.com/dtds/podcast-1.0.dtd" xmlns:podcast="https://podcastindex.org/namespace/1.0" xmlns:content="http://purl.org/rss/1.0/modules/content/" version="2.0"><channel><item>' .
$customRssString .
'</item></channel></rss>',
);
if (! $customXML instanceof SimpleXMLElement) {
// TODO: Failed to parse custom xml, should return error?
return $this;
}
$customRssArray = rss_to_array($customXML)['elements'][0]['elements'][0];

Yassine Doghri
committed

Benjamin Bellamy
committed
if (array_key_exists('elements', $customRssArray)) {

Yassine Doghri
committed
$this->attributes['custom_rss'] = json_encode($customRssArray['elements']);

Benjamin Bellamy
committed
} else {
$this->attributes['custom_rss'] = null;
}

Yassine Doghri
committed
return $this;

Benjamin Bellamy
committed
}
public function getPartnerLink(?string $serviceSlug = null): string
$partnerLink =
rtrim($this->getPodcast()->partner_link_url, '/') .
$this->getPodcast()
->partner_id .
urlencode((string) $this->attributes['guid']);
if ($serviceSlug !== null) {
$partnerLink .= '&_from=' . $serviceSlug;
}
return $partnerLink;
public function getPartnerImageUrl(string $serviceSlug = null): string
return rtrim($this->getPodcast()->partner_image_url, '/') .

Yassine Doghri
committed
'?pid=' .
$this->getPodcast()
->partner_id .

Yassine Doghri
committed
'&guid=' .
urlencode((string) $this->attributes['guid']) .
($serviceSlug !== null ? '&_from=' . $serviceSlug : '');

Yassine Doghri
committed
public function getPreviewLink(): string
{
if ($this->preview_id === null) {
// generate preview id
if (! $previewUUID = (new EpisodeModel())->setEpisodePreviewId($this->id)) {
throw new Exception('Could not set episode preview id');
}
$this->preview_id = $previewUUID;
}
return url_to('episode-preview', (string) $this->preview_id);
}
/**
* Returns the episode's clip count
*/
public function getClipCount(): int|string
{
if ($this->id === null) {
throw new RuntimeException('Episode must be created before getting number of video clips.');
}
return (new ClipModel())->getClipCount($this->podcast_id, $this->id);
}