跳过正文
  1. Blogs/

博客 AI 摘要及优化

·1732 字·4 分钟· ·
代码 Ai-Summary Hugo Blog AI Summary
目录

为何需要 AI 摘要
#

记不太清了,印象中在中学时代学过一篇课文,是讲如何读报纸的。
我在阅读时一直保持了当时文中介绍的习惯。
对于不需要精读的内容可以跳读。
现在 ChatGPT 爆火后出现了大量的应用,其中 AI 摘要插件对我的帮忙很大。
信息过剩后,我们被很多低价值的信息轰炸,这类插件可以让我快速的辨别一篇文章是否值得细读。
通过多方对比后,我选择了「 ChatGPT 总结助手 - Chrome 应用商店 」作为我浏览网页时的首选插件。
它的使用效果如图:

必须要 AI 摘要吗
#

不必要!

我集成 AI 摘要只是为了方便跟我有一样阅读习惯的人。
我的个人博客在于记录生活和写一些备忘录,对于别人来说全是一些低价值的数据。

我现在在博客上集成了 AI 摘要功能,是因为我使用了 TianliGPT
我当时因为出于“支持独立开发者”的情结,购买了 TianliGPT 的服务。
发现购买的 Key 很快就消耗完了,又买了一个 Key,发现还是不够,再买了一个 Key。
即便如此,我的博客中也只有不到三分之二的文章有 AI 摘要。
这样下去,感觉像个无底洞。
这个时候的 AI 摘要功能就变成了一个历史包袱。
如果取消 AI 摘要功能,那前面花的那些小钱就变成了沉没成本了。

有情怀不代表需要妥协 Geek 精神。
所以决定自己动手改造一下 AI 摘要。
这也符合当初换到 Hugo 静态博客的初衷,尽量把数据都静态化。
尽量让页面数据 SSG(Static-Site Generation)/ SSR(Server-Side Rendering)。

怎么优化 AI 摘要
#

推荐:https://github.com/Moraxyc/ai-summary-hugo 辅助生成 summary.json 的 Python 脚本

1. 方式
#

把 AI 摘要放在本地有 2 个思路:

  1. data 目录下用 .json 文件保存数据

  1. 把摘要结果放在文章 .md 文件的 Front Matter 里

2. 数据格式
#

我选择的是方式 1,把所有 AI 摘要放在 .json 文件中,这样方便后期维护,也不破坏现有文章。
数据来源:

  • TianliGPT 已经生成的摘要
  • 通过脚本从 ChatGPT 和 Claude2 获取的摘要

我现在用了好几个脚本来获取摘要,没有一个完美的脚本能胜任这一工作,所以就不贴脚本了。
TianliGPT 的摘要,可以在网页上按 F12 打开 DevTools 找到:

把多种方式获取到的 AI 摘要集中起来后,生成一个 summary.json 文件。
data 目录新建 data/summary/summary.json 文件,所有数据复制到其中,数据格式:

{
  "summaries": [
    {
      "title": "2010 年 12 月 15 日 雪景",
      "slug": "snows",
      "generated": true,
      "summary": "文章描述了作者在 2010 年 12 月 15 日所经历的雪景。作者提到,下雪之前天空只有几点零星的白天,没人预料到会下这么大的雪。作者认为家乡可能下得更大。文章还提到了关于地址设定和程序错误的问题,并以数字花园用爱发电作为结尾。"
    },
    {
      "title": "2011 向左,2010 向右",
      "slug": "goodbye2010",
      "generated": true,
      "summary": "文章讲述了作者对 2010 年的回顾和对 2011 年的期望。他认为自己在过去一年中太过于为别人而活,失去了自我。他希望新的一年能更加积极阳光,并将经历和感悟当成财富。最后,他表示不再傻逼地诉说苦难,祝大家新年快乐。"
    }
    ]
}

其中 generated 是我自用的一个判断词,可有可无,titleslug 是用于对应文章的锚点。

3. 渲染 AI 摘要
#

summary.json 文件中的 AI 摘要渲染到每篇文章的开头,修改一下 Hugo 文章的模板文件即可。
不同主题的模板文件可能不太一样,不过也差不离。
一般名叫 single.html ,位于主题的 layouts 或 Hugo 根目录的 layouts 下。
比如 DoIt 的模板位于 themes/DoIt/layouts/posts/single.html
我把它复制到根目录 layouts/posts/single.html 了, 同名模板优先渲染根目录
在文章模板 single.html{{ .Content }} 之前,插入以下代码:

<!-- 其他代码 -->

{{ $data := dict }}
{{ $path := "data/summary/summary.json" }}

{{ with resources.Get $path }}
  {{ with . | transform.Unmarshal }}
    {{ $data = . }}
  {{ end }}
{{ else }}
  {{ errorf "Unable to get global resource %q" $path }}
{{ end }}

{{ $currentSlug := .Params.slug }}

{{- if $data.summaries }}
  {{ $matchingSummary := index (where $data.summaries "slug" $currentSlug) 0 }}

  {{- if $matchingSummary }}
    <!-- 渲染 AI 摘要部分 -->
    <div class="post-ai">
        <div class="ai-title">
            <i class="fas fa-robot ai-title-icon"></i>
            <div class="ai-title-text">AI 摘要</div>
        </div>

        <div class="ai-explanation ai-explanation-content">
            {{ if $matchingSummary.summary }}
                {{ $matchingSummary.summary }}
            {{ else }}
                AI 摘要接口暂时失联……
            {{ end }}
        </div>
    </div>
  {{- else }}
    <p>未找到相关内容。</p>
  {{- end }}
{{- else }}
  <p>无法加载总结数据。</p>
{{- end }}

<!-- 渲染文档正文 -->
{{ .Content }}

<!-- 其他代码 -->

不需要打字机效果,上面的内容已经可用了。

如果需要打字机效果,需要取消 <div id="ai-explanation" class="ai-explanation"></div> 的注释。
需要给 <div class="ai-explanation-content" style="display: none;"> 添加 display: none 隐藏掉,它的作用仅给 Typeit 插件赋值。
添加 JS:

<script src="https://unpkg.com/typeit@8.7.1/dist/index.umd.js"></script>

<script>
document.addEventListener("DOMContentLoaded", function () {
    // 从 .ai-explanation-content 取值
    const matchingSummary = document.querySelector(".ai-explanation-content").textContent;

    new TypeIt("#ai-explanation", {
        strings: matchingSummary,
        speed: 50,
        lifeLike: true,
        waitUntilVisible: true,
    }).go();
});
</script>
4. 简单的 CSS
#
.post-ai {
    background: #f5f5f5;
    border-radius: 0.5rem;
    padding: 0.5rem;
    line-height: 1.3;
    border: 1px solid #cfe6f3;
    margin: 1rem 0;
}

.ai-title {
    display: flex;
    color: #2d96bd;
    border-radius: 0.5rem;
    align-items: center;
    padding: 0 0.25rem;
    cursor: default;
    user-select: none;
}

.ai-title-icon {
    width: 20px;
    height: auto;
    margin-right: 0.25rem;
}

.ai-title-text {
    font-weight: bold;
    margin-left: 0.25rem;
    line-height: 1;
}

.ai-explanation {
    margin-top: 1rem;
    padding: 0.5rem 1rem;
    background: #fff;
    border-radius: 0.5rem;
    border: 1px solid #cfe6f3;
    font-size: 0.95rem;
    line-height: 1.4;
    display: inline-block;
    width: 100%;
}

.ai-explanation span {
    margin-left: 0.5rem;
}

/* .ai-explanation-content {
    display: none !important;
} */

Load Giscus
or