惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

爱范儿
爱范儿
Security Latest
Security Latest
NISL@THU
NISL@THU
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
C
Cybersecurity and Infrastructure Security Agency CISA
Cloudbric
Cloudbric
T
Threat Research - Cisco Blogs
大猫的无限游戏
大猫的无限游戏
C
CXSECURITY Database RSS Feed - CXSecurity.com
阮一峰的网络日志
阮一峰的网络日志
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
雷峰网
雷峰网
C
Cisco Blogs
V
Vulnerabilities – Threatpost
S
Security Archives - TechRepublic
V
Visual Studio Blog
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
cs.AI updates on arXiv.org
cs.AI updates on arXiv.org
J
Java Code Geeks
D
Darknet – Hacking Tools, Hacker News & Cyber Security
Know Your Adversary
Know Your Adversary
博客园 - 叶小钗
腾讯CDC
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
P
Privacy International News Feed
P
Palo Alto Networks Blog
博客园_首页
V
V2EX
WordPress大学
WordPress大学
Schneier on Security
Schneier on Security
月光博客
月光博客
博客园 - 司徒正美
Google DeepMind News
Google DeepMind News
TaoSecurity Blog
TaoSecurity Blog
博客园 - 聂微东
酷 壳 – CoolShell
酷 壳 – CoolShell
人人都是产品经理
人人都是产品经理
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
博客园 - 【当耐特】
The Cloudflare Blog
罗磊的独立博客
美团技术团队
N
News | PayPal Newsroom
K
KPMG report finds enterprise disconnect between AI and its ROI | CIO
Last Week in AI
Last Week in AI
K
Kaspersky official blog
Google Online Security Blog
Google Online Security Blog
S
SegmentFault 最新的问题
Application and Cybersecurity Blog
Application and Cybersecurity Blog
T
Tailwind CSS Blog

优世界

OpenWrt 路由器改纯 AP 模式记录(Cudy TR3000 + 中兴 F50) Ubuntu 24.04 安装 Zabbix 8.0 全记录 我的静态博客动态化方案 Claude Code 启动脚本 我的5G路由器方案:中兴F50+cudy tr3000 256MB 还是弃用了使用多年的全拼输入法 临走前一份襄阳牛肉面 EdgeOne Pages 部署 Twikoo 评论 一个朋友圈风格 Hugo 主题 搓了一个仿朋友圈的Hugo主题 一个人烧烤被拒单了 周末闲暇时间翻修了一下博客 我把Vercel换成了EdgeOne Pages Artalk评论区接入AI摘要的尝试 给Hugo博客添加瀑布流相册功能 Hugo静态博客如何实现搜索功能 Artalk评论系统实现段落评论功能 为什么我觉得网页昼夜切换那么重要 博客六周年:从折腾到回归平淡 如何hugo静态实现友联朋友圈功能 脂溢性皮炎的烦恼 还是放弃了iPhone16e 米环勿扰同步问题 除草日记 Hugo使用GitHub Action自动刷新多吉云CDN缓存 迁移博客至hugo 添加ikun摆件 parsec远程软件报6023错误 win11右键菜单改回win10方法 主板电流声 主机配置单 一次点亮 访客体验优化 博客除草 老爷机升级 完善主题 家乡随拍 襄阳唐城 进厂日记 苏州一日游 四月随笔 评论置顶 Time Taker 近来二三事 感谢哥哥给的网站 又又又双叒叕换主题啦 生活篇:疫情放开前后的这些日子 .cc域名后缀续费即将涨价 Apple Watch Series7 三个月的使用体验 2022年·襄阳第一场雪,谨以此片记录 记一次莫名其妙ddos攻击,致谢天御云高防cdn 毕业篇:不出意外,这是我最后几个月的大学生活啦 毕业纪念篇:图书馆 生活篇:我的三年封校生活 没错,我回来了,湖北管局一天不到通过备案,强的! 因需要更换备案主体,临时关闭博客通知 记录篇:将笔记本联想小新pro13网卡由螃蟹网卡换成AX210网卡 主力机由荣耀20切换到iPhone13使用半年的感受 生活篇:温馨提示假期余额不足,浅浅总结一下这个暑假 typecho实现QQ头像用户评论加密,注意:pigeon,twitter用户有彩蛋哦! 生活篇:离校,暑假,租房,面试,致那些日子的琐碎事 生活篇:我跳绳的那些日子,谈谈自己的变化 去除typecho1.2.0正式版的后台提示更新bug 利用fontspider压缩博客字体大小,达到加快访问速度 iphone快捷指令发布动态说说,支持大部分typecho主题 寒假二三事 2022,除夕过后的那些事 2022,致我的春节回忆录 祝大家元旦快乐,给自己的博客加一个对联和灯笼 更换掉jsdelivr,改用腾讯云静态网站托管,网站速度比之前提升了不少 这次的落日比较有特色,西边摇摇欲坠的咸鸭蛋 盘点一下建站以来所注册的域名,我居然注册了一堆学费米 可惜不能一直做小孩子,总要长大,不知道实习生活会是什么样子 这组照片的主题,咱就叫它光吧 双十一已经变味了,不知道从什么时候反感双十一了 第一次尝试ai画简笔画,本是给自己设计logo误打误撞画起了简笔画 不用改变图片原地址,实现图片自动转webp格式,速看,一会删 让typecho支持webp格式的图片,告别阿里云oss和腾讯云cos被恶意刷流量的风险 运动会闲暇这几天,简单给大家分享一下日常吧 我又双叒叕换主题,我发现我好像一直在折腾 记录人生第一次洗牙,不得不说感觉真的特别好 我为什么要写博客?这位博主给了我答案 我想,这是一个我人生中最特殊的中秋国庆节 理工的晚霞,拿起相机记录青春的样子 别让抖音支配了你的美好大学生活,尝试做一些自己感兴趣的事 一岁一礼,一寸欢喜,生日快乐,致我的二十岁 2021暑假总结,记录一下这个充实的暑假 Twitter主题设置仿mac UI 语法高亮代码方法 Twitter主题加入加载耗时,访问总量功能 一把过,科二结束! 再次投资科目二,科目二花了400,希望19号一把过吧 宝塔面板设置Typecho伪静态去掉index.php教程 开学倒计时,再见,老家 或许不是没有年味了,只是快乐已经不属于我们这一辈人了 五福开奖,你好,2021 新春快乐!牛年大吉! 祝母校越办越好 军训太痛苦了,希望早点结束 高考加油,相信自己 高考即将来临,老师为我们加油打气
博客友链实时健康监测方案
2026-02-01 · via 优世界

