• 为 VitePress 网站添加 RSS 订阅支持


    省流:使用 vitepress-plugin-rss 这个插件

    前言

    在看许多个人博客站点的时候,右上角总会有个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
    

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

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

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

    核心实现原理解析

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

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

    buildEnd 修改

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

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

    下面是简单的demo

    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 方法即可

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

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

    icon 添加

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

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

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

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

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

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

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

    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,欢迎大家试用和反馈

    参考

  • 相关阅读:
    go语言操作etcd
    selenium的Chrome116版驱动下载
    JDBC笔记
    【CSP-J 2021】总结
    住院管理管理系统
    K210学习笔记——MaixHub在线训练模型(新版)
    全真模拟题!PMP提分必练
    【Flink 实战系列】Flink CDC 实时同步 Mysql 全量加增量数据到 Hudi
    linux的ping及telnet
    python实现矩阵转置
  • 原文地址:https://www.cnblogs.com/roseAT/p/17649853.html