@kubb/core 5.0.0-alpha.4 → 5.0.0-alpha.41

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 (71) hide show
  1. package/dist/PluginDriver-BQwm8hDd.cjs +1729 -0
  2. package/dist/PluginDriver-BQwm8hDd.cjs.map +1 -0
  3. package/dist/PluginDriver-CgXFtmNP.js +1617 -0
  4. package/dist/PluginDriver-CgXFtmNP.js.map +1 -0
  5. package/dist/index.cjs +915 -1901
  6. package/dist/index.cjs.map +1 -1
  7. package/dist/index.d.ts +268 -264
  8. package/dist/index.js +894 -1863
  9. package/dist/index.js.map +1 -1
  10. package/dist/mocks.cjs +164 -0
  11. package/dist/mocks.cjs.map +1 -0
  12. package/dist/mocks.d.ts +74 -0
  13. package/dist/mocks.js +159 -0
  14. package/dist/mocks.js.map +1 -0
  15. package/dist/types-C6NCtNqM.d.ts +2151 -0
  16. package/package.json +11 -14
  17. package/src/FileManager.ts +131 -0
  18. package/src/FileProcessor.ts +84 -0
  19. package/src/Kubb.ts +174 -85
  20. package/src/PluginDriver.ts +941 -0
  21. package/src/constants.ts +33 -43
  22. package/src/createAdapter.ts +25 -0
  23. package/src/createKubb.ts +605 -0
  24. package/src/createPlugin.ts +31 -0
  25. package/src/createRenderer.ts +57 -0
  26. package/src/createStorage.ts +58 -0
  27. package/src/defineGenerator.ts +88 -100
  28. package/src/defineLogger.ts +13 -3
  29. package/src/defineParser.ts +45 -0
  30. package/src/definePlugin.ts +90 -7
  31. package/src/defineResolver.ts +453 -0
  32. package/src/devtools.ts +14 -14
  33. package/src/index.ts +12 -17
  34. package/src/mocks.ts +234 -0
  35. package/src/renderNode.ts +35 -0
  36. package/src/storages/fsStorage.ts +29 -9
  37. package/src/storages/memoryStorage.ts +2 -2
  38. package/src/types.ts +821 -152
  39. package/src/utils/TreeNode.ts +47 -9
  40. package/src/utils/diagnostics.ts +4 -1
  41. package/src/utils/executeStrategies.ts +16 -13
  42. package/src/utils/getBarrelFiles.ts +88 -15
  43. package/src/utils/isInputPath.ts +10 -0
  44. package/src/utils/packageJSON.ts +75 -0
  45. package/dist/chunk-ByKO4r7w.cjs +0 -38
  46. package/dist/hooks.cjs +0 -50
  47. package/dist/hooks.cjs.map +0 -1
  48. package/dist/hooks.d.ts +0 -49
  49. package/dist/hooks.js +0 -46
  50. package/dist/hooks.js.map +0 -1
  51. package/dist/types-Bbh1o0yW.d.ts +0 -1057
  52. package/src/BarrelManager.ts +0 -74
  53. package/src/PackageManager.ts +0 -180
  54. package/src/PluginManager.ts +0 -668
  55. package/src/PromiseManager.ts +0 -40
  56. package/src/build.ts +0 -420
  57. package/src/config.ts +0 -56
  58. package/src/defineAdapter.ts +0 -22
  59. package/src/defineStorage.ts +0 -56
  60. package/src/errors.ts +0 -1
  61. package/src/hooks/index.ts +0 -8
  62. package/src/hooks/useKubb.ts +0 -22
  63. package/src/hooks/useMode.ts +0 -11
  64. package/src/hooks/usePlugin.ts +0 -11
  65. package/src/hooks/usePluginManager.ts +0 -11
  66. package/src/utils/FunctionParams.ts +0 -155
  67. package/src/utils/formatters.ts +0 -56
  68. package/src/utils/getConfigs.ts +0 -30
  69. package/src/utils/getPlugins.ts +0 -23
  70. package/src/utils/linters.ts +0 -25
  71. package/src/utils/resolveOptions.ts +0 -93
