schema-dsl 2.0.0 → 2.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +130 -113
- package/LICENSE +21 -21
- package/README.md +628 -628
- package/dist/{DslBuilder-DkLaOo9Q.d.ts → DslBuilder-BIgQOAXp.d.ts} +2 -0
- package/dist/{DslBuilder-DQDN0ZxZ.d.cts → DslBuilder-CjHTucNQ.d.cts} +2 -0
- package/dist/{Validator-hFWKGxir.d.ts → Validator-CllRdrY0.d.ts} +1 -1
- package/dist/{Validator-C7GsVQOH.d.cts → Validator-D6okG9tr.d.cts} +1 -1
- package/dist/index.cjs +75 -29
- package/dist/index.d.cts +10 -4
- package/dist/index.d.ts +10 -4
- package/dist/index.js +75 -29
- package/dist/plugins/custom-format.cjs +33 -17
- package/dist/plugins/custom-format.d.cts +1 -1
- package/dist/plugins/custom-format.d.ts +1 -1
- package/dist/plugins/custom-format.js +33 -17
- package/dist/plugins/custom-type-example.cjs +33 -17
- package/dist/plugins/custom-type-example.d.cts +1 -1
- package/dist/plugins/custom-type-example.d.ts +1 -1
- package/dist/plugins/custom-type-example.js +33 -17
- package/dist/plugins/custom-validator.cjs +0 -2
- package/dist/plugins/custom-validator.d.cts +1 -1
- package/dist/plugins/custom-validator.d.ts +1 -1
- package/dist/plugins/custom-validator.js +0 -2
- package/docs/FEATURE-INDEX.md +553 -553
- package/docs/add-custom-locale.md +496 -496
- package/docs/add-keyword.md +24 -24
- package/docs/api-reference.md +1047 -1047
- package/docs/api.md +13 -13
- package/docs/best-practices-project-structure.md +417 -417
- package/docs/best-practices.md +712 -712
- package/docs/cache-manager.md +344 -344
- package/docs/compile.md +45 -45
- package/docs/conditional-api.md +1307 -1307
- package/docs/custom-extensions-guide.md +339 -339
- package/docs/design-philosophy.md +606 -606
- package/docs/doc-index.md +324 -324
- package/docs/dsl-syntax.md +714 -714
- package/docs/dynamic-locale.md +608 -608
- package/docs/enum.md +482 -482
- package/docs/error-handling.md +1975 -1975
- package/docs/export-guide.md +501 -501
- package/docs/export-limitations.md +567 -567
- package/docs/faq.md +596 -596
- package/docs/frontend-i18n-guide.md +307 -307
- package/docs/i18n-user-guide.md +487 -487
- package/docs/i18n.md +476 -476
- package/docs/index.md +48 -48
- package/docs/json-schema-basics.md +40 -40
- package/docs/label-vs-description.md +271 -271
- package/docs/markdown-exporter.md +406 -406
- package/docs/mongodb-exporter.md +302 -302
- package/docs/multi-language.md +26 -26
- package/docs/multi-type-support.md +322 -322
- package/docs/mysql-exporter.md +280 -280
- package/docs/number-operators.md +449 -449
- package/docs/optional-marker-guide.md +326 -326
- package/docs/performance-guide.md +49 -49
- package/docs/plugin-system.md +381 -381
- package/docs/plugin-type-registration.md +34 -34
- package/docs/postgresql-exporter.md +311 -311
- package/docs/public/favicon.svg +4 -4
- package/docs/quick-start.md +435 -435
- package/docs/runtime-locale-support.md +532 -532
- package/docs/schema-helper.md +345 -345
- package/docs/schema-utils-advanced-issues.md +23 -23
- package/docs/schema-utils-best-practices.md +20 -20
- package/docs/schema-utils-chaining.md +150 -150
- package/docs/schema-utils.md +524 -524
- package/docs/security-checklist.md +20 -20
- package/docs/string-extensions.md +488 -488
- package/docs/troubleshooting.md +486 -486
- package/docs/type-converter.md +310 -310
- package/docs/type-reference.md +242 -242
- package/docs/typescript-guide.md +584 -584
- package/docs/union-type-guide.md +157 -157
- package/docs/union-types.md +284 -284
- package/docs/validate-async.md +491 -491
- package/docs/validate-batch.md +49 -49
- package/docs/validate-dsl-object-support.md +578 -578
- package/docs/validate.md +506 -506
- package/docs/validation-guide.md +502 -502
- package/docs/validator.md +39 -39
- package/package.json +131 -131
- package/plugins/custom-format.cjs +8 -8
- package/plugins/custom-type-example.cjs +8 -8
- package/plugins/custom-validator.cjs +8 -8
- package/src/adapters/DslAdapter.ts +111 -111
- package/src/adapters/index.ts +1 -1
- package/src/config/constants.ts +83 -83
- package/src/config/index.ts +2 -2
- package/src/config/patterns.ts +77 -77
- package/src/core/CacheManager.ts +169 -159
- package/src/core/ConditionalBuilder.ts +382 -382
- package/src/core/ConditionalRuntime.ts +27 -27
- package/src/core/ConditionalValidator.ts +254 -254
- package/src/core/DslBuilder.ts +687 -677
- package/src/core/ErrorCodes.ts +38 -38
- package/src/core/ErrorFormatter.ts +271 -271
- package/src/core/JSONSchemaCore.ts +65 -65
- package/src/core/Locale.ts +187 -187
- package/src/core/MessageTemplate.ts +42 -42
- package/src/core/ObjectDslBuilder.ts +64 -64
- package/src/core/PluginManager.ts +326 -326
- package/src/core/StringExtensions.ts +140 -140
- package/src/core/TemplateEngine.ts +44 -44
- package/src/core/Validator.ts +448 -448
- package/src/errors/I18nError.ts +159 -159
- package/src/errors/ValidationError.ts +105 -105
- package/src/exporters/BaseExporter.ts +60 -60
- package/src/exporters/MarkdownExporter.ts +305 -305
- package/src/exporters/MongoDBExporter.ts +126 -126
- package/src/exporters/MySQLExporter.ts +156 -155
- package/src/exporters/PostgreSQLExporter.ts +222 -222
- package/src/exporters/index.ts +18 -18
- package/src/index.ts +651 -633
- package/src/locales/en-US.ts +160 -160
- package/src/locales/es-ES.ts +160 -160
- package/src/locales/fr-FR.ts +160 -160
- package/src/locales/index.ts +103 -103
- package/src/locales/ja-JP.ts +160 -160
- package/src/locales/types.ts +156 -156
- package/src/locales/zh-CN.ts +160 -160
- package/src/parser/ConstraintParser.ts +101 -101
- package/src/parser/DslParser.ts +470 -470
- package/src/parser/SchemaCompiler.ts +66 -66
- package/src/parser/TypeRegistry.ts +250 -250
- package/src/parser/index.ts +6 -6
- package/src/plugins/custom-format.ts +124 -126
- package/src/plugins/custom-type-example.ts +106 -108
- package/src/plugins/custom-validator.ts +138 -140
- package/src/types/conditional.ts +28 -28
- package/src/types/config.ts +59 -59
- package/src/types/dsl.ts +131 -131
- package/src/types/error.ts +60 -60
- package/src/types/index.ts +17 -17
- package/src/types/infer.ts +127 -127
- package/src/types/plugin.ts +58 -58
- package/src/types/safe-regex.d.ts +9 -9
- package/src/types/schema.ts +66 -66
- package/src/types/validate.ts +71 -71
- package/src/utils/SchemaHelper.ts +196 -196
- package/src/utils/SchemaUtils.ts +365 -346
- package/src/utils/TypeConverter.ts +215 -215
- package/src/utils/index.ts +10 -10
- package/src/validators/CustomKeywords.ts +477 -477
package/docs/plugin-system.md
CHANGED
|
@@ -1,381 +1,381 @@
|
|
|
1
|
-
# 插件系统
|
|
2
|
-
|
|
3
|
-
> **更新**: 2026-05-01
|
|
4
|
-
> **状态**: ✅ 稳定
|
|
5
|
-
|
|
6
|
-
---
|
|
7
|
-
|
|
8
|
-
## 概述
|
|
9
|
-
|
|
10
|
-
`PluginManager` 是一个独立的插件管理器,负责:
|
|
11
|
-
|
|
12
|
-
- 注册 / 安装 / 卸载插件
|
|
13
|
-
- 管理插件钩子
|
|
14
|
-
- 提供 `EventEmitter` 兼容事件系统
|
|
15
|
-
- 通过 `context` 暴露插件注册表和钩子表
|
|
16
|
-
|
|
17
|
-
> **重要说明**
|
|
18
|
-
> `PluginManager` 本身不会自动接入 `dsl()`、`Validator`、各类 Exporter 的执行流程。
|
|
19
|
-
> 如果你希望在验证、编译或导出阶段运行某些 hook,需要由你的集成代码显式调用 `pluginManager.runHook(...)`。
|
|
20
|
-
|
|
21
|
-
---
|
|
22
|
-
|
|
23
|
-
## 快速开始
|
|
24
|
-
|
|
25
|
-
```javascript
|
|
26
|
-
const { PluginManager } = require('schema-dsl');
|
|
27
|
-
const schemaDsl = require('schema-dsl');
|
|
28
|
-
|
|
29
|
-
const pluginManager = new PluginManager();
|
|
30
|
-
|
|
31
|
-
const myPlugin = {
|
|
32
|
-
name: 'my-plugin',
|
|
33
|
-
version: '1.0.0',
|
|
34
|
-
description: '我的自定义插件',
|
|
35
|
-
|
|
36
|
-
install(core, options, context) {
|
|
37
|
-
console.log('installed:', !!core, options);
|
|
38
|
-
console.log('registered plugins:', context.plugins.size);
|
|
39
|
-
},
|
|
40
|
-
|
|
41
|
-
uninstall(core, context) {
|
|
42
|
-
console.log('cleanup:', !!core, context.hooks.size);
|
|
43
|
-
}
|
|
44
|
-
};
|
|
45
|
-
|
|
46
|
-
pluginManager.register(myPlugin);
|
|
47
|
-
pluginManager.install(schemaDsl, 'my-plugin', { enabled: true });
|
|
48
|
-
pluginManager.uninstall('my-plugin', schemaDsl);
|
|
49
|
-
```
|
|
50
|
-
|
|
51
|
-
---
|
|
52
|
-
|
|
53
|
-
## 插件对象结构
|
|
54
|
-
|
|
55
|
-
```javascript
|
|
56
|
-
module.exports = {
|
|
57
|
-
// 必填
|
|
58
|
-
name: 'plugin-name',
|
|
59
|
-
install(core, options, context) {
|
|
60
|
-
// 安装逻辑
|
|
61
|
-
},
|
|
62
|
-
|
|
63
|
-
// 可选
|
|
64
|
-
version: '1.0.0',
|
|
65
|
-
description: '插件描述',
|
|
66
|
-
uninstall(core, context) {
|
|
67
|
-
// 卸载逻辑
|
|
68
|
-
},
|
|
69
|
-
hooks: {
|
|
70
|
-
onBeforeValidate(schema, data) {},
|
|
71
|
-
onAfterValidate(result) {}
|
|
72
|
-
},
|
|
73
|
-
options: {
|
|
74
|
-
enabled: true
|
|
75
|
-
}
|
|
76
|
-
};
|
|
77
|
-
```
|
|
78
|
-
|
|
79
|
-
### 参数说明
|
|
80
|
-
|
|
81
|
-
| 参数 | 说明 |
|
|
82
|
-
|---|---|
|
|
83
|
-
| `core` | 传给 `install()` / `uninstall()` 的核心对象,通常是 `require('schema-dsl')` 的结果或你自己的集成对象 |
|
|
84
|
-
| `options` | 安装时的合并配置:`{ ...plugin.options, ...installOptions }` |
|
|
85
|
-
| `context.plugins` | 当前已注册插件的 `Map<string, Plugin>` |
|
|
86
|
-
| `context.hooks` | 当前 hook 注册表的 `Map<string, Function[]>` |
|
|
87
|
-
|
|
88
|
-
---
|
|
89
|
-
|
|
90
|
-
## 钩子系统
|
|
91
|
-
|
|
92
|
-
### 1. 自动触发的内置生命周期
|
|
93
|
-
|
|
94
|
-
这些 hook 由 `PluginManager` 自动触发:
|
|
95
|
-
|
|
96
|
-
| 名称 | 触发时机 | 参数 |
|
|
97
|
-
|---|---|---|
|
|
98
|
-
| `onBeforeRegister` | `register(plugin)` 写入注册表前 | `(plugin)` |
|
|
99
|
-
| `onAfterRegister` | `register(plugin)` 完成后 | `(plugin)` |
|
|
100
|
-
| `onError` | 某个 hook 执行抛错后 | `(error, meta)` |
|
|
101
|
-
|
|
102
|
-
### 2. 约定式 hook 名称
|
|
103
|
-
|
|
104
|
-
以下名称是常用约定,`PluginManager` 支持注册它们,但**是否执行取决于你的代码是否调用 `runHook()`**:
|
|
105
|
-
|
|
106
|
-
| 名称 | 常见用途 |
|
|
107
|
-
|---|---|
|
|
108
|
-
| `onBeforeValidate` / `onAfterValidate` | 验证前后 |
|
|
109
|
-
| `onBeforeCompile` / `onAfterCompile` | 编译前后 |
|
|
110
|
-
| `onBeforeExport` / `onAfterExport` | 导出前后 |
|
|
111
|
-
| `beforeParse` / `afterParse` | 解析阶段 |
|
|
112
|
-
| `beforeValidate` / `afterValidate` | v2 风格命名 |
|
|
113
|
-
| `beforeCompile` / `afterCompile` | v2 风格命名 |
|
|
114
|
-
|
|
115
|
-
> `hook()` / `runHook()` 支持任意字符串名称,不限于上表。
|
|
116
|
-
|
|
117
|
-
### 3. 注册与运行 hook
|
|
118
|
-
|
|
119
|
-
```javascript
|
|
120
|
-
pluginManager.hook('onBeforeValidate', (schema, data) => {
|
|
121
|
-
console.log('验证前:', schema, data);
|
|
122
|
-
});
|
|
123
|
-
|
|
124
|
-
const results = await pluginManager.runHook('onBeforeValidate', schema, data);
|
|
125
|
-
```
|
|
126
|
-
|
|
127
|
-
### 4. 在插件中声明 hook
|
|
128
|
-
|
|
129
|
-
```javascript
|
|
130
|
-
const loggingPlugin = {
|
|
131
|
-
name: 'logging',
|
|
132
|
-
install() {},
|
|
133
|
-
hooks: {
|
|
134
|
-
onBeforeValidate(schema, data) {
|
|
135
|
-
console.log('before validate', schema, data);
|
|
136
|
-
},
|
|
137
|
-
onAfterValidate(result) {
|
|
138
|
-
console.log('after validate', result);
|
|
139
|
-
}
|
|
140
|
-
}
|
|
141
|
-
};
|
|
142
|
-
|
|
143
|
-
pluginManager.register(loggingPlugin);
|
|
144
|
-
await pluginManager.runHook('onBeforeValidate', schema, data);
|
|
145
|
-
```
|
|
146
|
-
|
|
147
|
-
---
|
|
148
|
-
|
|
149
|
-
## 事件系统
|
|
150
|
-
|
|
151
|
-
`PluginManager` 继承自 `EventEmitter`,因此可以使用:
|
|
152
|
-
|
|
153
|
-
- `on()`
|
|
154
|
-
- `once()`
|
|
155
|
-
- `off()`
|
|
156
|
-
- `emit()`
|
|
157
|
-
- `removeListener()`
|
|
158
|
-
- `removeAllListeners()`
|
|
159
|
-
|
|
160
|
-
### 可用事件
|
|
161
|
-
|
|
162
|
-
| 事件名 | 触发时机 | 参数 |
|
|
163
|
-
|---|---|---|
|
|
164
|
-
| `plugin:registered` | 插件注册成功 | `(plugin)` |
|
|
165
|
-
| `plugin:installed` | 插件安装成功 | `(plugin)` |
|
|
166
|
-
| `plugin:uninstalled` | 插件卸载成功 | `(plugin)` |
|
|
167
|
-
| `plugin:error` | 插件安装 / 卸载失败 | `({ plugin, error })` |
|
|
168
|
-
| `hook:error` | hook 执行失败 | `({ hookName, handler, error })` |
|
|
169
|
-
| `plugins:cleared` | `clear()` 完成后 | `()` |
|
|
170
|
-
|
|
171
|
-
### 示例
|
|
172
|
-
|
|
173
|
-
```javascript
|
|
174
|
-
pluginManager.on('plugin:registered', (plugin) => {
|
|
175
|
-
console.log('插件已注册:', plugin.name);
|
|
176
|
-
});
|
|
177
|
-
|
|
178
|
-
pluginManager.on('plugin:installed', (plugin) => {
|
|
179
|
-
console.log('插件已安装:', plugin.name);
|
|
180
|
-
});
|
|
181
|
-
|
|
182
|
-
pluginManager.on('plugin:error', ({ plugin, error }) => {
|
|
183
|
-
console.error('插件错误:', plugin.name, error.message);
|
|
184
|
-
});
|
|
185
|
-
|
|
186
|
-
pluginManager.on('hook:error', ({ hookName, error }) => {
|
|
187
|
-
console.error('Hook 错误:', hookName, error.message);
|
|
188
|
-
});
|
|
189
|
-
```
|
|
190
|
-
|
|
191
|
-
---
|
|
192
|
-
|
|
193
|
-
## API 参考
|
|
194
|
-
|
|
195
|
-
### `register(plugin)`
|
|
196
|
-
|
|
197
|
-
注册插件,并自动触发:
|
|
198
|
-
|
|
199
|
-
1. `onBeforeRegister`
|
|
200
|
-
2. 写入注册表 / 注册插件自带 hooks
|
|
201
|
-
3. `onAfterRegister`
|
|
202
|
-
4. `plugin:registered`
|
|
203
|
-
|
|
204
|
-
### `install(core, [pluginName], [options])`
|
|
205
|
-
|
|
206
|
-
安装插件。
|
|
207
|
-
|
|
208
|
-
- `install(core)`:安装所有已注册插件
|
|
209
|
-
- `install(core, 'name', options)`:安装指定插件
|
|
210
|
-
- `install()` 时会把第三个参数 `context` 传给插件
|
|
211
|
-
- 安装成功后触发 `plugin:installed`
|
|
212
|
-
- 安装失败时触发 `plugin:error`,然后抛错
|
|
213
|
-
|
|
214
|
-
### `unregister(name, [core])`
|
|
215
|
-
|
|
216
|
-
卸载插件并移除该插件注册的 hooks。
|
|
217
|
-
|
|
218
|
-
- `plugin.uninstall(core, context)` 成功后才会真正移除插件
|
|
219
|
-
- 卸载成功后触发 `plugin:uninstalled`
|
|
220
|
-
- 卸载失败时触发 `plugin:error`,然后抛错
|
|
221
|
-
|
|
222
|
-
### `uninstall(name, [core])`
|
|
223
|
-
|
|
224
|
-
`unregister()` 的别名,兼容 v1。
|
|
225
|
-
|
|
226
|
-
### `hook(name, handler)`
|
|
227
|
-
|
|
228
|
-
注册一个 hook 处理器。
|
|
229
|
-
|
|
230
|
-
### `unhook(name, handler)`
|
|
231
|
-
|
|
232
|
-
移除指定 hook 处理器。
|
|
233
|
-
|
|
234
|
-
### `runHook(name, ...args)`
|
|
235
|
-
|
|
236
|
-
异步运行某个 hook 下的全部处理器,返回结果数组。
|
|
237
|
-
|
|
238
|
-
- 单个 handler 抛错不会中断后续 handler
|
|
239
|
-
- 抛错时会触发 `hook:error`
|
|
240
|
-
- 同时会执行 `onError`
|
|
241
|
-
|
|
242
|
-
### `has(name)`
|
|
243
|
-
|
|
244
|
-
检查插件是否已注册。
|
|
245
|
-
|
|
246
|
-
### `get([name])`
|
|
247
|
-
|
|
248
|
-
- `get(name)`:获取单个插件
|
|
249
|
-
- `get()`:获取全部插件 `Map`
|
|
250
|
-
|
|
251
|
-
### `list()`
|
|
252
|
-
|
|
253
|
-
返回插件元数据数组:
|
|
254
|
-
|
|
255
|
-
```javascript
|
|
256
|
-
[{ name, version, description }]
|
|
257
|
-
```
|
|
258
|
-
|
|
259
|
-
### `clear([core])`
|
|
260
|
-
|
|
261
|
-
逐个卸载所有插件,清空注册表和 hooks,最后触发 `plugins:cleared`。
|
|
262
|
-
|
|
263
|
-
### 属性
|
|
264
|
-
|
|
265
|
-
| 属性 | 说明 |
|
|
266
|
-
|---|---|
|
|
267
|
-
| `pluginManager.plugins` | 插件注册表 `Map<string, Plugin>` |
|
|
268
|
-
| `pluginManager.hooks` | hook 注册表 `Map<string, Function[]>` |
|
|
269
|
-
| `pluginManager.context` | `{ plugins, hooks }` |
|
|
270
|
-
| `pluginManager.size` | 已注册插件数量 |
|
|
271
|
-
| `pluginManager.pluginCount` | 已注册插件数量(别名) |
|
|
272
|
-
|
|
273
|
-
---
|
|
274
|
-
|
|
275
|
-
## 最佳实践
|
|
276
|
-
|
|
277
|
-
### 1. 命名
|
|
278
|
-
|
|
279
|
-
使用 `kebab-case`:
|
|
280
|
-
|
|
281
|
-
```javascript
|
|
282
|
-
name: 'custom-validator'
|
|
283
|
-
name: 'mongodb-plugin'
|
|
284
|
-
```
|
|
285
|
-
|
|
286
|
-
### 2. 错误处理
|
|
287
|
-
|
|
288
|
-
```javascript
|
|
289
|
-
install(core, options, context) {
|
|
290
|
-
try {
|
|
291
|
-
// 安装逻辑
|
|
292
|
-
} catch (error) {
|
|
293
|
-
throw error;
|
|
294
|
-
}
|
|
295
|
-
}
|
|
296
|
-
```
|
|
297
|
-
|
|
298
|
-
### 3. 资源清理
|
|
299
|
-
|
|
300
|
-
```javascript
|
|
301
|
-
uninstall(core, context) {
|
|
302
|
-
// 清理注册的资源
|
|
303
|
-
}
|
|
304
|
-
```
|
|
305
|
-
|
|
306
|
-
### 4. 插件间通信
|
|
307
|
-
|
|
308
|
-
```javascript
|
|
309
|
-
install(core, options, context) {
|
|
310
|
-
if (!context.plugins.has('dependency-plugin')) {
|
|
311
|
-
throw new Error('需要先安装 dependency-plugin');
|
|
312
|
-
}
|
|
313
|
-
}
|
|
314
|
-
```
|
|
315
|
-
|
|
316
|
-
---
|
|
317
|
-
|
|
318
|
-
## 故障排查
|
|
319
|
-
|
|
320
|
-
### 插件未生效
|
|
321
|
-
|
|
322
|
-
```javascript
|
|
323
|
-
pluginManager.has('my-plugin');
|
|
324
|
-
pluginManager.list();
|
|
325
|
-
```
|
|
326
|
-
|
|
327
|
-
确认插件已注册、已安装,并且 `install()` 确实执行到了你的逻辑。
|
|
328
|
-
|
|
329
|
-
### hook 未触发
|
|
330
|
-
|
|
331
|
-
检查两件事:
|
|
332
|
-
|
|
333
|
-
1. hook 名称是否一致
|
|
334
|
-
2. 你的代码是否真的调用了 `pluginManager.runHook('hookName', ...)`
|
|
335
|
-
|
|
336
|
-
### 当前仓库的发布方式
|
|
337
|
-
|
|
338
|
-
当前仓库已恢复 v1 风格的官方插件子路径入口,可直接使用:
|
|
339
|
-
|
|
340
|
-
- `schema-dsl/plugins/custom-format`
|
|
341
|
-
- `schema-dsl/plugins/custom-validator`
|
|
342
|
-
- `schema-dsl/plugins/custom-type-example`
|
|
343
|
-
|
|
344
|
-
```javascript
|
|
345
|
-
const { PluginManager } = require('schema-dsl');
|
|
346
|
-
const schemaDsl = require('schema-dsl');
|
|
347
|
-
const customFormat = require('schema-dsl/plugins/custom-format');
|
|
348
|
-
|
|
349
|
-
const pluginManager = new PluginManager();
|
|
350
|
-
pluginManager.register(customFormat);
|
|
351
|
-
pluginManager.install(schemaDsl, 'custom-format');
|
|
352
|
-
```
|
|
353
|
-
|
|
354
|
-
```typescript
|
|
355
|
-
import { PluginManager } from 'schema-dsl';
|
|
356
|
-
import * as schemaDsl from 'schema-dsl';
|
|
357
|
-
import customTypeExample from 'schema-dsl/plugins/custom-type-example';
|
|
358
|
-
|
|
359
|
-
const pluginManager = new PluginManager();
|
|
360
|
-
pluginManager.register(customTypeExample);
|
|
361
|
-
pluginManager.install(schemaDsl, 'custom-type-example');
|
|
362
|
-
```
|
|
363
|
-
|
|
364
|
-
> ⚠️ 注意:官方插件子路径只补齐了 v1 已存在的三个示例插件入口;
|
|
365
|
-
> `PluginManager` 仍然不会自动接入 `validate()` / `compile()` / exporter 流程,hook 是否执行仍取决于你的集成代码是否显式调用 `runHook()`。
|
|
366
|
-
|
|
367
|
-
---
|
|
368
|
-
|
|
369
|
-
## 相关文档
|
|
370
|
-
|
|
371
|
-
- [API 参考](api-reference.md)
|
|
372
|
-
- [验证指南](validate.md)
|
|
373
|
-
- [DSL 语法](dsl-syntax.md)
|
|
374
|
-
|
|
375
|
-
---
|
|
376
|
-
|
|
377
|
-
## 对应示例文件
|
|
378
|
-
|
|
379
|
-
**示例入口**: [plugin-system.ts](https://github.com/vextjs/schema-dsl/blob/main/examples/docs/plugin-system.ts)
|
|
380
|
-
**说明**: 覆盖自定义插件的注册 / 安装 / 卸载、`runHook()` 执行结果,以及官方 `custom-format` 子路径插件的安装效果。
|
|
381
|
-
|
|
1
|
+
# 插件系统
|
|
2
|
+
|
|
3
|
+
> **更新**: 2026-05-01
|
|
4
|
+
> **状态**: ✅ 稳定
|
|
5
|
+
|
|
6
|
+
---
|
|
7
|
+
|
|
8
|
+
## 概述
|
|
9
|
+
|
|
10
|
+
`PluginManager` 是一个独立的插件管理器,负责:
|
|
11
|
+
|
|
12
|
+
- 注册 / 安装 / 卸载插件
|
|
13
|
+
- 管理插件钩子
|
|
14
|
+
- 提供 `EventEmitter` 兼容事件系统
|
|
15
|
+
- 通过 `context` 暴露插件注册表和钩子表
|
|
16
|
+
|
|
17
|
+
> **重要说明**
|
|
18
|
+
> `PluginManager` 本身不会自动接入 `dsl()`、`Validator`、各类 Exporter 的执行流程。
|
|
19
|
+
> 如果你希望在验证、编译或导出阶段运行某些 hook,需要由你的集成代码显式调用 `pluginManager.runHook(...)`。
|
|
20
|
+
|
|
21
|
+
---
|
|
22
|
+
|
|
23
|
+
## 快速开始
|
|
24
|
+
|
|
25
|
+
```javascript
|
|
26
|
+
const { PluginManager } = require('schema-dsl');
|
|
27
|
+
const schemaDsl = require('schema-dsl');
|
|
28
|
+
|
|
29
|
+
const pluginManager = new PluginManager();
|
|
30
|
+
|
|
31
|
+
const myPlugin = {
|
|
32
|
+
name: 'my-plugin',
|
|
33
|
+
version: '1.0.0',
|
|
34
|
+
description: '我的自定义插件',
|
|
35
|
+
|
|
36
|
+
install(core, options, context) {
|
|
37
|
+
console.log('installed:', !!core, options);
|
|
38
|
+
console.log('registered plugins:', context.plugins.size);
|
|
39
|
+
},
|
|
40
|
+
|
|
41
|
+
uninstall(core, context) {
|
|
42
|
+
console.log('cleanup:', !!core, context.hooks.size);
|
|
43
|
+
}
|
|
44
|
+
};
|
|
45
|
+
|
|
46
|
+
pluginManager.register(myPlugin);
|
|
47
|
+
pluginManager.install(schemaDsl, 'my-plugin', { enabled: true });
|
|
48
|
+
pluginManager.uninstall('my-plugin', schemaDsl);
|
|
49
|
+
```
|
|
50
|
+
|
|
51
|
+
---
|
|
52
|
+
|
|
53
|
+
## 插件对象结构
|
|
54
|
+
|
|
55
|
+
```javascript
|
|
56
|
+
module.exports = {
|
|
57
|
+
// 必填
|
|
58
|
+
name: 'plugin-name',
|
|
59
|
+
install(core, options, context) {
|
|
60
|
+
// 安装逻辑
|
|
61
|
+
},
|
|
62
|
+
|
|
63
|
+
// 可选
|
|
64
|
+
version: '1.0.0',
|
|
65
|
+
description: '插件描述',
|
|
66
|
+
uninstall(core, context) {
|
|
67
|
+
// 卸载逻辑
|
|
68
|
+
},
|
|
69
|
+
hooks: {
|
|
70
|
+
onBeforeValidate(schema, data) {},
|
|
71
|
+
onAfterValidate(result) {}
|
|
72
|
+
},
|
|
73
|
+
options: {
|
|
74
|
+
enabled: true
|
|
75
|
+
}
|
|
76
|
+
};
|
|
77
|
+
```
|
|
78
|
+
|
|
79
|
+
### 参数说明
|
|
80
|
+
|
|
81
|
+
| 参数 | 说明 |
|
|
82
|
+
|---|---|
|
|
83
|
+
| `core` | 传给 `install()` / `uninstall()` 的核心对象,通常是 `require('schema-dsl')` 的结果或你自己的集成对象 |
|
|
84
|
+
| `options` | 安装时的合并配置:`{ ...plugin.options, ...installOptions }` |
|
|
85
|
+
| `context.plugins` | 当前已注册插件的 `Map<string, Plugin>` |
|
|
86
|
+
| `context.hooks` | 当前 hook 注册表的 `Map<string, Function[]>` |
|
|
87
|
+
|
|
88
|
+
---
|
|
89
|
+
|
|
90
|
+
## 钩子系统
|
|
91
|
+
|
|
92
|
+
### 1. 自动触发的内置生命周期
|
|
93
|
+
|
|
94
|
+
这些 hook 由 `PluginManager` 自动触发:
|
|
95
|
+
|
|
96
|
+
| 名称 | 触发时机 | 参数 |
|
|
97
|
+
|---|---|---|
|
|
98
|
+
| `onBeforeRegister` | `register(plugin)` 写入注册表前 | `(plugin)` |
|
|
99
|
+
| `onAfterRegister` | `register(plugin)` 完成后 | `(plugin)` |
|
|
100
|
+
| `onError` | 某个 hook 执行抛错后 | `(error, meta)` |
|
|
101
|
+
|
|
102
|
+
### 2. 约定式 hook 名称
|
|
103
|
+
|
|
104
|
+
以下名称是常用约定,`PluginManager` 支持注册它们,但**是否执行取决于你的代码是否调用 `runHook()`**:
|
|
105
|
+
|
|
106
|
+
| 名称 | 常见用途 |
|
|
107
|
+
|---|---|
|
|
108
|
+
| `onBeforeValidate` / `onAfterValidate` | 验证前后 |
|
|
109
|
+
| `onBeforeCompile` / `onAfterCompile` | 编译前后 |
|
|
110
|
+
| `onBeforeExport` / `onAfterExport` | 导出前后 |
|
|
111
|
+
| `beforeParse` / `afterParse` | 解析阶段 |
|
|
112
|
+
| `beforeValidate` / `afterValidate` | v2 风格命名 |
|
|
113
|
+
| `beforeCompile` / `afterCompile` | v2 风格命名 |
|
|
114
|
+
|
|
115
|
+
> `hook()` / `runHook()` 支持任意字符串名称,不限于上表。
|
|
116
|
+
|
|
117
|
+
### 3. 注册与运行 hook
|
|
118
|
+
|
|
119
|
+
```javascript
|
|
120
|
+
pluginManager.hook('onBeforeValidate', (schema, data) => {
|
|
121
|
+
console.log('验证前:', schema, data);
|
|
122
|
+
});
|
|
123
|
+
|
|
124
|
+
const results = await pluginManager.runHook('onBeforeValidate', schema, data);
|
|
125
|
+
```
|
|
126
|
+
|
|
127
|
+
### 4. 在插件中声明 hook
|
|
128
|
+
|
|
129
|
+
```javascript
|
|
130
|
+
const loggingPlugin = {
|
|
131
|
+
name: 'logging',
|
|
132
|
+
install() {},
|
|
133
|
+
hooks: {
|
|
134
|
+
onBeforeValidate(schema, data) {
|
|
135
|
+
console.log('before validate', schema, data);
|
|
136
|
+
},
|
|
137
|
+
onAfterValidate(result) {
|
|
138
|
+
console.log('after validate', result);
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
};
|
|
142
|
+
|
|
143
|
+
pluginManager.register(loggingPlugin);
|
|
144
|
+
await pluginManager.runHook('onBeforeValidate', schema, data);
|
|
145
|
+
```
|
|
146
|
+
|
|
147
|
+
---
|
|
148
|
+
|
|
149
|
+
## 事件系统
|
|
150
|
+
|
|
151
|
+
`PluginManager` 继承自 `EventEmitter`,因此可以使用:
|
|
152
|
+
|
|
153
|
+
- `on()`
|
|
154
|
+
- `once()`
|
|
155
|
+
- `off()`
|
|
156
|
+
- `emit()`
|
|
157
|
+
- `removeListener()`
|
|
158
|
+
- `removeAllListeners()`
|
|
159
|
+
|
|
160
|
+
### 可用事件
|
|
161
|
+
|
|
162
|
+
| 事件名 | 触发时机 | 参数 |
|
|
163
|
+
|---|---|---|
|
|
164
|
+
| `plugin:registered` | 插件注册成功 | `(plugin)` |
|
|
165
|
+
| `plugin:installed` | 插件安装成功 | `(plugin)` |
|
|
166
|
+
| `plugin:uninstalled` | 插件卸载成功 | `(plugin)` |
|
|
167
|
+
| `plugin:error` | 插件安装 / 卸载失败 | `({ plugin, error })` |
|
|
168
|
+
| `hook:error` | hook 执行失败 | `({ hookName, handler, error })` |
|
|
169
|
+
| `plugins:cleared` | `clear()` 完成后 | `()` |
|
|
170
|
+
|
|
171
|
+
### 示例
|
|
172
|
+
|
|
173
|
+
```javascript
|
|
174
|
+
pluginManager.on('plugin:registered', (plugin) => {
|
|
175
|
+
console.log('插件已注册:', plugin.name);
|
|
176
|
+
});
|
|
177
|
+
|
|
178
|
+
pluginManager.on('plugin:installed', (plugin) => {
|
|
179
|
+
console.log('插件已安装:', plugin.name);
|
|
180
|
+
});
|
|
181
|
+
|
|
182
|
+
pluginManager.on('plugin:error', ({ plugin, error }) => {
|
|
183
|
+
console.error('插件错误:', plugin.name, error.message);
|
|
184
|
+
});
|
|
185
|
+
|
|
186
|
+
pluginManager.on('hook:error', ({ hookName, error }) => {
|
|
187
|
+
console.error('Hook 错误:', hookName, error.message);
|
|
188
|
+
});
|
|
189
|
+
```
|
|
190
|
+
|
|
191
|
+
---
|
|
192
|
+
|
|
193
|
+
## API 参考
|
|
194
|
+
|
|
195
|
+
### `register(plugin)`
|
|
196
|
+
|
|
197
|
+
注册插件,并自动触发:
|
|
198
|
+
|
|
199
|
+
1. `onBeforeRegister`
|
|
200
|
+
2. 写入注册表 / 注册插件自带 hooks
|
|
201
|
+
3. `onAfterRegister`
|
|
202
|
+
4. `plugin:registered`
|
|
203
|
+
|
|
204
|
+
### `install(core, [pluginName], [options])`
|
|
205
|
+
|
|
206
|
+
安装插件。
|
|
207
|
+
|
|
208
|
+
- `install(core)`:安装所有已注册插件
|
|
209
|
+
- `install(core, 'name', options)`:安装指定插件
|
|
210
|
+
- `install()` 时会把第三个参数 `context` 传给插件
|
|
211
|
+
- 安装成功后触发 `plugin:installed`
|
|
212
|
+
- 安装失败时触发 `plugin:error`,然后抛错
|
|
213
|
+
|
|
214
|
+
### `unregister(name, [core])`
|
|
215
|
+
|
|
216
|
+
卸载插件并移除该插件注册的 hooks。
|
|
217
|
+
|
|
218
|
+
- `plugin.uninstall(core, context)` 成功后才会真正移除插件
|
|
219
|
+
- 卸载成功后触发 `plugin:uninstalled`
|
|
220
|
+
- 卸载失败时触发 `plugin:error`,然后抛错
|
|
221
|
+
|
|
222
|
+
### `uninstall(name, [core])`
|
|
223
|
+
|
|
224
|
+
`unregister()` 的别名,兼容 v1。
|
|
225
|
+
|
|
226
|
+
### `hook(name, handler)`
|
|
227
|
+
|
|
228
|
+
注册一个 hook 处理器。
|
|
229
|
+
|
|
230
|
+
### `unhook(name, handler)`
|
|
231
|
+
|
|
232
|
+
移除指定 hook 处理器。
|
|
233
|
+
|
|
234
|
+
### `runHook(name, ...args)`
|
|
235
|
+
|
|
236
|
+
异步运行某个 hook 下的全部处理器,返回结果数组。
|
|
237
|
+
|
|
238
|
+
- 单个 handler 抛错不会中断后续 handler
|
|
239
|
+
- 抛错时会触发 `hook:error`
|
|
240
|
+
- 同时会执行 `onError`
|
|
241
|
+
|
|
242
|
+
### `has(name)`
|
|
243
|
+
|
|
244
|
+
检查插件是否已注册。
|
|
245
|
+
|
|
246
|
+
### `get([name])`
|
|
247
|
+
|
|
248
|
+
- `get(name)`:获取单个插件
|
|
249
|
+
- `get()`:获取全部插件 `Map`
|
|
250
|
+
|
|
251
|
+
### `list()`
|
|
252
|
+
|
|
253
|
+
返回插件元数据数组:
|
|
254
|
+
|
|
255
|
+
```javascript
|
|
256
|
+
[{ name, version, description }]
|
|
257
|
+
```
|
|
258
|
+
|
|
259
|
+
### `clear([core])`
|
|
260
|
+
|
|
261
|
+
逐个卸载所有插件,清空注册表和 hooks,最后触发 `plugins:cleared`。
|
|
262
|
+
|
|
263
|
+
### 属性
|
|
264
|
+
|
|
265
|
+
| 属性 | 说明 |
|
|
266
|
+
|---|---|
|
|
267
|
+
| `pluginManager.plugins` | 插件注册表 `Map<string, Plugin>` |
|
|
268
|
+
| `pluginManager.hooks` | hook 注册表 `Map<string, Function[]>` |
|
|
269
|
+
| `pluginManager.context` | `{ plugins, hooks }` |
|
|
270
|
+
| `pluginManager.size` | 已注册插件数量 |
|
|
271
|
+
| `pluginManager.pluginCount` | 已注册插件数量(别名) |
|
|
272
|
+
|
|
273
|
+
---
|
|
274
|
+
|
|
275
|
+
## 最佳实践
|
|
276
|
+
|
|
277
|
+
### 1. 命名
|
|
278
|
+
|
|
279
|
+
使用 `kebab-case`:
|
|
280
|
+
|
|
281
|
+
```javascript
|
|
282
|
+
name: 'custom-validator'
|
|
283
|
+
name: 'mongodb-plugin'
|
|
284
|
+
```
|
|
285
|
+
|
|
286
|
+
### 2. 错误处理
|
|
287
|
+
|
|
288
|
+
```javascript
|
|
289
|
+
install(core, options, context) {
|
|
290
|
+
try {
|
|
291
|
+
// 安装逻辑
|
|
292
|
+
} catch (error) {
|
|
293
|
+
throw error;
|
|
294
|
+
}
|
|
295
|
+
}
|
|
296
|
+
```
|
|
297
|
+
|
|
298
|
+
### 3. 资源清理
|
|
299
|
+
|
|
300
|
+
```javascript
|
|
301
|
+
uninstall(core, context) {
|
|
302
|
+
// 清理注册的资源
|
|
303
|
+
}
|
|
304
|
+
```
|
|
305
|
+
|
|
306
|
+
### 4. 插件间通信
|
|
307
|
+
|
|
308
|
+
```javascript
|
|
309
|
+
install(core, options, context) {
|
|
310
|
+
if (!context.plugins.has('dependency-plugin')) {
|
|
311
|
+
throw new Error('需要先安装 dependency-plugin');
|
|
312
|
+
}
|
|
313
|
+
}
|
|
314
|
+
```
|
|
315
|
+
|
|
316
|
+
---
|
|
317
|
+
|
|
318
|
+
## 故障排查
|
|
319
|
+
|
|
320
|
+
### 插件未生效
|
|
321
|
+
|
|
322
|
+
```javascript
|
|
323
|
+
pluginManager.has('my-plugin');
|
|
324
|
+
pluginManager.list();
|
|
325
|
+
```
|
|
326
|
+
|
|
327
|
+
确认插件已注册、已安装,并且 `install()` 确实执行到了你的逻辑。
|
|
328
|
+
|
|
329
|
+
### hook 未触发
|
|
330
|
+
|
|
331
|
+
检查两件事:
|
|
332
|
+
|
|
333
|
+
1. hook 名称是否一致
|
|
334
|
+
2. 你的代码是否真的调用了 `pluginManager.runHook('hookName', ...)`
|
|
335
|
+
|
|
336
|
+
### 当前仓库的发布方式
|
|
337
|
+
|
|
338
|
+
当前仓库已恢复 v1 风格的官方插件子路径入口,可直接使用:
|
|
339
|
+
|
|
340
|
+
- `schema-dsl/plugins/custom-format`
|
|
341
|
+
- `schema-dsl/plugins/custom-validator`
|
|
342
|
+
- `schema-dsl/plugins/custom-type-example`
|
|
343
|
+
|
|
344
|
+
```javascript
|
|
345
|
+
const { PluginManager } = require('schema-dsl');
|
|
346
|
+
const schemaDsl = require('schema-dsl');
|
|
347
|
+
const customFormat = require('schema-dsl/plugins/custom-format');
|
|
348
|
+
|
|
349
|
+
const pluginManager = new PluginManager();
|
|
350
|
+
pluginManager.register(customFormat);
|
|
351
|
+
pluginManager.install(schemaDsl, 'custom-format');
|
|
352
|
+
```
|
|
353
|
+
|
|
354
|
+
```typescript
|
|
355
|
+
import { PluginManager } from 'schema-dsl';
|
|
356
|
+
import * as schemaDsl from 'schema-dsl';
|
|
357
|
+
import customTypeExample from 'schema-dsl/plugins/custom-type-example';
|
|
358
|
+
|
|
359
|
+
const pluginManager = new PluginManager();
|
|
360
|
+
pluginManager.register(customTypeExample);
|
|
361
|
+
pluginManager.install(schemaDsl, 'custom-type-example');
|
|
362
|
+
```
|
|
363
|
+
|
|
364
|
+
> ⚠️ 注意:官方插件子路径只补齐了 v1 已存在的三个示例插件入口;
|
|
365
|
+
> `PluginManager` 仍然不会自动接入 `validate()` / `compile()` / exporter 流程,hook 是否执行仍取决于你的集成代码是否显式调用 `runHook()`。
|
|
366
|
+
|
|
367
|
+
---
|
|
368
|
+
|
|
369
|
+
## 相关文档
|
|
370
|
+
|
|
371
|
+
- [API 参考](api-reference.md)
|
|
372
|
+
- [验证指南](validate.md)
|
|
373
|
+
- [DSL 语法](dsl-syntax.md)
|
|
374
|
+
|
|
375
|
+
---
|
|
376
|
+
|
|
377
|
+
## 对应示例文件
|
|
378
|
+
|
|
379
|
+
**示例入口**: [plugin-system.ts](https://github.com/vextjs/schema-dsl/blob/main/examples/docs/plugin-system.ts)
|
|
380
|
+
**说明**: 覆盖自定义插件的注册 / 安装 / 卸载、`runHook()` 执行结果,以及官方 `custom-format` 子路径插件的安装效果。
|
|
381
|
+
|