getPodcastByHandle($params[0])) === null ) { throw PageNotFoundException::forPageNotFound(); } $this->podcast = $podcast; if ( ($episode = (new EpisodeModel())->getEpisodeBySlug($params[0], $params[1])) === null ) { throw PageNotFoundException::forPageNotFound(); } $this->episode = $episode; unset($params[1]); unset($params[0]); return $this->{$method}(...$params); } public function index(): string { // Prevent analytics hit when authenticated if (! auth()->loggedIn()) { $this->registerPodcastWebpageHit($this->episode->podcast_id); } $cacheName = implode( '_', array_filter([ 'page', "podcast#{$this->podcast->id}", "episode#{$this->episode->id}", service('request') ->getLocale(), is_unlocked($this->podcast->handle) ? 'unlocked' : null, auth() ->loggedIn() ? 'authenticated' : null, ]), ); if (! ($cachedView = cache($cacheName))) { $data = [ 'metatags' => get_episode_metatags($this->episode), 'podcast' => $this->podcast, 'episode' => $this->episode, ]; $secondsToNextUnpublishedEpisode = (new EpisodeModel())->getSecondsToNextUnpublishedEpisode( $this->podcast->id, ); if (auth()->loggedIn()) { helper('form'); return view('episode/comments', $data); } // The page cache is set to a decade so it is deleted manually upon podcast update return view('episode/comments', $data, [ 'cache' => $secondsToNextUnpublishedEpisode ? $secondsToNextUnpublishedEpisode : DECADE, 'cache_name' => $cacheName, ]); } return $cachedView; } public function activity(): string { // Prevent analytics hit when authenticated if (! auth()->loggedIn()) { $this->registerPodcastWebpageHit($this->episode->podcast_id); } $cacheName = implode( '_', array_filter([ 'page', "podcast#{$this->podcast->id}", "episode#{$this->episode->id}", 'activity', service('request') ->getLocale(), is_unlocked($this->podcast->handle) ? 'unlocked' : null, auth() ->loggedIn() ? 'authenticated' : null, ]), ); if (! ($cachedView = cache($cacheName))) { $data = [ 'metatags' => get_episode_metatags($this->episode), 'podcast' => $this->podcast, 'episode' => $this->episode, ]; $secondsToNextUnpublishedEpisode = (new EpisodeModel())->getSecondsToNextUnpublishedEpisode( $this->podcast->id, ); if (auth()->loggedIn()) { helper('form'); return view('episode/activity', $data); } // The page cache is set to a decade so it is deleted manually upon podcast update return view('episode/activity', $data, [ 'cache' => $secondsToNextUnpublishedEpisode ? $secondsToNextUnpublishedEpisode : DECADE, 'cache_name' => $cacheName, ]); } return $cachedView; } public function embed(string $theme = 'light-transparent'): string { header('Content-Security-Policy: frame-ancestors http://*:* https://*:*'); // Prevent analytics hit when authenticated if (! auth()->loggedIn()) { $this->registerPodcastWebpageHit($this->episode->podcast_id); } $session = Services::session(); $session->start(); if (isset($_SERVER['HTTP_REFERER'])) { $session->set('embed_domain', parse_url((string) $_SERVER['HTTP_REFERER'], PHP_URL_HOST)); } $cacheName = implode( '_', array_filter([ 'page', "podcast#{$this->podcast->id}", "episode#{$this->episode->id}", 'embed', $theme, service('request') ->getLocale(), is_unlocked($this->podcast->handle) ? 'unlocked' : null, ]), ); if (! ($cachedView = cache($cacheName))) { $themeData = EpisodeModel::$themes[$theme]; $data = [ 'podcast' => $this->podcast, 'episode' => $this->episode, 'theme' => $theme, 'themeData' => $themeData, ]; $secondsToNextUnpublishedEpisode = (new EpisodeModel())->getSecondsToNextUnpublishedEpisode( $this->podcast->id, ); // The page cache is set to a decade so it is deleted manually upon podcast update return view('embed', $data, [ 'cache' => $secondsToNextUnpublishedEpisode ? $secondsToNextUnpublishedEpisode : DECADE, 'cache_name' => $cacheName, ]); } return $cachedView; } public function oembedJSON(): ResponseInterface { return $this->response->setJSON([ 'type' => 'rich', 'version' => '1.0', 'title' => $this->episode->title, 'provider_name' => $this->podcast->title, 'provider_url' => $this->podcast->link, 'author_name' => $this->podcast->title, 'author_url' => $this->podcast->link, 'html' => '', 'width' => config('Embed') ->width, 'height' => config('Embed') ->height, 'thumbnail_url' => $this->episode->cover->og_url, 'thumbnail_width' => config('Images') ->podcastCoverSizes['og']['width'], 'thumbnail_height' => config('Images') ->podcastCoverSizes['og']['height'], ]); } public function oembedXML(): ResponseInterface { $oembed = new SimpleXMLElement(""); $oembed->addChild('type', 'rich'); $oembed->addChild('version', '1.0'); $oembed->addChild('title', $this->episode->title); $oembed->addChild('provider_name', $this->podcast->title); $oembed->addChild('provider_url', $this->podcast->link); $oembed->addChild('author_name', $this->podcast->title); $oembed->addChild('author_url', $this->podcast->link); $oembed->addChild('thumbnail', $this->episode->cover->og_url); $oembed->addChild('thumbnail_width', (string) config('Images')->podcastCoverSizes['og']['width']); $oembed->addChild('thumbnail_height', (string) config('Images')->podcastCoverSizes['og']['height']); $oembed->addChild( 'html', htmlspecialchars( '', ), ); $oembed->addChild('width', (string) config('Embed')->width); $oembed->addChild('height', (string) config('Embed')->height); // @phpstan-ignore-next-line return $this->response->setXML($oembed); } /** * @noRector ReturnTypeDeclarationRector */ public function episodeObject(): Response { $podcastObject = new PodcastEpisode($this->episode); return $this->response ->setContentType('application/json') ->setBody($podcastObject->toJSON()); } /** * @noRector ReturnTypeDeclarationRector */ public function comments(): Response { /** * get comments: aggregated replies from posts referring to the episode */ $episodeComments = model(PostModel::class) ->whereIn('in_reply_to_id', function (BaseBuilder $builder): BaseBuilder { return $builder->select('id') ->from(config('Fediverse')->tablesPrefix . 'posts') ->where('episode_id', $this->episode->id); }) ->where('`published_at` <= UTC_TIMESTAMP()', null, false) ->orderBy('published_at', 'ASC'); $pageNumber = (int) $this->request->getGet('page'); if ($pageNumber < 1) { $episodeComments->paginate(12); $pager = $episodeComments->pager; $collection = new OrderedCollectionObject(null, $pager); } else { $paginatedComments = $episodeComments->paginate(12, 'default', $pageNumber); $pager = $episodeComments->pager; $orderedItems = []; if ($paginatedComments !== null) { foreach ($paginatedComments as $comment) { $orderedItems[] = (new NoteObject($comment))->toArray(); } } // @phpstan-ignore-next-line $collection = new OrderedCollectionPage($pager, $orderedItems); } return $this->response ->setContentType('application/activity+json') ->setHeader('Access-Control-Allow-Origin', '*') ->setBody($collection->toJSON()); } public function audio(): RedirectResponse | ResponseInterface { // check if episode is premium? $subscription = null; // check if podcast is already unlocked before any token validation if ($this->episode->is_premium && ($subscription = service('premium_podcasts')->subscription( $this->episode->podcast->handle )) === null) { // look for token as GET parameter if (($token = $this->request->getGet('token')) === null) { return $this->response->setStatusCode(401) ->setJSON([ 'errors' => [ 'status' => 401, 'title' => 'Unauthorized', 'detail' => 'Episode is premium, you must provide a token to unlock it.', ], ]); } // check if there's a valid subscription for the provided token if (($subscription = (new SubscriptionModel())->validateSubscription( $this->episode->podcast->handle, $token )) === null) { return $this->response->setStatusCode(401, 'Invalid token!') ->setJSON([ 'errors' => [ 'status' => 401, 'title' => 'Unauthorized', 'detail' => 'Invalid token!', ], ]); } } $session = Services::session(); $session->start(); $serviceName = ''; if ($this->request->getGet('_from')) { $serviceName = $this->request->getGet('_from'); } elseif ($session->get('embed_domain') !== null) { $serviceName = $session->get('embed_domain'); } elseif ($session->get('referer') !== null && $session->get('referer') !== '- Direct -') { $serviceName = parse_url((string) $session->get('referer'), PHP_URL_HOST); } $audioFileSize = $this->episode->audio->file_size; $audioFileHeaderSize = $this->episode->audio->header_size; $audioDuration = $this->episode->audio->duration; // bytes_threshold: number of bytes that must be downloaded for an episode to be counted in download analytics // - if audio is less than or equal to 60s, then take the audio file_size // - if audio is more than 60s, then take the audio file_header_size + 60s $bytesThreshold = $audioDuration <= 60 ? $audioFileSize : $audioFileHeaderSize + (int) floor((($audioFileSize - $audioFileHeaderSize) / $audioDuration) * 60); helper('analytics'); podcast_hit( $this->episode->podcast_id, $this->episode->id, $bytesThreshold, $audioFileSize, $audioDuration, $this->episode->published_at->getTimestamp(), $serviceName, $subscription !== null ? $subscription->id : null ); $analyticsConfig = config('Analytics'); return redirect()->to($analyticsConfig->getAudioUrl($this->episode, $this->request->getGet())); } }