最近偷窥大佬【似水流年】的友链RSS聚合页,在流年兄的大力指导和支持下终于完美上线。

展示:朋友的圈 – 品味苏州 (pwsz.com) 导航简称(友圈)。

此页面聚合了本站友链中经常更新的朋友们的rss数据,终于不用再每次一个个点击网站查看了,据说每年可以节省一个好友鼠标,科技啊,呵~让人们开源节流!

经过大佬的悉心指导,让我也弄懂了一些原理,简单讲讲:

1、创建一个php文件,比如 rss.php ,放在页面模板的文件夹下,据说有的是在主题的 page文件夹,而我的就直接是主题的根目录,又要深深吐槽下island主题作者大伟哥的不靠谱,连个文件夹都不给人家!

<?php
/*
Template Name: 似水流年-RSS 朋友圈
*/

date_default_timezone_set('Asia/Shanghai');
get_header();

// 引入 SimplePie 类
require_once(ABSPATH. WPINC. '/class-simplepie.php');

// 主内容区域
echo '<div id="primary" class="content-area">';
echo '<main id="main" class="site-main" role="main">';
echo '<p style="color: red; font-size: 14px; margin-top: 10px; margin-bottom: 20px; text-align: center; display: flex; align-items: center; justify-content: center;">以下友情链接网站最新内容每 8 小时获取更新一次</p>';

// 获取并缓存友情链接的最新文章数据
function get_cached_friend_link_rss_data() {
$cache_key = 'friend_link_rss_data';
// 尝试从对象缓存中获取数据
$cached_data = wp_cache_get($cache_key);
if ($cached_data) {
return $cached_data;
}
$all_articles = array();
$links = get_bookmarks(array(
'orderby' => 'name',
'category' => 0,
'order' => 'ASC',
'hide_invisible' => 1,
'categorize' => 0,
'title_li' => '',
'echo' => 0
));

foreach ($links as $link) {
if ($link->link_rss &&!empty($link->link_rss)) {
$domain = parse_url($link->link_url, PHP_URL_HOST);
if (!$domain) {
continue;
}

$feed_url = $link->link_rss;
$cache_file = get_stylesheet_directory(). '/rsscache/'. $domain. '.xml';

// 如果缓存文件不存在或超过 8 小时,且 RSS 内容有更新,才更新缓存
if (!file_exists($cache_file) || time() - filemtime($cache_file) > 8 * HOUR_IN_SECONDS) {
$rss_content = get_rss_content($feed_url);
if ($rss_content) {
$existing_content = file_exists($cache_file)? file_get_contents($cache_file) : false;
if ($existing_content!== $rss_content) {
file_put_contents($cache_file, $rss_content);
}
}
}

$rss_items = get_rss_items($cache_file);
foreach ($rss_items as $item) {
$item_data = array(
'link_name' => $link->link_name,
'link_logo' => $link->link_image,
'link_url' => $link->link_url,
'item_title' => $item->get_title(),
'item_link' => $item->get_permalink(),
'item_date' => $item->get_date('Y-m-d H:i:s'),
'item_author' => $item->get_author()->get_name(),
'item_excerpt' => $item->get_description(),
'item_content' => $item->get_content()
);
$all_articles[$link->link_name][] = $item_data;
}
}
}

// 将数据存入对象缓存
wp_cache_set($cache_key, $all_articles, null, 8 * HOUR_IN_SECONDS);
return $all_articles;
}

// 获取最新的友情链接文章
function get_latest_friend_link_articles() {
$latest_articles = get_cached_friend_link_rss_data();
$all_articles = array();
foreach ($latest_articles as $link_name => $articles) {
foreach ($articles as $article) {
$all_articles[] = $article;
}
}

usort($all_articles, function ($a, $b) {
$date1 = date('Y-m-d', strtotime($a['item_date']));
$date2 = date('Y-m-d', strtotime($b['item_date']));
if ($date1 === $date2) {
return strcmp($a['link_name'], $b['link_name']);
} else {
return strtotime($b['item_date']) - strtotime($a['item_date']);
}
});

$sorted_articles = array();
foreach ($all_articles as $article) {
$sorted_articles[$article['link_name']][] = $article;
}

foreach ($sorted_articles as $link_name => $articles) {
usort($articles, function ($a, $b) {
return strtotime($b['item_date']) - strtotime($a['item_date']);
});
$sorted_articles[$link_name] = array_slice($articles, 0, 5);
}

return $sorted_articles;
}

