-
Notifications
You must be signed in to change notification settings - Fork 27
/
Copy pathsw-rules.js
263 lines (260 loc) · 9.11 KB
/
sw-rules.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
module.exports.config = {
/**
* 与 ServiceWorker 有关的配置项
* 若想禁止插件自动生成 sw,此项填 false 即可
* @type ?Object|boolean
*/
serviceWorker: {
/**
* 逃生门
* @type number
* @see https://kmar.top/posts/73014407/#6c7c33f0
*/
escape: 0,
/**
* 缓存库名称
* 发布网站后 **切勿修改** 该配置项!
* @type string
*/
cacheName: "AcrylicBlogCache",
/**
* 是否启用调试,启用后会在 sw 中插入一些辅助调试的代码,不建议开启
* @type boolean
*/
debug: false,
},
/**
* 与 SW 注册有关的配置项
* 若想禁止插件向 html 中插入注册代码,此项填 false 即可
* @type Object|boolean
*/
register: {
/**
* sw 注册成功时的动作
* @type ?VoidFunction
* */
onsuccess: undefined,
/**
* sw 注册失败时的动作
* ~若没有禁用 register,则该项为必填项,该项没有缺省值~
* @type ?VoidFunction
*/
onerror: undefined,
/**
* 生成注册 SW 的 HTML 代码片段
* @param root {string} 网页根目录的 URL
* @param hexoConfig {Object} Hexo 配置项
* @param pluginConfig {Object} SW 配置项
* @return {string} 一个 HTML 标签的字符串形式
*/
builder: (root, hexoConfig, pluginConfig) => {
const { onerror, onsuccess } = pluginConfig.register;
return `<script>
(() => {
const sw = navigator.serviceWorker
const error = ${onerror && onerror.toString()}
if (!sw?.register('${new URL(root).pathname}sw.js')
${onsuccess ? "?.then(" + onsuccess + ")" : ""}
?.catch(error)
) error()
})()
</script>`;
},
},
/**
* 与 DOM 端有关的配置
* 若想禁止插件自动生成 DOM 端 JS,此项填 false 即可
* @type {Object|boolean}
*/
dom: {
/**
* 缓存更新成功后触发的操作
* @type VoidFunction
*/
onsuccess: () => {
const isSnackbar = GLOBAL_CONFIG.Snackbar !== undefined;
isSnackbar && acy.snackbarShow("已刷新缓存,更新为最新内容");
},
},
/**
* 与插件生成的版本文件相关的配置项
* 该功能目前无法关闭
*/
json: {
/**
* 更新缓存时允许更新的最大 HTML 页面数量,需要更新的 HTML 文件数量超过这个值后会清除所有 HTML 缓存
* @type number
*/
maxHtml: 15,
/**
* 版本文件(update.json)字符数量限制,插件将保证版本文件的字符数量不超过该值
* @type number
*/
charLimit: 1024,
/**
* 文件缓存匹配采取精确模式
* 关闭时更新缓存时仅匹配文件名称,如 https://kmar.top/simple/a/index.html 仅匹配 /a/index.html
* 开启后更新缓存时将会匹配完整名称,如 https://kmar.top/simple/a/index.html 将匹配 /simple/a/index.html
* 两种方式各有优劣,开启后会增加 update.json 的空间占用,但会提升精确度
* 如果网站内没有多级目录结构,就可以放心大胆的关闭了
* key 值为文件拓展名,default 用于指代所有未列出的拓展名以及没有拓展名的文件
*/
precisionMode: {
default: false,
},
/**
* 是否合并指定项目
* 例如当 tags 为 true 时(假设标签目录为 https://kmar.top/tags/...)
* 如果标签页存在更新,则直接匹配 https://kmar.top/tags/ 目录下的所有文件
* **推荐将此项开启**
*/
merge: {
index: true,
tags: true,
archives: true,
categories: true,
/**
* 这里填写目录名称列表(不带两边的斜杠)
* @type string[]
*/
custom: [],
},
/**
* 生成版本文件时忽略的文件
* 注:匹配的时候不附带域名,只有 pathname,匹配的内容一定是博客本地的文件
* @type RegExp[]
*/
exclude: [/sw\.js$/],
},
/**
* 外部文件更新监听
* 该项缺省值为 false,若想允许插件监听外部文件的更新至少将值改为 `{}`
* @type Object|boolean
* @see https://kmar.top/posts/73014407/#c60b3060
*/
external: {
/**
* 拉取网络文件时的超时时间
* @type number
*/
timeout: 1500,
/**
* 匹配 JS 代码中的 URL
* 注意:字符串中的内容将被直接嵌入到正则表达式中,括号等特殊字符前需添加反斜杠,不允许使用括号
* 该项的缺省值为 `[]`,下方的值仅用于标明填写格式
* @see https://kmar.top/posts/73014407/#c60b3060
*/
js: [
{
head: "getScript(",
tail: ")",
},
],
/**
* 某些外链只要 URL 不变其内容就一定不会变
* 可以通过正则表达式排除这些外链的文件内容监控,加快构建速度
* 注意:当某一个文件被跳过拉取后,这个文件中包含的 URL 也会被跳过
* @type RegExp[]
*/
skip: [],
/**
* 在构建过程中替换部分链接,该替换结果不会影响文件内容
* 该设置项是为了应对构建服务器在国外,但是网站内部分缓存资源无法在国外访问导致拉取时超时的问题
* 该项的缺省值为 `[]`,下方的值仅用于标明填写格式
* @type Object[]
* @see https://kmar.top/posts/73014407/#4ea71e00
*/
replace: [
{
source: ["source0", "source1"],
dist: "dist",
},
],
},
/**
* 对 Hexo 中的变量进行排序
* 默认插件对 posts、tags、pages 三个变量进行排序
* 排序规则为优先按照字符串长度排序,若长度一致按照字典序排序
*
* 格式为 `name: value`
* value 的可能值为:字符串、非负整数、false
* 假定 Array<obj> 为要被排序的数据
* 当 value 为字符串和非负整数时,插件会以 `obj[value]` 的格式读取关键字
* 当 value 为 false 时,插件会直接以 `obj` 为关键字
* 注意:关键字必须为含有 length 属性且支持 < 操作符的类型
* 插件内置的 posts 规则如果用上面的格式写应该为:
* posts: 'title'
* 插件支持使用配置项覆盖插件内置规则
*
* 该项缺省值为 `{}`,下方的值仅用于标明填写格式
*/
sort: {
keywords: false,
},
};
/**
* 缓存列表
* @param clean 清理全站时是否删除其缓存
* @param match {function(URL)} 匹配规则
*/
module.exports.cacheList = {
// 这个 [simple] 就是规则的名称,该对象下可以包含多个规则,名称不影响缓存匹配
// 缓存匹配时按声明顺序进行匹配
simple: {
// [clean] 项用于声明符合该规则的缓存在进行全局清理时是否清除
// 如果你无法确定是否需要声明为 false 的话写 true 即可
clean: true,
// 该项用于匹配缓存,传入的参数是 URL 类型的,返回一个 boolean
match: (url, $eject) => {
const allowedHost = $eject.domain;
const allowedPaths = ["/404.html", "/css/index.css", "localhost"];
return url.host === allowedHost && allowedPaths.includes(url.pathname);
},
},
cdn: {
clean: false,
match: url =>
[
"cdn.cbd.int",
"lf26-cdn-tos.bytecdntp.com",
"lf6-cdn-tos.bytecdntp.com",
"lf3-cdn-tos.bytecdntp.com",
"lf9-cdn-tos.bytecdntp.com",
"cdn.staticfile.org",
"npm.elemecdn.com",
].includes(url.host) && url.pathname.match(/\.(js|css|woff2|woff|ttf|cur)$/),
},
};
/**
* 获取一个 URL 对应的备用 URL 列表,访问顺序按列表顺序,所有 URL 访问时参数一致
* @param srcUrl {string} 原始 URL
* @return {{list: string[], timeout: number}} 返回 null 或不返回表示对该 URL 不启用该功能。timeout 为超时时间(ms),list 为 URL 列表,列表不包含原始 URL 表示去除原始访问
*/
module.exports.getSpareUrls = srcUrl => {
if (srcUrl.startsWith('https://npm.elemecdn.com')) {
return {
timeout: 3000,
list: [
srcUrl,
`https://jsd.onmicrosoft.cn/${new URL(srcUrl).pathname}`
]
}
}
}
/**
* 获取要插入到 sw 中的变量或常量
* @param hexo hexo 对象
* @param rules 合并后的 sw-rules 对象
* @return {Object} 要插入的键值对
*/
module.exports.ejectValues = (hexo, rules) => {
return {
/** 这里的 key 为变量名,变量名仅允许包含英文字母和阿拉伯数字 */
domain: {
// 值的前缀,一般填 'const 或 let'
prefix: 'const',
// 要插入的值,只支持 string、number、boolean 和 bigint
value: new URL(hexo.config.url).host
}
}
}