rss-bridge/bridges/NationalGeographicBridge.php

376 lines
12 KiB
PHP
Raw Normal View History

<?php
class NationalGeographicBridge extends BridgeAbstract
{
const CONTEXT_BY_TOPIC = 'By Topic';
const PARAMETER_TOPIC = 'topic';
const PARAMETER_FULL_ARTICLE = 'full';
const TOPIC_MAGAZINE = 'Magazine';
const TOPIC_LATEST_STORIES = 'Latest Stories';
const CACHE_TIMEOUT = 900; //15 min
const NAME = 'National Geographic';
const URI = 'https://www.nationalgeographic.com/';
const DESCRIPTION = 'Fetches the latest articles from the National Geographic Magazine';
const MAINTAINER = 'csisoap';
const PARAMETERS = [
self::CONTEXT_BY_TOPIC => [
self::PARAMETER_TOPIC => [
'name' => 'Topic',
'type' => 'list',
'values' => [
self::TOPIC_MAGAZINE => 'magazine',
self::TOPIC_LATEST_STORIES => 'latest-stories'
],
'title' => 'Select your topic',
'defaultValue' => 'Magazine'
]
],
'global' => [
self::PARAMETER_FULL_ARTICLE => [
'name' => 'Full Article',
'type' => 'checkbox',
'title' => 'Enable to load full articles and other infos (takes longer)'
]
]
];
private $topicName = '';
const CONTEXT = 'eyJjb250ZW50VHlwZSI6IlVuaXNvbkh1YiIsInZhcmlhYmxlcyI6eyJsb2NhdG9yIjoiL3BhZ2VzL3
RvcGljL2xhdGVzdC1zdG9yaWVzIiwicG9ydGZvbGlvIjoibmF0Z2VvIiwicXVlcn
lUeXBlIjoiTE9DQVRPUiJ9LCJtb2R1bGVJZCI6bnVsbH0';
const LATEST_STORIES_ID = [
'1df278bb-0e3d-4a67-a0ce-8fae48392822-f2-m1'
];
const MAGAZINE_ID = [
'94d87d74-f41a-4a32-9acd-b591ba2df288-f2-m1',
'94d87d74-f41a-4a32-9acd-b591ba2df288-f5-m2',
];
public function getURI()
{
switch ($this->queriedContext) {
case self::CONTEXT_BY_TOPIC:
return self::URI . $this->getInput(self::PARAMETER_TOPIC);
default:
return parent::getURI();
}
}
private function getAPIURL($id)
{
$context = preg_replace('/\s*/m', '', self::CONTEXT);
$url = 'https://www.nationalgeographic.com/proxy/hub?context='
. $context . '&id=' . $id
. '&moduleType=InfiniteFeedModule&_xhr=pageContent';
return $url;
}
public function collectData()
{
$this->topicName = $this->getTopicName($this->getInput(self::PARAMETER_TOPIC));
switch ($this->topicName) {
case self::TOPIC_MAGAZINE:
return $this->collectMagazine();
case self::TOPIC_LATEST_STORIES:
return $this->collectLatestStories();
default:
returnServerError('Unknown topic: "' . $this->topicName . '"');
}
}
public function getName()
{
switch ($this->queriedContext) {
case self::CONTEXT_BY_TOPIC:
return static::NAME . ': ' . $this->topicName;
default:
return parent::getName();
}
}
private function getTopicName($topic)
{
return array_search($topic, static::PARAMETERS[self::CONTEXT_BY_TOPIC][self::PARAMETER_TOPIC]['values']);
}
private function collectMagazine()
{
$stories = [];
foreach (self::MAGAZINE_ID as $id) {
$uri = $this->getAPIURL($id);
$json_raw = getContents($uri);
$json = json_decode($json_raw, true)['tiles'];
$stories = array_merge($json, $stories);
}
foreach ($stories as $story) {
$this->addStory($story);
}
}
private function collectLatestStories()
{
$stories = [];
foreach (self::LATEST_STORIES_ID as $id) {
$uri = $this->getAPIURL($id);
$json_raw = getContents($uri);
$json = json_decode($json_raw, true)['tiles'];
$stories = array_merge($stories, $json);
}
foreach ($stories as $story) {
$this->addStory($story);
}
}
private function addStory($story)
{
$title = 'Unknown title';
$content = '';
$story_type = '';
$uri = '';
foreach ($story['ctas'] as $component) {
$uri = $component['url'];
$story_type = $component['icon'];
}
$item = [];
if (isset($story['description'])) {
$content = '<p>' . $story['description'] . '</p>';
}
$title = $story['title'];
$item['uri'] = $uri;
$item['title'] = $story['title'];
// if full article is requested!
if ($this->getInput(self::PARAMETER_FULL_ARTICLE)) {
if ($story_type != 'interactive') {
/* Nat Geo doesn't provided much info about interactive page
* and it requires JS to load the interactive.
*/
$article_data = $this->getFullArticle($item['uri']);
$item['timestamp'] = $article_data['published_date'];
$item['author'] = $article_data['authors'];
$item['content'] = $content . $article_data['content'];
} else {
$item['content'] = $content;
}
} else {
$item['content'] = $content;
}
$image = $story['img'];
2024-03-31 20:07:14 +02:00
$item['enclosures'][] = str_replace(' ', '%20', $image['src']);
2022-10-26 00:47:45 +02:00
foreach ($story['tags'] as $tag) {
$item['categories'][] = $tag['name'] ?? $tag;
}
$this->items[] = $item;
}
private function filterArticleData($data)
{
$article_module = array_filter(
$data,
function ($item) {
if (isset($item['id']) && $item['id'] == 'natgeo-template1-frame-1') {
return true;
}
}
);
$article_data = array_reduce(
$article_module,
function (array $carry, array $item) {
$module = $item['mods'];
return array_merge(
$carry,
array_filter(
$module,
function ($data) {
return $data['id'] == 'natgeo-template1-frame-1-module-1';
}
)
);
},
[]
);
return $article_data[0];
}
private function handleImages($image_module, $image_type)
{
$image_alt = '';
$image_credit = '';
$image_src = '';
$image_caption = '';
$caption = '';
switch ($image_type) {
case 'image':
case 'imagegroup':
2024-03-31 20:07:14 +02:00
$image = $image_module['image'] ?? null;
if (!$image) {
return '';
}
$image_src = $image['src'];
if (isset($image_module['alt'])) {
$image_alt = $image_module['alt'];
} elseif (isset($image['altText'])) {
$image_alt = $image['altText'];
}
if (isset($image['crdt'])) {
$image_credit = $image['crdt'];
}
$caption = ($image_module['caption'] ?? '');
break;
case 'photogallery':
$image_credit = ($image_module['caption']['credit'] ?? '');
$caption = $image_module['caption']['text'];
$image_src = $image_module['img']['src'];
$image_alt = $image_module['img']['altText'];
break;
case 'video':
$image_credit = ($image_module['credit'] ?? '');
$description = ($image_module['description'] ?? '');
$caption = $description . ' Video can be watched on the article\'s page';
$image = $image_module['image'];
$image_alt = $image['altText'];
$image_src = $image['src'];
}
$image_caption = $caption . ' ' . $image_credit
. '. Notes: Some image may have copyrighted on it.';
$wrapper = <<<EOD
<figure>
<img src="{$image_src}" alt="{$image_alt}">
<figcaption>$image_caption</figcaption>
</figure>
EOD;
return $wrapper;
}
private function getFullArticle($uri)
{
$html = getContents($uri);
$scriptRegex = '/window\[\'__natgeo__\'\]=(.*);<\/script>/';
preg_match($scriptRegex, $html, $matches, PREG_OFFSET_CAPTURE, 0);
$json = json_decode($matches[1][0], true);
2024-03-31 20:07:14 +02:00
if (isset($json['page']['content']['article']['frms'])) {
$unfiltered_data = $json['page']['content']['article']['frms'];
} else {
$unfiltered_data = $json['page']['content']['prismarticle']['frms'];
}
$filtered_data = $this->filterArticleData($unfiltered_data);
$article = $filtered_data['edgs'][0];
$contributors = $article['cntrbGrp'];
$authors = [];
if (count($contributors) > 0) {
$authors = $contributors[0]['contributors'];
}
$authors_name = '';
$counter = 0;
foreach ($authors as $author) {
$counter++;
if ($counter == count($authors)) {
$authors_name .= $author['displayName'];
} else {
$authors_name .= $author['displayName'] . ', ';
}
}
2024-03-31 20:07:14 +02:00
$published_date = $article['pbDt'] ?? $article['dt'];
$article_body = $article['bdy'];
$content = '';
foreach ($article_body as $body) {
switch ($body['type']) {
case 'p':
$content .= '<p>' . $body['cntnt']['mrkup'] . '</p>';
break;
case 'h2':
$content .= '<h2>' . $body['cntnt']['mrkup'] . '</h2>';
break;
case 'inline':
$module = $body['cntnt'];
if (empty($module)) {
continue 2;
}
switch ($module['cmsType']) {
case 'image':
$content .= $this->handleImages($module, $module['cmsType']);
break;
case 'imagegroup':
$images = $module['images'];
foreach ($images as $image) {
$content .= $this->handleImages($image, $module['cmsType']);
}
break;
case 'editorsNote':
$content .= $module['note'];
break;
case 'listicle':
$content .= '<h2>' . ($module['title'] ?? '(no title)') . '</h2>';
if (isset($module['image'])) {
$content .= $this->handleImages($module['image'], $module['image']['cmsType']);
}
$content .= '<p>' . ($module['text'] ?? '') . '</p>';
break;
case 'photogallery':
$gallery = $body['cntnt']['media'];
foreach ($gallery as $image) {
$content .= $this->handleImages($image, $module['cmsType']);
}
break;
case 'video':
$content .= $this->handleImages($module, $module['cmsType']);
break;
case 'pullquote':
$quote = $module['quote'];
$author_name = '';
$authors = ($module['byLineProps']['authors'] ?? []);
foreach ($authors as $author) {
$author_desc = ($author['authorDesc'] ?? '');
$author_name .= $author['displayName'] . ', ' . $author_desc;
}
$content .= <<<EOD
<figure>
<blockquote>
<p>$quote</p>
</blockquote>
<figcaption>$author_name</figcaption>
</figure>
EOD;
break;
}
break;
case 'ul':
$content .= $body['cntnt']['mrkup'] . '<hr>';
break;
}
}
return [
'content' => $content,
'published_date' => $published_date,
'authors' => $authors_name
];
}
}