// 获取并保存最新的 RSS 内容
function get_rss_content($feed_url) {
$response = wp_safe_remote_get($feed_url, array('sslverify' => true));
if (!is_wp_error($response) && wp_remote_retrieve_response_code($response) === 200) {
return wp_remote_retrieve_body($response);
} else {
$response = wp_safe_remote_get($feed_url, array('sslverify' => false));
if (!is_wp_error($response) && wp_remote_retrieve_response_code($response) === 200) {
return wp_remote_retrieve_body($response);
} else {
return '无法从提供 RSS 的 URL 获取 RSS 内容。';
}
}
}

// 使用 SimplePie 获取 RSS 项
function get_rss_items($cache_file) {
$rss = new SimplePie();
$rss->set_feed_url($cache_file);
$rss->init();
$rss->handle_content_type();
$rss->set_output_encoding('UTF-8');
return $rss->get_items();
}

// 获取并显示友情链接的最新文章
$latest_articles = get_latest_friend_link_articles();
foreach ($latest_articles as $link_name => $articles) {
?>
<div class="friend-link-container" style="border: 1px dashed #000; padding: 10px; margin-bottom: 10px; display: flex; align-items: center;">
<!-- 左侧显示友情链接的网站 Logo 和名称 -->
<div class="friend-link-info" style="width: 20%; text-align: center;">
<img src="<?php echo esc_url($articles[0]['link_logo']);?>" alt="<?php echo esc_attr($link_name);?> Logo" style="border-radius: 50%; width: 60px; height: 60px;">
<a href="<?php echo esc_url($articles[0]['link_url']);?>" target="_blank"><p style="margin: 0; padding: 0;"><?php echo esc_html($link_name);?></p></a>
</div>
<!-- 右侧显示最新更新的前五篇文章 -->
<div class="friend-link-articles" style="width: 80%; padding-left: 10px;">
<?php foreach ($articles as $article) :?>
<p style="margin: 0; padding: 0;">
<a href="<?php echo esc_url($article['item_link']);?>" target="_blank"><?php echo esc_html($article['item_title']);?></a>(<?php echo esc_html(date('Y-m-d', strtotime($article['item_date'])));?>)
</p>
<?php endforeach;?>
</div>
</div>
<?php
}

// 侧边栏
get_sidebar();

echo '</main><!-- #main -->';
echo '</div><!-- #primary -->';

get_footer();
?>

2、创建一个新页面,填写标题,比如:朋友圈,右侧的模板里,选择刚创建的页面模板“RSS 朋友圈”,保存。以后访问这个页面,就是RSS聚合页。

3、链接里,在高级一栏中填写“图片地址”和“RSS地址”,如不填,则不会出现在聚合页,填写后,就会自动采集RSS数据,8小时更新一次。

4、在主题的根目录,创建一个新的文件夹:rsscache,采集到的RSS数据会保存在这里文件夹里,8小时后再次采集时覆盖保存。

简单四步就搞定了。但是我的博客比较奇葩,实在是太老了(又要吐槽大伟哥,钱收了、手甩了),中间出现不少小bug,流年大佬不辞辛苦搞到半夜,一遍遍的帮我排查、修改代码,最终完美呈现,心中感激无以言表,哈哈哈~再次感谢优秀80后代表流年大佬~

2024.12.16 似水流年更新,优化如下:

减少不必要的缓存获取尝试,利用 Swoole 的协程等功能来实现并发获取 RSS 内容。

同时发起多个 HTTP 请求去获取不同友情链接的 RSS 内容,减少等待时间。

减少排序时的时间格式化操作,用array_multisort 等更高效的排序函数代替usort 这种相对通用但在大数据量下效率不算特别高的排序方式。

设置合理的重试次数、采用指数退避等重试间隔策略,增强在网络不稳定等情况下获取 RSS 内容的成功率。

注意:这个要求PHP版本大于7.4,并且安装redis、swoole或者swoole5扩展。另外 // 头部图片 是我的博客自改的,通用版用老代码中 // 主内容区域 那段代替即可。

<?php
/*
Template Name: RSS 朋友圈
*/

date_default_timezone_set('Asia/Shanghai');
get_header();

// 引入 SimplePie 类
require_once(ABSPATH. WPINC. '/class-simplepie.php');

