mirror of
https://github.com/RSS-Bridge/rss-bridge.git
synced 2025-04-05 00:59:35 +00:00
fix(rumble): improve bridge
This commit is contained in:
parent
05a9ac0f06
commit
894cdee57e
@ -4,17 +4,17 @@ class RumbleBridge extends BridgeAbstract
|
|||||||
{
|
{
|
||||||
const NAME = 'Rumble.com Bridge';
|
const NAME = 'Rumble.com Bridge';
|
||||||
const URI = 'https://rumble.com/';
|
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 MAINTAINER = 'dvikan, NotsoanoNimus';
|
||||||
const CACHE_TIMEOUT = 60 * 60; // 1h
|
const CACHE_TIMEOUT = 0;
|
||||||
|
|
||||||
const PARAMETERS = [
|
const PARAMETERS = [
|
||||||
[
|
[
|
||||||
'account' => [
|
'account' => [
|
||||||
'name' => 'Account',
|
'name' => 'Account',
|
||||||
'type' => 'text',
|
'type' => 'text',
|
||||||
'required' => true,
|
'required' => true,
|
||||||
'title' => 'Name of the target account to create into a feed.',
|
'title' => 'Name of the target account (e.g., 21UhrBitcoinPodcast)',
|
||||||
'defaultValue' => 'bjornandreasbullhansen',
|
|
||||||
],
|
],
|
||||||
'type' => [
|
'type' => [
|
||||||
'name' => 'Account Type',
|
'name' => 'Account Type',
|
||||||
@ -27,6 +27,12 @@ class RumbleBridge extends BridgeAbstract
|
|||||||
'User (All)' => 'user',
|
'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');
|
$account = $this->getInput('account');
|
||||||
$type = $this->getInput('type');
|
$type = $this->getInput('type');
|
||||||
$url = self::getURI();
|
$url = self::URI;
|
||||||
|
|
||||||
if (!preg_match('#^[\w\-_.@]+$#', $account) || strlen($account) > 64) {
|
if (!preg_match('#^[\w\-_.@]+$#', $account) || strlen($account) > 64) {
|
||||||
throw new \Exception('Invalid target account.');
|
returnServerError('Invalid target account.');
|
||||||
}
|
}
|
||||||
|
|
||||||
switch ($type) {
|
switch ($type) {
|
||||||
@ -54,40 +60,106 @@ class RumbleBridge extends BridgeAbstract
|
|||||||
$url .= "c/$account/livestreams";
|
$url .= "c/$account/livestreams";
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
// Shouldn't ever happen.
|
returnServerError('Invalid media type.');
|
||||||
throw new \Exception('Invalid media type.');
|
|
||||||
}
|
}
|
||||||
|
|
||||||
$dom = getSimpleHTMLDOM($url);
|
$html = $this->getContents($url);
|
||||||
foreach ($dom->find('ol.thumbnail__grid div.thumbnail__grid--item') as $video) {
|
if (!$html) {
|
||||||
$href = $video->find('a', 0)->href;
|
returnServerError("Failed to fetch $url");
|
||||||
|
}
|
||||||
|
|
||||||
$item = [
|
$items = [];
|
||||||
'title' => $video->find('h3', 0)->plaintext,
|
if (preg_match('/<script.*?application\/ld\+json.*?>(.*?)<\/script>/s', $html, $matches)) {
|
||||||
'author' => $account . '@rumble.com',
|
$jsonData = json_decode($matches[1], true);
|
||||||
'content' => defaultLinkTo($video, self::URI)->innertext,
|
if ($jsonData) {
|
||||||
];
|
$videos = isset($jsonData['@graph']) ? $jsonData['@graph'] : [$jsonData];
|
||||||
|
foreach ($videos as $item) {
|
||||||
$time = $video->find('time', 0);
|
if (isset($item['@type']) && $item['@type'] === 'VideoObject') {
|
||||||
if ($time) {
|
$items[] = $this->createItemFromJsonLd($item, $account);
|
||||||
$publishedAt = new \DateTimeImmutable($time->getAttribute('datetime'));
|
}
|
||||||
$item['timestamp'] = $publishedAt->getTimestamp();
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$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'] .= "<iframe src='{$json['embedUrl']}' frameborder='0' allowfullscreen></iframe>";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isset($json['description'])) {
|
||||||
|
$item['content'] .= '<p>' . html_entity_decode($json['description'], ENT_QUOTES, 'UTF-8') . '</p>';
|
||||||
|
}
|
||||||
|
if (isset($json['thumbnailUrl'])) {
|
||||||
|
$item['enclosures'] = [$json['thumbnailUrl']];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isset($json['duration'])) {
|
||||||
|
$item['content'] .= "<p>Duration: {$json['duration']}</p>";
|
||||||
|
$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()
|
public function getName()
|
||||||
{
|
{
|
||||||
if ($this->getInput('account')) {
|
return $this->getInput('account') ? "Rumble.com - {$this->getInput('account')}" : self::NAME;
|
||||||
return 'Rumble.com - ' . $this->getInput('account');
|
}
|
||||||
}
|
|
||||||
return self::NAME;
|
public function getCacheTimeout()
|
||||||
|
{
|
||||||
|
return (int) $this->getInput('cache_timeout') ?: self::CACHE_TIMEOUT;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user