diff --git a/bridges/RumbleBridge.php b/bridges/RumbleBridge.php index c1a565bb..6841e887 100644 --- a/bridges/RumbleBridge.php +++ b/bridges/RumbleBridge.php @@ -4,17 +4,17 @@ class RumbleBridge extends BridgeAbstract { const NAME = 'Rumble.com Bridge'; const URI = 'https://rumble.com/'; - const DESCRIPTION = 'Fetches the latest channel/user videos and livestreams.'; + const DESCRIPTION = 'Fetches detailed channel/user videos and livestreams from Rumble.'; const MAINTAINER = 'dvikan, NotsoanoNimus'; - const CACHE_TIMEOUT = 60 * 60; // 1h + const CACHE_TIMEOUT = 0; + const PARAMETERS = [ [ 'account' => [ 'name' => 'Account', 'type' => 'text', 'required' => true, - 'title' => 'Name of the target account to create into a feed.', - 'defaultValue' => 'bjornandreasbullhansen', + 'title' => 'Name of the target account (e.g., 21UhrBitcoinPodcast)', ], 'type' => [ 'name' => 'Account Type', @@ -27,6 +27,12 @@ class RumbleBridge extends BridgeAbstract 'User (All)' => 'user', ], ], + 'cache_timeout' => [ + 'name' => 'Cache Timeout (seconds)', + 'type' => 'number', + 'defaultValue' => 0, + 'title' => 'How long to cache the feed (0 for no caching)', + ], ] ]; @@ -34,10 +40,10 @@ class RumbleBridge extends BridgeAbstract { $account = $this->getInput('account'); $type = $this->getInput('type'); - $url = self::getURI(); + $url = self::URI; if (!preg_match('#^[\w\-_.@]+$#', $account) || strlen($account) > 64) { - throw new \Exception('Invalid target account.'); + returnServerError('Invalid target account.'); } switch ($type) { @@ -54,40 +60,106 @@ class RumbleBridge extends BridgeAbstract $url .= "c/$account/livestreams"; break; default: - // Shouldn't ever happen. - throw new \Exception('Invalid media type.'); + returnServerError('Invalid media type.'); } - $dom = getSimpleHTMLDOM($url); - foreach ($dom->find('ol.thumbnail__grid div.thumbnail__grid--item') as $video) { - $href = $video->find('a', 0)->href; + $html = $this->getContents($url); + if (!$html) { + returnServerError("Failed to fetch $url"); + } - $item = [ - 'title' => $video->find('h3', 0)->plaintext, - 'author' => $account . '@rumble.com', - 'content' => defaultLinkTo($video, self::URI)->innertext, - ]; - - $time = $video->find('time', 0); - if ($time) { - $publishedAt = new \DateTimeImmutable($time->getAttribute('datetime')); - $item['timestamp'] = $publishedAt->getTimestamp(); + $items = []; + if (preg_match('/(.*?)<\/script>/s', $html, $matches)) { + $jsonData = json_decode($matches[1], true); + if ($jsonData) { + $videos = isset($jsonData['@graph']) ? $jsonData['@graph'] : [$jsonData]; + foreach ($videos as $item) { + if (isset($item['@type']) && $item['@type'] === 'VideoObject') { + $items[] = $this->createItemFromJsonLd($item, $account); + } + } } - - $href = ltrim($href, '/'); - $itemUrl = Url::fromString(self::URI . $href); - // Remove tracking parameter in query string - $item['uri'] = $itemUrl->withQueryString(null)->__toString(); - - $this->items[] = $item; } + + if (empty($items)) { + $dom = $this->getSimpleHTMLDOM($url); + if ($dom) { + foreach ($dom->find('ol.thumbnail__grid div.thumbnail__grid--item') as $video) { + $items[] = $this->createItemFromHtml($video, $account); + } + } else { + returnServerError("Failed to parse HTML from $url"); + } + } + + $this->items = $items; + } + + private function createItemFromJsonLd(array $json, string $account): array + { + $item = [ + 'title' => html_entity_decode($json['name'] ?? 'Untitled', ENT_QUOTES, 'UTF-8'), + 'author' => $account . '@rumble.com', + 'uri' => $json['url'] ?? '', + 'timestamp' => (new DateTime($json['uploadDate'] ?? 'now'))->getTimestamp(), + 'content' => '', + ]; + + if (isset($json['embedUrl'])) { + $item['content'] .= ""; + } + + if (isset($json['description'])) { + $item['content'] .= '

' . html_entity_decode($json['description'], ENT_QUOTES, 'UTF-8') . '

'; + } + if (isset($json['thumbnailUrl'])) { + $item['enclosures'] = [$json['thumbnailUrl']]; + } + + if (isset($json['duration'])) { + $item['content'] .= "

Duration: {$json['duration']}

"; + $item['itunes:duration'] = $this->parseDurationToSeconds($json['duration']); + } + + return $item; + } + + private function createItemFromHtml($video, string $account): array + { + $href = $video->find('a', 0)->href ?? ''; + $item = [ + 'title' => $video->find('h3', 0)->plaintext ?? 'Untitled', + 'author' => $account . '@rumble.com', + 'content' => $this->defaultLinkTo($video->innertext, self::URI), + 'uri' => self::URI . ltrim($href, '/'), + ]; + + $time = $video->find('time', 0); + if ($time) { + $item['timestamp'] = (new DateTime($time->getAttribute('datetime')))->getTimestamp(); + } + + return $item; + } + + private function parseDurationToSeconds(string $duration): string + { + if (preg_match('/PT(\d+H)?(\d+M)?(\d+S)?/', $duration, $matches)) { + $hours = (int) str_replace('H', '', $matches[1] ?? 0); + $minutes = (int) str_replace('M', '', $matches[2] ?? 0); + $seconds = (int) str_replace('S', '', $matches[3] ?? 0); + return (string) ($hours * 3600 + $minutes * 60 + $seconds); + } + return $duration; } public function getName() { - if ($this->getInput('account')) { - return 'Rumble.com - ' . $this->getInput('account'); - } - return self::NAME; + return $this->getInput('account') ? "Rumble.com - {$this->getInput('account')}" : self::NAME; + } + + public function getCacheTimeout() + { + return (int) $this->getInput('cache_timeout') ?: self::CACHE_TIMEOUT; } }