最近给博客加上了一个实用的功能:友链实时健康监测

起因是备案审查越来越严格。对于个人博客来说,如果友链中的某个域名过期被抢注,摇身一变成为博彩或色情网站,而我的博客还挂着它的链接,那么我的域名很有可能因为“导流违规内容”而被注销备案,甚至面临关站风险。

为了避免这种“人在家中坐,锅从天上来”的情况,手动每天一个个点开检查显然是不现实的。于是,利用 GitHub Actions 和简单的 Node.js 脚本实现自动化监测,就成了最佳解决方案。

实现思路

整体流程分为三个环节:

  1. 数据源:读取博客现有的友链配置文件(YAML 格式)。
  2. 监测脚本:编写 Node.js 脚本,利用并发和连接复用技术,批量检测 HTTP 状态码。
  3. 自动化与通知:通过 GitHub Actions 定时执行,生成状态报告(JSON),并在发现异常时发送邮件通知。
  4. 前端展示:在页面上以不打扰的 UI 风格(黑白极简风格)展示友链状态。

1. 编写监测脚本

scripts/check_links.js 中,我使用了 axios 发送请求,并配合 https.Agent 实现了连接复用(Keep-Alive),同时通过并发池控制请求速率。

这个脚本的核心亮点是:

  • 并发控制:开启 10 个 Worker 并行检测,大幅缩短耗时。
  • 连接复用:开启 keepAlive: true,减少 TCP/TLS 握手开销。
  • 智能重试:优先使用 HEAD 请求(省流量),失败后自动降级为 GET 请求。
  • 异常汇总:将失效链接写入 failed_links_summary.txt,用于后续邮件通知。
const fs = require('fs');
const yaml = require('js-yaml');
const axios = require('axios');
const path = require('path');
const https = require('https');
const http = require('http');

// ... 路径配置 ...

