基于Pagefind实现静态博客站内搜索
立泉Jekyll
生成的静态博客并不支持站内搜索,以往实现要么使用搜索引擎的site:mudan.me
限定站点,要么搭配服务器自建全文搜索服务,但前者索引更新缓慢,后者则太“重”,总之对于倾向免费托管的静态博客并没有特别好的方案。
直到2022年Pagefind出现,它是一个基于Rust
和WebAssembly
的轻量静态搜索工具。先使用其Terminal
命令扫描站点生成索引文件和对应的检索API
,所谓“API”其实就是Wasm
程序提供的Javascript
函数接口,搜索时在Web
页面加载调用它们即可自动访问索引文件生成结果。
这种模式只是从浏览器访问云端静态资源,并不需要通常意义上的“搜索服务器”,而且它的搜索算法也会尽量降低检索时传输的数据量来节省带宽。以本站为例,每次搜索下载的数据低于分段字体,所以不必担心高频访问的问题。
索引
生成索引需要先安装Pagefind
,官方提供npm
包,可以通过npx
直接使用:
# 扫描`--site`指定的目录,将索引输出到`--output-path`指定的目录下
npx -y pagefind --site _site --output-path npm/pagefind
# 注意此命令并不会清除已有数据,而是覆盖
# 所以每次执行前需要手动删除上次的索引,否则可能会越积越多
也可以下载预编译二进制文件安装到本机,注意如果要处理中文,必须安装pagefind_extended
而不是默认的pagefind
,否则搜索结果可能会很奇怪。上面npx
使用的就是extended
版本。
# 解压后需要把`pagefind_extended`可执行文件添加到环境变量中,以方便在Terminal中使用
# 对macOS,只需将其软链接到`/usr/local/bin`中
ln -s /Users/apqx/Documents/MyApp/pagefind_extended-v1.1.1-x86_64-apple-darwin/pagefind_extended /usr/local/bin/pagefind
# 生成索引,由于已经安装到本机,所以省去了`npx`检测网络的步骤,执行时更快
pagefind --site _site --output-path npm/pagefind
执行后会在./npm/pagefind
目录中生成以下内容:
# 索引分页
|-fragment/
|-index/
# 搜索API
|-pagefind.js
# 默认UI
|-pagefind-ui.js
|-pagefind-ui.css
# 模块化UI
|-pagefind-modular-ui.js
|-pagefind-modular-ui.css
# 其它索引项和Wasm程序文件
...
注意到里面有名为*-ui
的JS
和CSS
,它们是Pagefind
预置的搜索界面,简单几步就能嵌入站点:
<!-- 引入UI资源 -->
<link href="/pagefind/pagefind-ui.css" rel="stylesheet">
<script src="/pagefind/pagefind-ui.js"></script>
<div id="search"></div>
<script>
// 创建UI插入点
window.addEventListener('DOMContentLoaded', (event) => {
new PagefindUI({ element: "#search", showSubResults: true });
});
</script>
pagefind-ui
大概长这样,搜索结果包含下级标题和摘要,效果很不错。而且关键样式都可以通过CSS
全局变量自定义,比如背景色对应的是--pagefind-ui-background
。
至于pagefind-modular-ui
则是模块化的组件UI
,搜索框和搜索结果分开,可以自由插入到指定位置,配置更加灵活。详细信息参考这里。
一般情况下,这些文件应该和博客主站放在一起,但对于考虑国内访问的海外站点,也可把它们单独放到大陆的服务器或OSS
上,以优化访问速度。
搜索
如果对预置UI
不满意,可以使用Pagefind
提供的JS
函数接口手动调用搜索API
,自行处理搜索逻辑和结果展示。
// pagefind.js文件URL,我在部署时把它放到了国内OSS上,与主站分离
const pagefindUrl: string = isDebug() ? "/npm/pagefind/pagefind.js" : "https://apqx.***.com/blog/pagefind/pagefind.js"
// 动态加载
const pagefind = await import(pagefindUrl)
// 初始化
pagefind.init()
// 执行搜索,获取结果
const pagefindResult = await this.pagefind.search(key)
配合Material Design
做出的搜索Dialog
:
注意图中Tips
,Pagefind
为实现轻量化是对单词
建立索引,而它的中文分词
并不完善,一些人名之类的低频词并不能被正确识别,这个时候可以尝试在单字
之间加空格,它会搜索同时包含这些字
的结果,匹配准确度更好。
Webpack
如上所述调用搜索API
很简单,只需要import
对应的JS
文件而已,但是在Webpack
中,import
的JS
必须是本地文件且会被直接打包而非运行时动态从指定位置加载,导致Pagefind
搜索函数总是不能正确访问到云端索引…这个莫名其妙的问题困扰我很久,毕竟不是Web
开发者,明明按照文档一步步做,结果却一直卡住走不通…之后不断试错,Google
一长串Error
翻好多页才终于找到与之相关的一篇文章。
正确做法仅仅是添加一个webpackIgnore
注释,告诉Webpack
不要打包import
的这个pagefind.js
文件,留到运行时再加载。
const pagefind = await import(/*webpackIgnore: true*/ pagefindUrl)
就是这么简单🙄。
未完待续
对于博客,Pagefind
提供的基础搜索已经足够,此外它也支持定制索引范围、指定索引元数据、搜索过滤、搜索排序、搜索多个源等高阶功能,如果配置得当甚至可以替代Jekyll
的Tag
标签和Pagination
分页。
这也是我最近考虑要做的事,分页,用它还是Jekyll
的原生功能呢。