如何快速为 VitePress 添加 RSS 订阅支持

本文涉及的产品
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: 前言在看许多个人博客站点的时候,右上角总会有个RSS订阅的标志恰好我的博客也是基于 VitePress 搭建的,就想看看能不能也实现这个功能呢?动手前先搜了一下,先是看到了vitepress-blog-zaun上有这个RSS的实现支持,再搜了一下发现Vue的官方博客 vuejs/blog 也是用的这样的实现大概就是自定义 VitePress 的 buildEnd 钩子,在里面实现逻辑获取 md 文件列表,然后通过 feed 生成 RSS 文件,整个逻辑就 50+ 行代码

前言

在看许多个人博客站点的时候,右上角总会有个RSS订阅的标志

恰好我的博客也是基于 VitePress 搭建的,就想看看能不能也实现这个功能呢?

动手前先搜了一下,先是看到了vitepress-blog-zaun上有这个RSS的实现支持,再搜了一下发现Vue的官方博客 vuejs/blog 也是用的这样的实现

大概就是自定义 VitePress 的 buildEnd 钩子,在里面实现逻辑获取 md 文件列表,然后通过 feed 生成 RSS 文件,整个逻辑就 50+ 行代码

由于我的博客还分离了独立的主题包 @sugarat/theme,我想把这个功能加到我的主题包里,这样使用这个主题的就可以简单的配置一下就能使用了,当然也为了方便广大 VitePress 用户更加简便的使用,我将这段逻辑单独分离封装到了 vitepress-plugin-rss 这个插件里。

接下来我将会先介绍一下如何食用这个插件,再介绍它的核心实现原理

插件使用

通过 pnpm/npm/yarn 安装插件

pnpm add vitepress-plugin-rss

.vitepress/config.ts 配置文件中添加配置使用

下面是最基础的使用配置

import { RssPlugin, RSSOptions } from 'vitepress-plugin-rss'
const baseUrl = 'https://sugarat.top'
const RSS: RSSOptions = {
  title: '粥里有勺糖',
  baseUrl,
  copyright: 'Copyright (c) 2018-present, 粥里有勺糖',
}
export default defineConfig({
  vite: {
    // ↓↓↓↓↓
    plugins: [RssPlugin(RSS)]
    // ↑↑↑↑↑
  }
})

然后运行 build 命令,你可以看到在rendering pages...后打印了生成 feed.rss 日志...

pnpm run build

image.png

同时会在导航栏的 socialLinks 中添加 rss 图标链接

image.png

使用是不是非常简单,只需要 10 行代码。

如果你对插件的实现原理感兴趣,请接着往下看 🎉 🎉 🎉。

核心实现原理解析

VitePress 的拓展在官方文档 Use Cases 部分有提到

image.png

其是基于 Vite 的,因此可以使用 Vite 的插件机制来实现主题内容的拓展。

buildEnd 修改

从官方的demo种可以看到,RSS 的生成逻辑是放在 buildEnd 中的,因此咱们插件也需要实现间接修改 buildEnd 方法

这个非常的简单,利用 Vite 的插件提供的 configResolved 钩子就行

下面是简单的demo

ts

复制代码

import { SiteConfig } from'vi
import { SiteConfig } from 'vitepress'
let resolveConfig: any = null
function configResolved(config: any) {
  // 避免多次执行
  if (resolveConfig) {
    return
  }
  resolveConfig = config
  const VPConfig: SiteConfig = config.vitepress
  if (!VPConfig) {
    return
  }
  const selfBuildEnd = VPConfig.buildEnd
  // 自定义 buildEnd 方法,添加 rss 生成支持
  VPConfig.buildEnd = async (siteConfig: any) => {
    // 调用自己的
    await selfBuildEnd?.(siteConfig)
    console.log('buildEnd', '生成 rss 文件');
  }
}

通过config.vitepress即可拿到vitepress的配置,然后重新定义 buildEnd 方法即可

这里可以直接快速的验证一下

image.png

运行后可以看到打印了 buildEnd 生成 rss 文件,说明我们的插件的修改已经生效了

image.png

icon 添加

这个也非常的简单,VitePress 在官方文档里有介绍 socialLinks

image.png

我们只需要在配置修改中添加一个 socialLinks 的配置即可

接着上述的demo,添加如下代码

VPConfig.site.themeConfig.socialLinks = [
  {
    icon: {
      svg: 'svg icon'
    },
    link: 'rss url'
  },
  ...VPConfig.site.themeConfig.socialLinks
]

svg的图标可以通过 xicons 这个网站查找

比如我这里找了一个 sun 的图标配上

image.png

启动博客后就能看见右上角这个小太阳了

image.png

MD文件获取与解析

这个是最核心的逻辑了,① 需要获取所有的 md 文件,② 解析里面的 frontmatter ③ 渲染HTML

这个在 vuejs/blog 中可以看到使用的是 VitePress 内置的 createContentLoader 方法(里面包含了上述3部分逻辑)

