145 lines
4.0 KiB
PHP
145 lines
4.0 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace OCA\RePod\Core\EpisodeAction;
|
|
|
|
use OCA\GPodderSync\Core\EpisodeAction\EpisodeActionReader as CoreEpisodeActionReader;
|
|
use OCA\GPodderSync\Db\EpisodeAction\EpisodeActionRepository;
|
|
use OCA\RePod\Service\UserService;
|
|
|
|
class EpisodeActionReader extends CoreEpisodeActionReader
|
|
{
|
|
public function __construct(
|
|
private EpisodeActionRepository $episodeActionRepository,
|
|
private UserService $userService
|
|
) {}
|
|
|
|
/**
|
|
* Base: https://github.com/pbek/nextcloud-nextpod/blob/main/lib/Core/EpisodeAction/EpisodeActionExtraData.php#L119.
|
|
* Specs : https://github.com/Podcast-Standards-Project/PSP-1-Podcast-RSS-Specification/blob/main/README.md.
|
|
*
|
|
* @return EpisodeActionExtraData[]
|
|
* @throws \Exception if the XML data could not be parsed
|
|
*/
|
|
public function parseRssXml(string $xmlString, ?int $fetchedAtUnix = null): array {
|
|
$episodes = [];
|
|
$xml = new \SimpleXMLElement($xmlString);
|
|
$channel = $xml->channel;
|
|
$title = (string) $channel->title;
|
|
|
|
// Find episode by url and add data for it
|
|
/** @var \SimpleXMLElement $item */
|
|
foreach ($channel->item as $item) {
|
|
$url = (string) $item->enclosure['url'];
|
|
$type = (string) $item->enclosure['type'];
|
|
$size = (int) $item->enclosure['length'];
|
|
$guid = (string) $item->guid;
|
|
|
|
$iTunesItemChildren = $item->children('itunes', true);
|
|
$iTunesChannelChildren = $channel->children('itunes', true);
|
|
|
|
// Get episode action
|
|
$action = $this->episodeActionRepository->findByGuid($guid, $this->userService->getUserUID());
|
|
|
|
if ($action) {
|
|
$url = $action->getEpisode();
|
|
} else {
|
|
$action = $this->episodeActionRepository->findByEpisodeUrl($url, $this->userService->getUserUID());
|
|
}
|
|
|
|
// Get episode name
|
|
$name = (string) $item->title;
|
|
|
|
// Get episode link
|
|
$link = $this->stringOrNull($item->link);
|
|
|
|
// Get episode image
|
|
$image = $this->stringOrNull($item->image->url);
|
|
|
|
if (!isset($image) && isset($iTunesItemChildren)) {
|
|
$imageAttributes = $iTunesItemChildren->image->attributes();
|
|
$image = $this->stringOrNull(isset($imageAttributes) ? (string) $imageAttributes->href : '');
|
|
}
|
|
|
|
if (!isset($image)) {
|
|
$image = $this->stringOrNull($channel->image->url);
|
|
}
|
|
|
|
if (!isset($image) && isset($iTunesChannelChildren)) {
|
|
$imageAttributes = $iTunesChannelChildren->image->attributes();
|
|
$image = $this->stringOrNull(isset($imageAttributes) ? (string) $imageAttributes->href : '');
|
|
}
|
|
|
|
if (!isset($image)) {
|
|
preg_match('/<itunes:image\s+href="([^"]+)"/', $xmlString, $matches);
|
|
if (count($matches) > 1) {
|
|
$image = $this->stringOrNull($matches[1]);
|
|
}
|
|
}
|
|
|
|
// Get episode description
|
|
$itemContent = $item->children('content', true);
|
|
if (isset($itemContent)) {
|
|
$description = $this->stringOrNull($itemContent->encoded);
|
|
} else {
|
|
$description = $this->stringOrNull($item->description);
|
|
}
|
|
|
|
if (!isset($description) && isset($iTunesItemChildren)) {
|
|
$description = $this->stringOrNull($iTunesItemChildren->summary);
|
|
}
|
|
|
|
// Remove tags
|
|
$description = strip_tags(str_replace(['<br>', '<br/>', '<br />'], "\n", $description ?? ''));
|
|
|
|
// Get episode duration
|
|
if (isset($iTunesItemChildren)) {
|
|
$duration = $this->stringOrNull($iTunesItemChildren->duration);
|
|
} else {
|
|
$duration = $this->stringOrNull($item->duration);
|
|
}
|
|
|
|
// Get episode pubDate
|
|
$pubDate = $this->stringOrNull($item->pubDate);
|
|
if (isset($pubDate)) {
|
|
try {
|
|
$pubDate = new \DateTime($pubDate);
|
|
} catch (\Exception $e) {
|
|
$pubDate = null;
|
|
}
|
|
}
|
|
|
|
$episodes[] = new EpisodeActionExtraData(
|
|
$title,
|
|
$url,
|
|
$name,
|
|
$link,
|
|
$image,
|
|
$description,
|
|
$fetchedAtUnix ?? (new \DateTime())->getTimestamp(),
|
|
$guid,
|
|
$type,
|
|
$size,
|
|
$pubDate,
|
|
$duration,
|
|
$action
|
|
);
|
|
}
|
|
|
|
return $episodes;
|
|
}
|
|
|
|
/**
|
|
* @param null|\SimpleXMLElement|string $value
|
|
*/
|
|
private function stringOrNull($value): ?string {
|
|
/** @psalm-suppress RiskyTruthyFalsyComparison */
|
|
if (!empty($value)) {
|
|
return (string) $value;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
}
|