// 头部图片
echo '<div id="header_slider" class=" fadeInDown animated ">';
echo '<div class="slider"><ul class="lightSlider"><a href="https://pwsz.com/hobby/4493.html#lqmxp"target="_black" title="送你一张明信片"><img width="759" height="289" src="https://pwsz.com/myimg/pydq_banner.jpg" class="attachment-large-image size-large-image wp-post-image" alt="" decoding="async" fetchpriority="high" /></a></ul></div>';

// 获取并缓存友情链接的最新文章数据
function get_cached_friend_link_rss_data() {
$cache_key = 'friend_link_rss_data';
$last_update_time_key = 'last_update_time_'.$cache_key;
$last_update_time = wp_cache_get($last_update_time_key);
if ($last_update_time && (time() - $last_update_time < 7 * HOUR_IN_SECONDS)) {
return wp_cache_get($cache_key);
}

$all_articles = array();
$links = get_bookmarks(array(
'orderby' => 'name',
'category' => 0,
'order' => 'ASC',
'hide_invisible' => 1,
'categorize' => 0,
'title_li' => '',
'echo' => 0
));

$feed_urls = [];
foreach ($links as $link) {
if ($link->link_rss &&!empty($link->link_rss)) {
$domain = parse_url($link->link_url, PHP_URL_HOST);
if (!$domain) {
continue;
}

$feed_urls[] = $link->link_rss;
}
}

// 利用Swoole协程并发获取RSS内容
$rss_contents = get_rss_content_concurrently($feed_urls);
$index = 0;
foreach ($links as $link) {
if ($link->link_rss &&!empty($link->link_rss)) {
$domain = parse_url($link->link_url, PHP_URL_HOST);
if (!$domain) {
continue;
}

$feed_url = $link->link_rss;
$cache_file = get_stylesheet_directory(). '/rsscache/'. $domain. '.xml';

// 获取到的对应RSS内容
$rss_content = $rss_contents[$index];
$index++;

// 如果缓存文件不存在或超过 8 小时,且 RSS 内容有更新,才更新缓存
if (!file_exists($cache_file) || time() - filemtime($cache_file) > 8 * HOUR_IN_SECONDS) {
if ($rss_content) {
$existing_content = file_exists($cache_file)? file_get_contents($cache_file) : false;
if ($existing_content!== $rss_content) {
file_put_contents($cache_file, $rss_content);
}
}
}

$rss_items = get_rss_items($cache_file);
foreach ($rss_items as $item) {
$item_data = array(
'link_name' => $link->link_name,
'link_logo' => $link->link_image,
'link_url' => $link->link_url,
'item_title' => $item->get_title(),
'item_link' => $item->get_permalink(),
'item_date' => $item->get_date('Y-m-d H:i:s'),
'formatted_date' => date('Y-m-d', strtotime($item->get_date('Y-m-d H:i:s'))),
'item_author' => $item->get_author()->get_name(),
'item_excerpt' => $item->get_description(),
'item_content' => $item->get_content()
);
$all_articles[$link->link_name][] = $item_data;
}
}
}

// 将数据存入对象缓存,并设置本次更新时间缓存
wp_cache_set($cache_key, $all_articles, null, 8 * HOUR_IN_SECONDS);
wp_cache_set($last_update_time_key, time());
return $all_articles;
}

// 利用Swoole协程并发获取RSS内容的函数,完善错误处理和重试机制
function get_rss_content_concurrently($feed_urls) {
$results = [];
// 拆分配置项,明确各个参数对应的值
$host = '127.0.0.1';
$port = 80;
$timeout = 1.0; // 设置连接超时时间,单位秒,可根据实际调整
$write_timeout = 1.0; // 设置写超时时间,单位秒,可根据实际调整
$ssl = false; // 这里先设置为不使用SSL连接,可根据实际需求修改为true,如果需要SSL加密的话

$client = new Swoole\Coroutine\Http\Client($host, $port, $ssl);
$client->set([
'timeout' => $timeout,
'write_timeout' => $write_timeout,
]);

if (php_sapi_name() === 'cli') {
// 如果是PHP CLI模式,执行协程相关代码
$max_retries = 3; // 设置最大重试次数
$retry_delay = 1; // 初始重试延迟时间(秒)

foreach ($feed_urls as $index => $feed_url) {
$retry_count = 0;
do {
$response = null;
go(function () use ($index, $feed_url, &$response, $client) {
$response = $client->get($feed_url);
});
while ($response === null) {
Swoole\Coroutine::sleep(0.1);
}
if ($response->statusCode === 200) {
$results[$index] = $response->body;
break; // 获取成功则跳出重试循环
} else {
$retry_count++;
if ($retry_count < $max_retries) {
// 指数退避增加重试延迟时间
$retry_delay *= 2;
Swoole\Coroutine::sleep($retry_delay);
} else {
$results[$index] = '无法从提供 RSS 的 URL 获取 RSS 内容。';
break;
}
}
} while ($retry_count < $max_retries);

}
$client->close();
} else {
// 如果是非CLI模式(比如Web服务器环境等),采用同步方式获取RSS内容,可参考原代码中的同步获取逻辑进行完善
foreach ($feed_urls as $feed_url) {
$response = wp_safe_remote_get($feed_url, array('sslverify' => false));
if (!is_wp_error($response) && wp_remote_retrieve_response_code($response) === 200) {
$results[] = wp_remote_retrieve_body($response);
} else {
$results[] = '无法从提供 RSS 的 URL 获取 RSS 内容。';
}
}
}

return $results;
}