这里把其核心实现拆出来,方便大家理解和更好的自定义(笔者在插件里也没直接使用 createContentLoader 这个方法)

① 通过 fast-glob 获取所有的 md 文件

import glob from 'fast-glob'
const files = glob.sync(`${srcDir}/**/*.md`, { ignore: ['node_modules'] })

 其中 srcDir 即文章所在的目录,可以通过如下方式获取到相对路径

// config 即 SiteConfig
const srcDir =
    config.srcDir.replace(config.root, '').replace(/^\//, '') ||
    process.argv.slice(2)?.[1] ||
    '.'

② 通过 gray-matter 解析 frontmatter

这里frontmatter就是文章开头里两个---之间的内容

例如

---
title: 示例标题
description: 文章介绍
---

利用 gray-matter 解析

import matter from 'gray-matter'
import fs from 'fs'
for (const file of files) {
  const fileContent = fs.readFileSync(file, 'utf-8')
  const { data: frontmatter, excerpt } = matter(fileContent, {
    excerpt: true
  })
}

其中 excerpt 即为文章的摘要信息(description)

③ MD 渲染为 HTML

这个使用 VitePress 提供的 createMarkdownRenderer 即可

 // 由于插件里最后构建成 CJS/ESM 两种格式,VitePress 最新的版本支持 ESM,因此需要动态引入
const { createMarkdownRenderer } = await import('vitepress')
const mdRender = await createMarkdownRenderer(
  config.srcDir,
  config.markdown,
  config.site.base,
  config.logger
)
for (const file of files) {
  const fileContent = fs.readFileSync(file, 'utf-8')
  // 生成html
  const html = mdRender.render(fileContent)
}

RSS文件生成

通过上面的 markdown 文件的解析,我们已经拿到了所有的文章信息,接下来就是通过 feed 这个库生成 RSS 文件了

import { Feed } from 'feed'
const feedOptions = {
  // ...
}
const feed = new Feed(feedOptions)
for (const file of files){
  // 通过前面解析的信息,生成 feed item 
  feed.addItem({
    title,
    id: link,
    link,
    description,
    content: html,
    author: [
      {
        name: author,
        ...authorInfo
      }
    ],
    image: frontmatter?.cover,
    date: new Date(date)
  })
}
const RSSFilename = 'feed.rss'
const RSSFilepath = path.join(config.outDir, RSSFilename)
// 生成 rss 文件
writeFileSync(RSSFilepath, feed.rss2())

最后

插件的完整源码见 GitHub,欢迎大家试用和反馈

相关文章
|
3月前
Nest.js 实战 (十二):优雅地使用事件发布/订阅模块 Event Emitter
这篇文章介绍了在Nest.js构建应用时,如何通过事件/发布-订阅模式使应用程序更健壮、灵活、易于扩展,并简化服务间通信。文章主要围绕@nestjs/event-emitter模块展开,这是一个基于eventemitter2库的社区模块,提供了事件发布/订阅功能,使得实现事件驱动架构变得简单。文章还介绍了如何使用该模块,包括安装依赖、初始化模块、注册EventEmitterModule、使用装饰器简化监听等。最后总结,集成@nestjs/event-emitter模块可以提升应用程序的事件驱动能力,构建出更为松耦合、易扩展且高度灵活的系统架构,是构建现代、响应迅速且具有高度解耦特性的Nest.
|
3月前
|
JSON Java UED
uniapp:使用DCloud的uni-push推送消息通知(在线模式)java实现
以上展示了使用Java结合DCloud的uni-push进行在线消息推送的基本步骤和实现方法。实际部署时,可能需要依据实际项目的规模,业务场景及用户基数进行必要的调整和优化,确保消息推送机制在保证用户体验的同时也满足业务需求。
196 0
|
4月前
|
测试技术 API
Theme——借助github的API来实现主题自动更新
Theme——借助github的API来实现主题自动更新
32 0
|
7月前
|
设计模式 JavaScript 前端开发
JS中发布/订阅模式的简单应用
JS中发布/订阅模式的简单应用
|
JSON 数据格式
DTHttpJson UE4插件使用说明
DTHttpJson UE4插件使用说明
235 0
EMQ
|
存储 物联网 Docker
v1.8.3 进行中:MQTT X CLI 支持多主题订阅,优化输出显示
九月,MQTT 5.0客户端工具MQTT X 1.8.3版本持续开发中,为会话过期间隔添加默认值,并优化了MQTT X CLI(命令行)的默认输出显示。
EMQ
168 0
v1.8.3 进行中:MQTT X CLI 支持多主题订阅,优化输出显示
tiny-emitter.js:一个小型的事件订阅发布库
tiny-emitter.js:一个小型的事件订阅发布库
226 0
|
调度
仿Flow构建器创建数据流
仿Flow构建器创建数据流
139 0
仿Flow构建器创建数据流
|
开发工具 git
为Docsify自动生成RSS订阅
搜索 Docsify 的时候发现了一个在少数派上发布的配置 RSS 订阅的文章
179 0
为Docsify自动生成RSS订阅