// 优化 Axios 实例:开启 Keep-Alive 复用连接
const axiosInstance = axios.create({
    timeout: 10000,
    maxRedirects: 5,
    httpAgent: new http.Agent({ keepAlive: true }),
    httpsAgent: new https.Agent({ keepAlive: true }),
    headers: {
        'User-Agent': 'Mozilla/5.0 ...' // 模拟浏览器 UA
    },
    validateStatus: (status) => status >= 200 && status < 400
});

async function checkUrl(url) {
    try {
        await axiosInstance.head(url);
        return true;
    } catch (error) {
        try {
            await axiosInstance.get(url); // HEAD 失败尝试 GET
            return true;
        } catch (err2) {
            return false;
        }
    }
}

// ... 并发 Worker 逻辑 ...

2. 配置 GitHub Actions

为了保证数据的实时性,我在 .github/workflows/upy.yml 中配置了定时任务。

  • 定时触发:每 12 小时自动运行一次 (cron: '0 */12 * * *')。
  • 邮件通知:如果发现失效友链,Workflow 会读取脚本生成的摘要文件,并通过邮件推送到我的邮箱。
on:
  schedule:
    - cron: '0 */12 * * *'   # 每12小时自动检查一次

jobs:
  deploy:
    steps:
    # ... 其他步骤 ...

    - name: Check Link Status
      run: node scripts/check_links.js

    - name: Read Failed Links Summary
      if: always()
      run: |
        if [ -f failed_links_summary.txt ]; then
          echo "LINK_SUMMARY<<EOF" >> $GITHUB_ENV
          cat failed_links_summary.txt >> $GITHUB_ENV
          echo "EOF" >> $GITHUB_ENV
        fi        

    # ... 构建步骤 ...

    - name: Send deployment success notification
      if: success()
      uses: dawidd6/action-send-mail@v3
      with:
        # ... 邮件配置 ...
        body: |
          # ... 其他信息 ...
          
          📊 友链监测报告
          ----------------------------------
          ${{ env.LINK_SUMMARY }}          

3. 前端展示:极简设计

为了不破坏博客原本的黑白极简风格,我并没有使用传统的“红/绿”大色块,而是设计了一套低调的状态指示器。

JS 逻辑 (main.js)

前端脚本会异步加载 link_status.json,并智能匹配 URL(自动处理 https 和尾部 / 的差异)。

window.initLinkStatus = function() {
    fetch('/link_status.json?t=' + new Date().getTime())
        .then(response => response.json())
        .then(data => {
            // ... 遍历 DOM 元素 ...
            // getUrlStatus 逻辑:尝试匹配 url, url/, https/http 变体
            
            if (status === 'alive') {
                $item.append('<span class="link-active-badge" title="可访问"></span>');
            } else if (status === 'dead') {
                $item.append('<span class="link-dead-badge" title="无法访问"></span>');
            }
        });
};

CSS 样式 (main.css)

这里花了一些心思调整配色。为了避免“红绿灯”配色的突兀感:

  • 正常状态:使用实心灰点 (#9ca3af),在暗色模式下为浅灰 (#d1d5db)。
  • 失效状态:使用空心灰圈(白色/深色中心 + 灰色边框)。

这种设计既能传达状态信息,又完美融入了黑白主题。

/* Card View - Active (实心灰点) */
.link-active-badge {
    position: absolute;
    top: 8px; right: 8px;
    width: 8px; height: 8px;
    background-color: #9ca3af;
    border-radius: 50%;
    border: 2px solid #fff;
}

/* Card View - Dead (空心灰圈) */
.link-dead-badge {
    position: absolute;
    top: 8px; right: 8px;
    width: 8px; height: 8px;
    background-color: #fff;
    border-radius: 50%;
    border: 2px solid #9ca3af; /* 灰色边框形成空心效果 */
}

总结

这套方案最大的好处是零成本、自动化且优雅

  1. 零成本:完全利用 GitHub Actions 的计算资源。
  2. 自动化:定时检查 + 邮件报警,无需人工干预。
  3. 优雅:前端 UI 克制且统一,不喧宾夺主。

对于注重 SEO 和网站安全的博主来说,这是一个性价比极高的“基建”工作。既对自己负责,也对访客负责。毕竟,谁也不想点击一个链接后发现是 404,或者跳转到奇怪的页面吧。