Skip to content
Snippets Groups Projects
Episode.php 22.2 KiB
Newer Older
  • Learn to ignore specific revisions
  •  * @copyright  2020 Ad Aures
    
     * @license    https://www.gnu.org/licenses/agpl-3.0.en.html AGPL3
     * @link       https://castopod.org/
     */
    
    use App\Models\EpisodeCommentModel;
    
    use CodeIgniter\HTTP\Files\UploadedFile;
    
    use Exception;
    
    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\Media\Entities\Audio;
    use Modules\Media\Entities\Chapters;
    use Modules\Media\Entities\Image;
    use Modules\Media\Entities\Transcript;
    use Modules\Media\Models\MediaModel;
    
    use SimpleXMLElement;
    
    /**
     * @property int $id
     * @property int $podcast_id
     * @property Podcast $podcast
    
     * @property ?string $preview_id
     * @property string $preview_link
    
     * @property string $link
     * @property string $guid
     * @property string $slug
     * @property string $title
    
     * @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
    
     * @property string|null $location_name
     * @property string|null $location_geo
    
     * @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 int $downloads
    
     * @property EpisodeComment[]|null $comments
    
     * @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
    
        protected ?Audio $audio = null;
    
        protected string $audio_opengraph_url;
    
        protected string $embed_url;
    
        protected ?Image $cover = null;
    
        protected ?Transcript $transcript = null;
    
        protected ?Chapters $chapters = null;
    
        protected int $downloads = 0;
    
    
        protected string $custom_rss_string;
    
        protected $dates = ['published_at', 'created_at', 'updated_at'];
    
            'id'                    => 'integer',
            'podcast_id'            => 'integer',
    
            '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',
    
        public function setCover(UploadedFile | File $file = null): self
    
            if (! $file instanceof File || ($file instanceof UploadedFile && ! $file->isValid())) {
    
            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(),
    
    ->podcastCoverSizes,
    
                    'uploaded_by' => $this->attributes['updated_by'],
                    'updated_by'  => $this->attributes['updated_by'],
    
                ]);
                $cover->setFile($file);
    
                $this->attributes['cover_id'] = (new MediaModel('image'))->saveMedia($cover);
            }
    
            if ($this->cover instanceof Image) {
                return $this->cover;
    
            if ($this->cover_id === null) {
                $this->cover = $this->getPodcast()
                    ->getCover();
    
                return $this->cover;
            }
    
            $this->cover = (new MediaModel('image'))->getMediaById($this->cover_id);
    
    
        public function setAudio(UploadedFile | File $file = null): self
    
            if (! $file instanceof File || ($file instanceof UploadedFile && ! $file->isValid())) {
    
                $this->getAudio()
                    ->setFile($file);
                $this->getAudio()
    
                    ->updated_by = $this->attributes['updated_by'];
    
                (new MediaModel('audio'))->updateMedia($this->getAudio());
            } else {
    
                    'file_key'      => 'podcasts/' . $this->getPodcast()->handle . '/' . $file->getRandomName(),
    
                    'language_code' => $this->getPodcast()
                        ->language_code,
    
                    'uploaded_by' => $this->attributes['updated_by'],
                    'updated_by'  => $this->attributes['updated_by'],
    
                $this->attributes['audio_id'] = (new MediaModel())->saveMedia($audio);
    
            if (! $this->audio instanceof Audio) {
    
                $this->audio = (new MediaModel('audio'))->getMediaById($this->audio_id);
            }
    
        public function setTranscript(UploadedFile | File $file = null): self
    
            if (! $file instanceof File || ($file instanceof UploadedFile && ! $file->isValid())) {
    
            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(),
    
    ->language_code,
    
                    'uploaded_by' => $this->attributes['updated_by'],
                    'updated_by'  => $this->attributes['updated_by'],
    
                $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);
    
        public function setChapters(UploadedFile | File $file = null): self
    
            if (! $file instanceof File || ($file instanceof UploadedFile && ! $file->isValid())) {
    
            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(),
    
    ->language_code,
    
                    'uploaded_by' => $this->attributes['updated_by'],
                    'updated_by'  => $this->attributes['updated_by'],
    
                $this->attributes['chapters_id'] = (new MediaModel('chapters'))->saveMedia($chapters);
    
            }
    
            return $this;
        }
    
        public function getChapters(): ?Chapters
    
            if ($this->chapters_id !== null && ! $this->chapters instanceof Chapters) {
    
                $this->chapters = (new MediaModel('chapters'))->getMediaById($this->chapters_id);
    
            $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
    
    Benjamin Bellamy's avatar
    Benjamin Bellamy committed
        {
    
            return $this->getAudioUrl() . '?_from=-+Website+-';
    
        public function getAudioOpengraphUrl(): string
    
            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;
    
            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.
    
            if ($this->chapters instanceof Chapters) {
    
                return $this->chapters->file_url;
    
        /**
         * Returns the episode's persons
         *
    
        public function getPersons(): array
    
                throw new RuntimeException('Episode must be created before getting persons.');
    
                $this->persons = (new PersonModel())->getEpisodePersons($this->podcast_id, $this->id);
    
                throw new RuntimeException('Episode must be created before getting soundbites.');
    
            if ($this->soundbites === null) {
                $this->soundbites = (new ClipModel())->getEpisodeSoundbites($this->getPodcast()->id, $this->id);
    
                throw new RuntimeException('Episode must be created before getting posts.');
    
            if ($this->posts === null) {
                $this->posts = (new PostModel())->getEpisodePosts($this->id);
    
         */
        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);
    
        public function getLink(): string
    
            return url_to('episode', esc($this->getPodcast()->handle), esc($this->attributes['slug']));
    
        public function getEmbedUrl(string $theme = null): string
    
                    ? url_to('embed-theme', esc($this->getPodcast()->handle), esc($this->attributes['slug']), $theme)
    
                    : 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;
    
        public function getPodcast(): ?Podcast
    
            return (new PodcastModel())->getPodcastById($this->podcast_id);
    
        public function setDescriptionMarkdown(string $descriptionMarkdown): static
    
            ];
    
            $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;
    
            $this->attributes['description_html'] = $converter->convert($descriptionMarkdown);
    
        public function getDescriptionHtml(?string $serviceSlug = null): string
    
                $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()
    
        public function getDescription(): string
    
            if ($this->description === null) {
                $this->description = trim(
    
                    preg_replace('~\s+~', ' ', strip_tags((string) $this->attributes['description_html'])),
    
        public function getPublicationStatus(): string
    
                if (! $this->published_at instanceof Time) {
    
                } elseif ($this->getPodcast()->publication_status !== 'published') {
                    $this->publication_status = 'with_podcast';
    
                } elseif ($this->published_at->isBefore(Time::now())) {
                    $this->publication_status = 'published';
                } else {
                    $this->publication_status = 'scheduled';
                }
    
    
        /**
         * Saves the location name and fetches OpenStreetMap info
         */
    
        public function setLocation(?Location $location = null): static
    
            if (! $location instanceof Location) {
    
                $this->attributes['location_name'] = null;
                $this->attributes['location_geo'] = null;
    
                ! isset($this->attributes['location_name']) ||
    
                $this->attributes['location_name'] !== $location->name
    
                $this->attributes['location_name'] = $location->name;
                $this->attributes['location_geo'] = $location->geo;
                $this->attributes['location_osm'] = $location->osm;
    
        public function getLocation(): ?Location
        {
            if ($this->location_name === null) {
                return null;
            }
    
    
            if (! $this->location instanceof Location) {
    
                $this->location = new Location($this->location_name, $this->location_geo, $this->location_osm);
    
        public function getCustomRssString(): string
    
            if ($this->custom_rss === null) {
    
                '<?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"/>',
    
            array_to_rss([
                'elements' => $this->custom_rss,
    
            return str_replace(['<item>', '</item>'], '', (string) $xmlNode->asXML());
    
        public function setCustomRssString(?string $customRssString = null): static
    
            if ($customRssString === '') {
                $this->attributes['custom_rss'] = null;
    
    
            $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];
    
            if (array_key_exists('elements', $customRssArray)) {
    
                $this->attributes['custom_rss'] = json_encode($customRssArray['elements']);
    
        public function getPartnerLink(?string $serviceSlug = null): string
    
            $partnerLink =
                rtrim($this->getPodcast()->partner_link_url, '/') .
    
                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, '/') .
    
                urlencode((string) $this->attributes['guid']) .
    
                ($serviceSlug !== null ? '&_from=' . $serviceSlug : '');
    
    
        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);
        }