feat(telegram): add pagination fetching of messages (#4394)

* feat(telegram): add pagination fetching of messages

* docs
This commit is contained in:
Dag 2025-01-04 19:00:26 +01:00 committed by GitHub
parent f9e9c8101e
commit 48cb7d71ed
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
3 changed files with 71 additions and 28 deletions

View file

@ -15,6 +15,14 @@ class TelegramBridge extends BridgeAbstract
] ]
] ]
]; ];
const CONFIGURATION = [
'max_pages' => [
'required' => false,
'defaultValue' => 1,
],
];
const TEST_DETECT_PARAMETERS = [ const TEST_DETECT_PARAMETERS = [
'https://t.me/s/rssbridge' => ['username' => 'rssbridge'], 'https://t.me/s/rssbridge' => ['username' => 'rssbridge'],
'https://t.me/rssbridge' => ['username' => 'rssbridge'], 'https://t.me/rssbridge' => ['username' => 'rssbridge'],
@ -26,7 +34,7 @@ class TelegramBridge extends BridgeAbstract
'https://rssbridge.t.me/' => ['username' => 'rssbridge'], 'https://rssbridge.t.me/' => ['username' => 'rssbridge'],
]; ];
const CACHE_TIMEOUT = 60 * 15; // 15 mins const CACHE_TIMEOUT = 60 * 60; // 1h
private $feedName = ''; private $feedName = '';
private $enclosures = []; private $enclosures = [];
@ -36,33 +44,56 @@ class TelegramBridge extends BridgeAbstract
public function collectData() public function collectData()
{ {
$html = getSimpleHTMLDOM($this->getURI()); $pages = 0;
$url = 'https://t.me/s/' . $this->normalizeUsername();
$channelTitle = $html->find('div.tgme_channel_info_header_title span', 0)->plaintext ?? ''; $max_pages = $this->getOption('max_pages');
// Hard-coded upper bound of 100 loops
while ($pages < $max_pages && $pages < 100) {
$pages++;
$dom = getSimpleHTMLDOM($url);
$channelTitle = $dom->find('div.tgme_channel_info_header_title span', 0)->plaintext ?? '';
$channelTitle = htmlspecialchars_decode($channelTitle, ENT_QUOTES); $channelTitle = htmlspecialchars_decode($channelTitle, ENT_QUOTES);
$this->feedName = $channelTitle . ' (@' . $this->normalizeUsername() . ')'; $this->feedName = $channelTitle . ' (@' . $this->normalizeUsername() . ')';
$posts = $html->find('div.tgme_widget_message_wrap.js-widget_message_wrap');
if (!$channelTitle && !$posts) { $messages = $dom->find('div.tgme_widget_message_wrap.js-widget_message_wrap');
if (!$channelTitle && !$messages) {
throw new \Exception('Unable to find channel. The channel is non-existing or non-public.'); throw new \Exception('Unable to find channel. The channel is non-existing or non-public.');
} }
foreach ($posts as $messageDiv) {
foreach (array_reverse($messages) as $message) {
$this->itemTitle = ''; $this->itemTitle = '';
$this->enclosures = []; $this->enclosures = [];
$item = []; $item = [];
$item['uri'] = $messageDiv->find('a.tgme_widget_message_date', 0)->href; $item['uri'] = $message->find('a.tgme_widget_message_date', 0)->href;
$item['content'] = $this->processContent($messageDiv); $item['content'] = $this->processContent($message);
$item['title'] = $this->itemTitle; $item['title'] = $this->itemTitle;
$item['timestamp'] = $messageDiv->find('span.tgme_widget_message_meta', 0)->find('time', 0)->datetime; $item['timestamp'] = $message->find('span.tgme_widget_message_meta', 0)->find('time', 0)->datetime;
$item['enclosures'] = $this->enclosures; $item['enclosures'] = $this->enclosures;
$messageOwner = $messageDiv->find('a.tgme_widget_message_owner_name', 0); $messageOwner = $message->find('a.tgme_widget_message_owner_name', 0);
if ($messageOwner) { if ($messageOwner) {
$item['author'] = html_entity_decode(trim($messageOwner->plaintext), ENT_QUOTES); $item['author'] = html_entity_decode(trim($messageOwner->plaintext), ENT_QUOTES);
} }
$this->items[] = $item; array_unshift($this->items, $item);
} }
$more = $dom->find('> div.tgme_widget_message_centered.js-messages_more_wrap a', 0);
if ($more && str_contains($more->href, 'before')) {
$url = 'https://t.me/' . $more->href;
} else {
break;
}
}
$this->logger->info(sprintf('Fetched %s messages from %s pages (%s)', count($this->items), $pages, $url));
$this->items = array_reverse($this->items); $this->items = array_reverse($this->items);
} }
@ -369,12 +400,7 @@ EOD;
private function normalizeUsername() private function normalizeUsername()
{ {
// todo: can be replaced with ltrim($username, '@'); return ltrim($this->getInput('username'), '@');
$username = $this->getInput('username');
if (substr($username, 0, 1) === '@') {
return substr($username, 1);
}
return $username;
} }
public function detectParameters($url) public function detectParameters($url)

View file

@ -155,6 +155,11 @@ port = 11211
; --- Bridge specific configuration ------ ; --- Bridge specific configuration ------
[TelegramBridge]
; Max pages to fetch (1 page => 20 messages), min=1 max=100
max_pages = 1
[DiscogsBridge] [DiscogsBridge]
; Sets the personal access token for interactions with Discogs. When ; Sets the personal access token for interactions with Discogs. When

View file

@ -0,0 +1,12 @@
# TelegramBridge
By default, it fetches a single page with up to 20 messages.
To increase this limit, tweak the `max_pages` config:
```ini
[TelegramBridge]
; Fetch a maximum of 3 pages (requires 3 http requests)
max_pages = 3
```