// 获取最新的友情链接文章,改进排序算法
function get_latest_friend_link_articles() {
$latest_articles = get_cached_friend_link_rss_data();
$all_articles = [];
foreach ($latest_articles as $link_name => $articles) {
foreach ($articles as $article) {
$all_articles[] = $article;
}
}

// 先按照格式化日期排序(整体文章排序)
$formatted_date_column = array_column($all_articles, 'formatted_date');
$link_name_column = array_column($all_articles, 'link_name');
array_multisort($formatted_date_column, SORT_DESC, $link_name_column, SORT_ASC, $all_articles);

$sorted_articles = [];
foreach ($all_articles as $article) {
$sorted_articles[$article['link_name']][] = $article;
}

foreach ($sorted_articles as $link_name => $articles) {
// 对每个友情链接下的文章按照原始日期再排序,取前五篇
$date_column = array_column($articles, 'item_date');
array_multisort($date_column, SORT_DESC, $articles);
$sorted_articles[$link_name] = array_slice($articles, 0, 5);
}

return $sorted_articles;
}

// 获取并保存最新的 RSS 内容
function get_rss_content($feed_url) {
$response = wp_safe_remote_get($feed_url, array('sslverify' => true));
if (!is_wp_error($response) && wp_remote_retrieve_response_code($response) === 200) {
return wp_remote_retrieve_body($response);
} else {
$response = wp_safe_remote_get($feed_url, array('sslverify' => false));
if (!is_wp_error($response) && wp_remote_retrieve_response_code($response) === 200) {
return wp_remote_retrieve_body($response);
} else {
return '无法从提供 RSS 的 URL 获取 RSS 内容。';
}
}
}

// 使用SimplePie获取RSS项
function get_rss_items($cache_file) {
$rss = new SimplePie();
$rss->set_feed_url($cache_file);
$rss->init();
$rss->handle_content_type();
$rss->set_output_encoding('UTF-8');
return $rss->get_items();
}

// 获取并显示友情链接的最新文章
$latest_articles = get_latest_friend_link_articles();
foreach ($latest_articles as $link_name => $articles) {
?>
<div class="friend-link-container" style="border: 1px dashed #000; padding: 10px; margin-bottom: 10px; display: flex; align-items: center;">
<!-- 左侧显示友情链接的网站 Logo 和名称 -->
<div class="friend-link-info" style="width: 20%; text-align: center;">
<img src="<?php echo esc_url($articles[0]['link_logo']);?>" alt="<?php echo esc_attr($link_name);?> Logo" style="border-radius: 50%; width: 60px; height: 60px;">
<a href="<?php echo esc_url($articles[0]['link_url']);?>" target="_blank"><p style="margin: 0; padding: 0;"><?php echo esc_html($link_name);?></p></a>
</div>
<!-- 右侧显示最新更新的前五篇文章 -->
<div class="friend-link-articles" style="width: 80%; padding-left: 10px;">
<?php foreach ($articles as $article) :?>
<p style="margin: 0; padding: 0;">
<a href="<?php echo esc_url($article['item_link']);?>" target="_blank"><?php echo esc_html($article['item_title']);?></a>(<?php echo esc_html($article['formatted_date']);?>)
</p>
<?php endforeach;?>
</div>
</div>
<?php
}

// 侧边栏
get_sidebar();

echo '</main><!-- #main -->';
echo '</div><!-- #primary -->';

get_footer();
?>