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.
Files changed (145) hide show
  1. package/CHANGELOG.md +130 -113
  2. package/LICENSE +21 -21
  3. package/README.md +628 -628
  4. package/dist/{DslBuilder-DkLaOo9Q.d.ts → DslBuilder-BIgQOAXp.d.ts} +2 -0
  5. package/dist/{DslBuilder-DQDN0ZxZ.d.cts → DslBuilder-CjHTucNQ.d.cts} +2 -0
  6. package/dist/{Validator-hFWKGxir.d.ts → Validator-CllRdrY0.d.ts} +1 -1
  7. package/dist/{Validator-C7GsVQOH.d.cts → Validator-D6okG9tr.d.cts} +1 -1
  8. package/dist/index.cjs +75 -29
  9. package/dist/index.d.cts +10 -4
  10. package/dist/index.d.ts +10 -4
  11. package/dist/index.js +75 -29
  12. package/dist/plugins/custom-format.cjs +33 -17
  13. package/dist/plugins/custom-format.d.cts +1 -1
  14. package/dist/plugins/custom-format.d.ts +1 -1
  15. package/dist/plugins/custom-format.js +33 -17
  16. package/dist/plugins/custom-type-example.cjs +33 -17
  17. package/dist/plugins/custom-type-example.d.cts +1 -1
  18. package/dist/plugins/custom-type-example.d.ts +1 -1
  19. package/dist/plugins/custom-type-example.js +33 -17
  20. package/dist/plugins/custom-validator.cjs +0 -2
  21. package/dist/plugins/custom-validator.d.cts +1 -1
  22. package/dist/plugins/custom-validator.d.ts +1 -1
  23. package/dist/plugins/custom-validator.js +0 -2
  24. package/docs/FEATURE-INDEX.md +553 -553
  25. package/docs/add-custom-locale.md +496 -496
  26. package/docs/add-keyword.md +24 -24
  27. package/docs/api-reference.md +1047 -1047
  28. package/docs/api.md +13 -13
  29. package/docs/best-practices-project-structure.md +417 -417
  30. package/docs/best-practices.md +712 -712
  31. package/docs/cache-manager.md +344 -344
  32. package/docs/compile.md +45 -45
  33. package/docs/conditional-api.md +1307 -1307
  34. package/docs/custom-extensions-guide.md +339 -339
  35. package/docs/design-philosophy.md +606 -606
  36. package/docs/doc-index.md +324 -324
  37. package/docs/dsl-syntax.md +714 -714
  38. package/docs/dynamic-locale.md +608 -608
  39. package/docs/enum.md +482 -482
  40. package/docs/error-handling.md +1975 -1975
  41. package/docs/export-guide.md +501 -501
  42. package/docs/export-limitations.md +567 -567
  43. package/docs/faq.md +596 -596
  44. package/docs/frontend-i18n-guide.md +307 -307
  45. package/docs/i18n-user-guide.md +487 -487
  46. package/docs/i18n.md +476 -476
  47. package/docs/index.md +48 -48
  48. package/docs/json-schema-basics.md +40 -40
  49. package/docs/label-vs-description.md +271 -271
  50. package/docs/markdown-exporter.md +406 -406
  51. package/docs/mongodb-exporter.md +302 -302
  52. package/docs/multi-language.md +26 -26
  53. package/docs/multi-type-support.md +322 -322
  54. package/docs/mysql-exporter.md +280 -280
  55. package/docs/number-operators.md +449 -449
  56. package/docs/optional-marker-guide.md +326 -326
  57. package/docs/performance-guide.md +49 -49
  58. package/docs/plugin-system.md +381 -381
  59. package/docs/plugin-type-registration.md +34 -34
  60. package/docs/postgresql-exporter.md +311 -311
  61. package/docs/public/favicon.svg +4 -4
  62. package/docs/quick-start.md +435 -435
  63. package/docs/runtime-locale-support.md +532 -532
  64. package/docs/schema-helper.md +345 -345
  65. package/docs/schema-utils-advanced-issues.md +23 -23
  66. package/docs/schema-utils-best-practices.md +20 -20
  67. package/docs/schema-utils-chaining.md +150 -150
  68. package/docs/schema-utils.md +524 -524
  69. package/docs/security-checklist.md +20 -20
  70. package/docs/string-extensions.md +488 -488
  71. package/docs/troubleshooting.md +486 -486
  72. package/docs/type-converter.md +310 -310
  73. package/docs/type-reference.md +242 -242
  74. package/docs/typescript-guide.md +584 -584
  75. package/docs/union-type-guide.md +157 -157
  76. package/docs/union-types.md +284 -284
  77. package/docs/validate-async.md +491 -491
  78. package/docs/validate-batch.md +49 -49
  79. package/docs/validate-dsl-object-support.md +578 -578
  80. package/docs/validate.md +506 -506
  81. package/docs/validation-guide.md +502 -502
  82. package/docs/validator.md +39 -39
  83. package/package.json +131 -131
  84. package/plugins/custom-format.cjs +8 -8
  85. package/plugins/custom-type-example.cjs +8 -8
  86. package/plugins/custom-validator.cjs +8 -8
  87. package/src/adapters/DslAdapter.ts +111 -111
  88. package/src/adapters/index.ts +1 -1
  89. package/src/config/constants.ts +83 -83
  90. package/src/config/index.ts +2 -2
  91. package/src/config/patterns.ts +77 -77
  92. package/src/core/CacheManager.ts +169 -159
  93. package/src/core/ConditionalBuilder.ts +382 -382
  94. package/src/core/ConditionalRuntime.ts +27 -27
  95. package/src/core/ConditionalValidator.ts +254 -254
  96. package/src/core/DslBuilder.ts +687 -677
  97. package/src/core/ErrorCodes.ts +38 -38
  98. package/src/core/ErrorFormatter.ts +271 -271
  99. package/src/core/JSONSchemaCore.ts +65 -65
  100. package/src/core/Locale.ts +187 -187
  101. package/src/core/MessageTemplate.ts +42 -42
  102. package/src/core/ObjectDslBuilder.ts +64 -64
  103. package/src/core/PluginManager.ts +326 -326
  104. package/src/core/StringExtensions.ts +140 -140
  105. package/src/core/TemplateEngine.ts +44 -44
  106. package/src/core/Validator.ts +448 -448
  107. package/src/errors/I18nError.ts +159 -159
  108. package/src/errors/ValidationError.ts +105 -105
  109. package/src/exporters/BaseExporter.ts +60 -60
  110. package/src/exporters/MarkdownExporter.ts +305 -305
  111. package/src/exporters/MongoDBExporter.ts +126 -126
  112. package/src/exporters/MySQLExporter.ts +156 -155
  113. package/src/exporters/PostgreSQLExporter.ts +222 -222
  114. package/src/exporters/index.ts +18 -18
  115. package/src/index.ts +651 -633
  116. package/src/locales/en-US.ts +160 -160
  117. package/src/locales/es-ES.ts +160 -160
  118. package/src/locales/fr-FR.ts +160 -160
  119. package/src/locales/index.ts +103 -103
  120. package/src/locales/ja-JP.ts +160 -160
  121. package/src/locales/types.ts +156 -156
  122. package/src/locales/zh-CN.ts +160 -160
  123. package/src/parser/ConstraintParser.ts +101 -101
  124. package/src/parser/DslParser.ts +470 -470
  125. package/src/parser/SchemaCompiler.ts +66 -66
  126. package/src/parser/TypeRegistry.ts +250 -250
  127. package/src/parser/index.ts +6 -6
  128. package/src/plugins/custom-format.ts +124 -126
  129. package/src/plugins/custom-type-example.ts +106 -108
  130. package/src/plugins/custom-validator.ts +138 -140
  131. package/src/types/conditional.ts +28 -28
  132. package/src/types/config.ts +59 -59
  133. package/src/types/dsl.ts +131 -131
  134. package/src/types/error.ts +60 -60
  135. package/src/types/index.ts +17 -17
  136. package/src/types/infer.ts +127 -127
  137. package/src/types/plugin.ts +58 -58
  138. package/src/types/safe-regex.d.ts +9 -9
  139. package/src/types/schema.ts +66 -66
  140. package/src/types/validate.ts +71 -71
  141. package/src/utils/SchemaHelper.ts +196 -196
  142. package/src/utils/SchemaUtils.ts +365 -346
  143. package/src/utils/TypeConverter.ts +215 -215
  144. package/src/utils/index.ts +10 -10
  145. package/src/validators/CustomKeywords.ts +477 -477
@@ -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
+