package/src/mocks.ts ADDED
@@ -0,0 +1,234 @@
1
+ import { resolve } from 'node:path'
2
+ import type { FileNode, OperationNode, SchemaNode, Visitor } from '@kubb/ast'
3
+ import { transform } from '@kubb/ast'
4
+ import { FileManager } from './FileManager.ts'
5
+ import { PluginDriver } from './PluginDriver.ts'
6
+ import { applyHookResult } from './renderNode.ts'
7
+ import type {
8
+ Adapter,
9
+ AdapterFactoryOptions,
10
+ Config,
11
+ Generator,
12
+ GeneratorContext,
13
+ Plugin,
14
+ PluginFactoryOptions,
15
+ ResolveNameParams,
16
+ ResolvePathParams,
17
+ } from './types.ts'
18
+
19
+ function toCamelOrPascal(text: string, pascal: boolean): string {
20
+ const normalized = text
21
+ .trim()
22
+ .replace(/([a-z\d])([A-Z])/g, '$1 $2')
23
+ .replace(/([A-Z]+)([A-Z][a-z])/g, '$1 $2')
24
+ .replace(/(\d)([a-z])/g, '$1 $2')
25
+
26
+ const words = normalized.split(/[\s\-_./\\:]+/).filter(Boolean)
27
+
28
+ return words
29
+ .map((word, i) => {
30
+ const allUpper = word.length > 1 && word === word.toUpperCase()
31
+ if (allUpper) return word
32
+ if (i === 0 && !pascal) return word.charAt(0).toLowerCase() + word.slice(1)
33
+ return word.charAt(0).toUpperCase() + word.slice(1)
34
+ })
35
+ .join('')
36
+ .replace(/[^a-zA-Z0-9]/g, '')
37
+ }
38
+
39
+ function camelCase(text: string): string {
40
+ return toCamelOrPascal(text, false)
41
+ }
42
+
43
+ function pascalCase(text: string): string {
44
+ return toCamelOrPascal(text, true)
45
+ }
46
+
47
+ /**
48
+ * Creates a minimal `PluginDriver` mock suitable for unit tests.
49
+ */
50
+ export function createMockedPluginDriver(options: { name?: string; plugin?: Plugin; config?: Config } = {}): PluginDriver {
51
+ return {
52
+ resolveName: (result: ResolveNameParams) => {
53
+ if (result.type === 'file') {
54
+ return camelCase(options?.name || result.name)
55
+ }
56
+
57
+ if (result.type === 'type') {
58
+ return pascalCase(result.name)
59
+ }
60
+
61
+ if (result.type === 'function') {
62
+ return camelCase(result.name)
63
+ }
64
+
65
+ return camelCase(result.name)
66
+ },
67
+ config: options?.config ?? {
68
+ root: '.',
69
+ output: {
70
+ path: './path',
71
+ },
72
+ },
73
+ resolvePath: ({ baseName }: ResolvePathParams) => baseName,
74
+ getFile: ({
75
+ name,
76
+ extname,
77
+ pluginName,
78
+ options: fileOptions,
79
+ }: {
80
+ name: string
81
+ extname: `.${string}`
82
+ pluginName: string
83
+ options?: { group?: { tag?: string; path?: string } }
84
+ }) => {
85
+ const baseName = `${name}${extname}`
86
+ const groupDir = fileOptions?.group?.tag ?? fileOptions?.group?.path?.split('/').filter(Boolean)[0]
87
+ const filePath = groupDir ? `${groupDir}/${baseName}` : baseName
88
+
89
+ return {
90
+ path: filePath,
91
+ baseName,
92
+ meta: { pluginName },
93
+ }
94
+ },
95
+ getPlugin(_pluginName: Plugin['name']): Plugin | undefined {
96
+ return options?.plugin
97
+ },
98
+ fileManager: new FileManager(),
99
+ } as unknown as PluginDriver
100
+ }
101
+
102
+ /**
103
+ * Creates a minimal `Adapter` mock suitable for unit tests.
104
+ *
105
+ * - `parse` returns an empty `InputNode` by default; override via `options.parse`.
106
+ * - `getImports` returns `[]` by default (single-file mode, no cross-file imports).
107
+ */
108
+ export function createMockedAdapter<TOptions extends AdapterFactoryOptions = AdapterFactoryOptions>(
109
+ options: {
110
+ name?: TOptions['name']
111
+ resolvedOptions?: TOptions['resolvedOptions']
112
+ inputNode?: Adapter<TOptions>['inputNode']
113
+ parse?: Adapter<TOptions>['parse']
114
+ getImports?: Adapter<TOptions>['getImports']
115
+ } = {},
116
+ ): Adapter<TOptions> {
117
+ return {
118
+ name: (options.name ?? 'oas') as TOptions['name'],
119
+ options: (options.resolvedOptions ?? {}) as TOptions['resolvedOptions'],
120
+ inputNode: options.inputNode ?? null,
121
+ parse: options.parse ?? (async () => ({ kind: 'Input' as const, schemas: [], operations: [] })),
122
+ getImports: options.getImports ?? ((_node: SchemaNode, _resolve: (schemaName: string) => { name: string; path: string }) => []),
123
+ } as Adapter<TOptions>
124
+ }
125
+
126
+ /**
127
+ * Creates a minimal `Plugin` mock suitable for unit tests.
128
+ *
129
+ * @example
130
+ * const plugin = createMockedPlugin<PluginTs>({ name: '@kubb/plugin-ts', options })
131
+ */
132
+ export function createMockedPlugin<TOptions extends PluginFactoryOptions = PluginFactoryOptions>(params: {
133
+ name: TOptions['name']
134
+ options: TOptions['resolvedOptions']
135
+ resolver?: TOptions['resolver']
136
+ transformer?: Visitor
137
+ dependencies?: Array<string>
138
+ }): Plugin<TOptions> {
139
+ return {
140
+ name: params.name,
141
+ options: params.options,
142
+ resolver: params.resolver,
143
+ transformer: params.transformer,
144
+ dependencies: params.dependencies,
145
+ install: () => {},
146
+ inject: () => undefined as TOptions['context'],
147
+ } as unknown as Plugin<TOptions>
148
+ }
149
+
150
+ type RenderGeneratorOptions<TOptions extends PluginFactoryOptions> = {
151
+ config: Config
152
+ adapter: Adapter
153
+ driver: PluginDriver
154
+ plugin: Plugin<TOptions>
155
+ options: TOptions['resolvedOptions']
156
+ resolver: TOptions['resolver']
157
+ }
158
+
159
+ function createMockedPluginContext<TOptions extends PluginFactoryOptions>(opts: RenderGeneratorOptions<TOptions>): Omit<GeneratorContext<TOptions>, 'options'> {
160
+ const root = resolve(opts.config.root, opts.config.output.path)
161
+
162
+ return {
163
+ config: opts.config,
164
+ root,
165
+ getMode: (output: { path: string }) => PluginDriver.getMode(resolve(root, output.path)),
166
+ adapter: opts.adapter,
167
+ resolver: opts.resolver,
168
+ plugin: opts.plugin,
169
+ driver: opts.driver,
170
+ inputNode: { kind: 'Input', schemas: [], operations: [] },
171
+ upsertFile: async (...files: Array<FileNode>) => opts.driver.fileManager.upsert(...files),
172
+ warn: (msg: string) => console.warn(msg),
173
+ error: (msg: string) => console.error(msg),
174
+ info: (msg: string) => console.info(msg),
175
+ openInStudio: async () => {},
176
+ } as unknown as Omit<GeneratorContext<TOptions>, 'options'>
177
+ }
178
+
179
+ /**
180
+ * Renders a generator's `schema` method in a test context.
181
+ *
182
+ * @example
183
+ * await renderGeneratorSchema(typeGenerator, node, { config, adapter, driver, plugin, options, resolver })
184
+ * await matchFiles(driver.fileManager.files)
185
+ */
186
+ export async function renderGeneratorSchema<TOptions extends PluginFactoryOptions>(
187
+ generator: Generator<TOptions>,
188
+ node: SchemaNode,
189
+ opts: RenderGeneratorOptions<TOptions>,
190
+ ): Promise<void> {
191
+ if (!generator.schema) return
192
+ const context = createMockedPluginContext(opts)
193
+ const transformedNode = opts.plugin.transformer ? transform(node, opts.plugin.transformer) : node
194
+ const result = await generator.schema(transformedNode, { ...context, options: opts.options })
195
+ await applyHookResult(result, opts.driver, generator.renderer ?? undefined)
196
+ }
197
+
198
+ /**
199
+ * Renders a generator's `operation` method in a test context.
200
+ *
201
+ * @example
202
+ * await renderGeneratorOperation(typeGenerator, node, { config, adapter, driver, plugin, options, resolver })
203
+ * await matchFiles(driver.fileManager.files)
204
+ */
205
+ export async function renderGeneratorOperation<TOptions extends PluginFactoryOptions>(
206
+ generator: Generator<TOptions>,
207
+ node: OperationNode,
208
+ opts: RenderGeneratorOptions<TOptions>,
209
+ ): Promise<void> {
210
+ if (!generator.operation) return
211
+ const context = createMockedPluginContext(opts)
212
+ const transformedNode = opts.plugin.transformer ? transform(node, opts.plugin.transformer) : node
213
+ const result = await generator.operation(transformedNode, { ...context, options: opts.options })
214
+ await applyHookResult(result, opts.driver, generator.renderer ?? undefined)
215
+ }
216
+
217
+ /**
218
+ * Renders a generator's `operations` method in a test context.
219
+ *
220
+ * @example
221
+ * await renderGeneratorOperations(classClientGenerator, nodes, { config, adapter, driver, plugin, options, resolver })
222
+ * await matchFiles(driver.fileManager.files)
223
+ */
224
+ export async function renderGeneratorOperations<TOptions extends PluginFactoryOptions>(
225
+ generator: Generator<TOptions>,
226
+ nodes: Array<OperationNode>,
227
+ opts: RenderGeneratorOptions<TOptions>,
228
+ ): Promise<void> {
229
+ if (!generator.operations) return
230
+ const context = createMockedPluginContext(opts)
231
+ const transformedNodes = opts.plugin.transformer ? nodes.map((n) => transform(n, opts.plugin.transformer!)) : nodes
232
+ const result = await generator.operations(transformedNodes, { ...context, options: opts.options })
233
+ await applyHookResult(result, opts.driver, generator.renderer ?? undefined)
234
+ }
@@ -0,0 +1,35 @@
1
+ import type { FileNode } from '@kubb/ast'
2
+ import type { RendererFactory } from './createRenderer.ts'
3
+ import type { PluginDriver } from './PluginDriver.ts'
4
+
5
+ /**
6
+ * Handles the return value of a plugin AST hook or generator method.
7
+ *
8
+ * - Renderer output → rendered via the provided `rendererFactory` (e.g. JSX), files stored in `driver.fileManager`
9
+ * - `Array<FileNode>` → added directly into `driver.fileManager`
10
+ * - `void` / `null` / `undefined` → no-op (plugin handled it via `this.upsertFile`)
11
+ *
12
+ * Pass a `rendererFactory` (e.g. `jsxRenderer` from `@kubb/renderer-jsx`) when the result
13
+ * may be a renderer element. Generators that only return `Array<FileNode>` do not need one.
14
+ */
15
+ export async function applyHookResult<TElement = unknown>(
16
+ result: TElement | Array<FileNode> | void,
17
+ driver: PluginDriver,
18
+ rendererFactory?: RendererFactory<TElement>,
19
+ ): Promise<void> {
20
+ if (!result) return
21
+
22
+ if (Array.isArray(result)) {
23
+ driver.fileManager.upsert(...(result as Array<FileNode>))
24
+ return
25
+ }
26
+
27
+ if (!rendererFactory) {
28
+ return
29
+ }
30
+
31
+ const renderer = rendererFactory()
32
+ await renderer.render(result)
33
+ driver.fileManager.upsert(...renderer.files)
34
+ renderer.unmount()
35
+ }
@@ -2,7 +2,14 @@ import type { Dirent } from 'node:fs'
2
2
  import { access, readdir, readFile, rm } from 'node:fs/promises'
