mirror of
https://code.castopod.org/adaures/castopod.git
synced 2024-08-02 01:23:22 +02:00
413 lines
14 KiB
PHP
413 lines
14 KiB
PHP
<?php
|
|
|
|
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\Controllers;
|
|
|
|
use App\Entities\Episode;
|
|
use App\Entities\Podcast;
|
|
use App\Libraries\NoteObject;
|
|
use App\Libraries\PodcastEpisode;
|
|
use App\Models\EpisodeModel;
|
|
use App\Models\PodcastModel;
|
|
use App\Models\PostModel;
|
|
use CodeIgniter\Database\BaseBuilder;
|
|
use CodeIgniter\Exceptions\PageNotFoundException;
|
|
use CodeIgniter\HTTP\RedirectResponse;
|
|
use CodeIgniter\HTTP\Response;
|
|
use CodeIgniter\HTTP\ResponseInterface;
|
|
use Config\Services;
|
|
use Modules\Analytics\AnalyticsTrait;
|
|
use Modules\Fediverse\Objects\OrderedCollectionObject;
|
|
use Modules\Fediverse\Objects\OrderedCollectionPage;
|
|
use Modules\PremiumPodcasts\Models\SubscriptionModel;
|
|
use SimpleXMLElement;
|
|
|
|
class EpisodeController extends BaseController
|
|
{
|
|
use AnalyticsTrait;
|
|
|
|
protected Podcast $podcast;
|
|
|
|
protected Episode $episode;
|
|
|
|
public function _remap(string $method, string ...$params): mixed
|
|
{
|
|
if (count($params) < 2) {
|
|
throw PageNotFoundException::forPageNotFound();
|
|
}
|
|
|
|
if (
|
|
($podcast = (new PodcastModel())->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' =>
|
|
'<iframe src="' .
|
|
$this->episode->embed_url .
|
|
'" width="100%" height="' . config('Embed')->height . '" frameborder="0" scrolling="no"></iframe>',
|
|
'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("<?xml version='1.0' encoding='utf-8' standalone='yes'?><oembed></oembed>");
|
|
|
|
$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(
|
|
'<iframe src="' .
|
|
$this->episode->embed_url .
|
|
'" width="100%" height="' . config('Embed')->height . '" frameborder="0" scrolling="no"></iframe>',
|
|
),
|
|
);
|
|
$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()));
|
|
}
|
|
}
|