3
3
  import { join, resolve } from 'node:path'
4
4
  import { clean, write } from '@internals/utils'
5
- import { defineStorage } from '../defineStorage.ts'
5
+ import { createStorage } from '../createStorage.ts'
6
+
7
+ /**
8
+ * Detects the filesystem error used to indicate that a path does not exist.
9
+ */
10
+ function isMissingPathError(error: unknown): error is NodeJS.ErrnoException {
11
+ return typeof error === 'object' && error !== null && 'code' in error && (error as NodeJS.ErrnoException).code === 'ENOENT'
12
+ }
6
13
 
7
14
  /**
8
15
  * Built-in filesystem storage driver.
@@ -27,21 +34,29 @@ import { defineStorage } from '../defineStorage.ts'
27
34
  * })
28
35
  * ```
29
36
  */
30
- export const fsStorage = defineStorage(() => ({
37
+ export const fsStorage = createStorage(() => ({
31
38
  name: 'fs',
32
39
  async hasItem(key: string) {
33
40
  try {
34
41
  await access(resolve(key))
35
42
  return true
36
- } catch {
37
- return false
43
+ } catch (error) {
44
+ if (isMissingPathError(error)) {
45
+ return false
46
+ }
47
+
48
+ throw new Error(`Failed to access storage item "${key}"`, { cause: error as Error })
38
49
  }
39
50
  },
40
51
  async getItem(key: string) {
41
52
  try {
42
53
  return await readFile(resolve(key), 'utf8')
43
- } catch {
44
- return null
54
+ } catch (error) {
55
+ if (isMissingPathError(error)) {
56
+ return null
57
+ }
58
+
59
+ throw new Error(`Failed to read storage item "${key}"`, { cause: error as Error })
45
60
  }
46
61
  },
47
62
  async setItem(key: string, value: string) {
@@ -52,13 +67,18 @@ export const fsStorage = defineStorage(() => ({
52
67
  },
53
68
  async getKeys(base?: string) {
54
69
  const keys: Array<string> = []
70
+ const resolvedBase = resolve(base ?? process.cwd())
55
71
 
56
72
  async function walk(dir: string, prefix: string): Promise<void> {
57
73
  let entries: Array<Dirent>
58
74
  try {
59
75
  entries = (await readdir(dir, { withFileTypes: true })) as Array<Dirent>
60
- } catch {
61
- return
76
+ } catch (error) {
77
+ if (isMissingPathError(error)) {
78
+ return
79
+ }
80
+
81
+ throw new Error(`Failed to list storage keys under "${resolvedBase}"`, { cause: error as Error })
62
82
  }
63
83
  for (const entry of entries) {
64
84
  const rel = prefix ? `${prefix}/${entry.name}` : entry.name
@@ -70,7 +90,7 @@ export const fsStorage = defineStorage(() => ({
70
90
  }
71
91
  }
72
92
 
73
- await walk(resolve(base ?? process.cwd()), '')
93
+ await walk(resolvedBase, '')
74
94
 
75
95
  return keys
76
96
  },
@@ -1,4 +1,4 @@
1
- import { defineStorage } from '../defineStorage.ts'
1
+ import { createStorage } from '../createStorage.ts'
2
2
 
3
3
  /**
4
4
  * In-memory storage driver. Useful for testing and dry-run scenarios where
@@ -17,7 +17,7 @@ import { defineStorage } from '../defineStorage.ts'
17
17
  * })
18
18
  * ```
19
19
  */
20
- export const memoryStorage = defineStorage(() => {
20
+ export const memoryStorage = createStorage(() => {
21
21
  const store = new Map<string, string>()
22
22
 
23
23
  return {