@shepherdjerred/helm-types 0.0.0-dev.706
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/README.md +209 -0
- package/dist/cli.js +22085 -0
- package/dist/index.js +23 -0
- package/package.json +66 -0
- package/src/chart-fetcher.ts +171 -0
- package/src/chart-info-parser.ts +72 -0
- package/src/cli.ts +215 -0
- package/src/code-generator.ts +226 -0
- package/src/comment-parser.ts +180 -0
- package/src/config.ts +147 -0
- package/src/helm-types.ts +16 -0
- package/src/index.ts +28 -0
- package/src/interface-generator.ts +238 -0
- package/src/reset.d.ts +1 -0
- package/src/schemas.ts +39 -0
- package/src/type-converter-helpers.ts +180 -0
- package/src/type-converter.ts +509 -0
- package/src/type-inference.ts +548 -0
- package/src/types.ts +38 -0
- package/src/utils.ts +76 -0
- package/src/yaml-comment-filters.ts +103 -0
- package/src/yaml-comment-regex-parser.ts +150 -0
- package/src/yaml-comments.ts +507 -0
- package/src/yaml-preprocess.ts +235 -0
|
@@ -0,0 +1,548 @@
|
|
|
1
|
+
import type {
|
|
2
|
+
JSONSchemaProperty,
|
|
3
|
+
TypeScriptInterface,
|
|
4
|
+
TypeProperty,
|
|
5
|
+
} from "./types.ts";
|
|
6
|
+
import type { HelmValue } from "./schemas.ts";
|
|
7
|
+
import {
|
|
8
|
+
StringSchema,
|
|
9
|
+
ActualNumberSchema,
|
|
10
|
+
ActualBooleanSchema,
|
|
11
|
+
NullSchema,
|
|
12
|
+
UndefinedSchema,
|
|
13
|
+
ArraySchema,
|
|
14
|
+
HelmValueSchema,
|
|
15
|
+
StringBooleanSchema,
|
|
16
|
+
} from "./schemas.ts";
|
|
17
|
+
import {
|
|
18
|
+
capitalizeFirst,
|
|
19
|
+
sanitizePropertyName,
|
|
20
|
+
sanitizeTypeName,
|
|
21
|
+
} from "./utils.ts";
|
|
22
|
+
|
|
23
|
+
/**
|
|
24
|
+
* Convert JSON schema type to TypeScript type string
|
|
25
|
+
*/
|
|
26
|
+
export function jsonSchemaToTypeScript(schema: JSONSchemaProperty): string {
|
|
27
|
+
// Handle oneOf - union of types
|
|
28
|
+
if (schema.oneOf) {
|
|
29
|
+
const types = schema.oneOf.map((s) => jsonSchemaToTypeScript(s));
|
|
30
|
+
return types.join(" | ");
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
// Handle anyOf - union of types
|
|
34
|
+
if (schema.anyOf) {
|
|
35
|
+
const types = schema.anyOf.map((s) => jsonSchemaToTypeScript(s));
|
|
36
|
+
return types.join(" | ");
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
// Handle enum
|
|
40
|
+
if (schema.enum) {
|
|
41
|
+
return schema.enum
|
|
42
|
+
.map((v) =>
|
|
43
|
+
StringSchema.safeParse(v).success ? `"${String(v)}"` : String(v),
|
|
44
|
+
)
|
|
45
|
+
.join(" | ");
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
// Handle array type
|
|
49
|
+
if (schema.type === "array" && schema.items) {
|
|
50
|
+
const itemType = jsonSchemaToTypeScript(schema.items);
|
|
51
|
+
return `${itemType}[]`;
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
// Handle basic types
|
|
55
|
+
const stringTypeCheck = StringSchema.safeParse(schema.type);
|
|
56
|
+
if (stringTypeCheck.success) {
|
|
57
|
+
switch (stringTypeCheck.data) {
|
|
58
|
+
case "string":
|
|
59
|
+
return "string";
|
|
60
|
+
case "number":
|
|
61
|
+
case "integer":
|
|
62
|
+
return "number";
|
|
63
|
+
case "boolean":
|
|
64
|
+
return "boolean";
|
|
65
|
+
case "object":
|
|
66
|
+
return "object";
|
|
67
|
+
case "array":
|
|
68
|
+
return "unknown[]";
|
|
69
|
+
case "null":
|
|
70
|
+
return "null";
|
|
71
|
+
default:
|
|
72
|
+
return "unknown";
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
// Handle multiple types
|
|
77
|
+
const arrayTypeCheck = ArraySchema.safeParse(schema.type);
|
|
78
|
+
if (arrayTypeCheck.success) {
|
|
79
|
+
return arrayTypeCheck.data
|
|
80
|
+
.map((t: unknown) => {
|
|
81
|
+
if (!StringSchema.safeParse(t).success) {
|
|
82
|
+
return "unknown";
|
|
83
|
+
}
|
|
84
|
+
const typeStr = String(t);
|
|
85
|
+
switch (typeStr) {
|
|
86
|
+
case "string":
|
|
87
|
+
return "string";
|
|
88
|
+
case "number":
|
|
89
|
+
case "integer":
|
|
90
|
+
return "number";
|
|
91
|
+
case "boolean":
|
|
92
|
+
return "boolean";
|
|
93
|
+
case "object":
|
|
94
|
+
return "object";
|
|
95
|
+
case "array":
|
|
96
|
+
return "unknown[]";
|
|
97
|
+
case "null":
|
|
98
|
+
return "null";
|
|
99
|
+
default:
|
|
100
|
+
return "unknown";
|
|
101
|
+
}
|
|
102
|
+
})
|
|
103
|
+
.join(" | ");
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
return "unknown";
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
/**
|
|
110
|
+
* Infer TypeScript type from actual runtime value
|
|
111
|
+
*/
|
|
112
|
+
export function inferTypeFromValue(value: unknown): string | null {
|
|
113
|
+
// Check null/undefined
|
|
114
|
+
if (
|
|
115
|
+
NullSchema.safeParse(value).success ||
|
|
116
|
+
UndefinedSchema.safeParse(value).success
|
|
117
|
+
) {
|
|
118
|
+
return null;
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
// Check for actual boolean
|
|
122
|
+
if (ActualBooleanSchema.safeParse(value).success) {
|
|
123
|
+
return "boolean";
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
// Check for actual number
|
|
127
|
+
if (ActualNumberSchema.safeParse(value).success) {
|
|
128
|
+
return "number";
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
// Check if it's a string that looks like a boolean
|
|
132
|
+
if (StringBooleanSchema.safeParse(value).success) {
|
|
133
|
+
return "boolean";
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
// Check if it's a string that looks like a number
|
|
137
|
+
const stringCheck = StringSchema.safeParse(value);
|
|
138
|
+
if (stringCheck.success) {
|
|
139
|
+
const trimmed = stringCheck.data.trim();
|
|
140
|
+
if (
|
|
141
|
+
trimmed !== "" &&
|
|
142
|
+
!Number.isNaN(Number(trimmed)) &&
|
|
143
|
+
Number.isFinite(Number(trimmed))
|
|
144
|
+
) {
|
|
145
|
+
return "number";
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
// Check for array
|
|
150
|
+
if (ArraySchema.safeParse(value).success) {
|
|
151
|
+
return "array";
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
// Check for object
|
|
155
|
+
if (HelmValueSchema.safeParse(value).success) {
|
|
156
|
+
return "object";
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
// Plain string
|
|
160
|
+
if (StringSchema.safeParse(value).success) {
|
|
161
|
+
return "string";
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
return "unknown";
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
/**
|
|
168
|
+
* Check if inferred type is compatible with schema type
|
|
169
|
+
*/
|
|
170
|
+
export function typesAreCompatible(
|
|
171
|
+
inferredType: string,
|
|
172
|
+
schemaType: string,
|
|
173
|
+
): boolean {
|
|
174
|
+
// Exact match
|
|
175
|
+
if (inferredType === schemaType) {
|
|
176
|
+
return true;
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
// Check if the inferred type is part of a union in the schema
|
|
180
|
+
// For example: schemaType might be "number | \"default\"" and inferredType is "string"
|
|
181
|
+
const schemaTypes = schemaType
|
|
182
|
+
.split("|")
|
|
183
|
+
.map((t) => t.trim().replaceAll(/^["']|["']$/g, ""));
|
|
184
|
+
|
|
185
|
+
// If schema is a union, check if inferred type is compatible with any part
|
|
186
|
+
if (schemaTypes.length > 1) {
|
|
187
|
+
for (const st of schemaTypes) {
|
|
188
|
+
// Handle quoted strings in unions (like "default")
|
|
189
|
+
if (st.startsWith('"') && st.endsWith('"') && inferredType === "string") {
|
|
190
|
+
return true;
|
|
191
|
+
}
|
|
192
|
+
if (st === inferredType) {
|
|
193
|
+
return true;
|
|
194
|
+
}
|
|
195
|
+
// Arrays
|
|
196
|
+
if (st.endsWith("[]") && inferredType === "array") {
|
|
197
|
+
return true;
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
// Handle array types
|
|
203
|
+
if (schemaType.endsWith("[]") && inferredType === "array") {
|
|
204
|
+
return true;
|
|
205
|
+
}
|
|
206
|
+
|
|
207
|
+
// Handle specific string literals - if schema expects specific strings and value is a string
|
|
208
|
+
if (schemaType.includes('"') && inferredType === "string") {
|
|
209
|
+
return true;
|
|
210
|
+
}
|
|
211
|
+
|
|
212
|
+
// unknown is compatible with everything (schema might be less specific)
|
|
213
|
+
if (schemaType === "unknown" || inferredType === "unknown") {
|
|
214
|
+
return true;
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
return false;
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
/**
|
|
221
|
+
* Convert Helm values to TypeScript interface
|
|
222
|
+
*/
|
|
223
|
+
export function convertToTypeScriptInterface(options: {
|
|
224
|
+
values: HelmValue;
|
|
225
|
+
interfaceName: string;
|
|
226
|
+
schema?: JSONSchemaProperty | null;
|
|
227
|
+
yamlComments?: Map<string, string>;
|
|
228
|
+
keyPrefix?: string;
|
|
229
|
+
}): TypeScriptInterface {
|
|
230
|
+
const keyPrefix = options.keyPrefix ?? "";
|
|
231
|
+
const properties: Record<string, TypeProperty> = {};
|
|
232
|
+
const schemaProps = options.schema?.properties;
|
|
233
|
+
|
|
234
|
+
for (const [key, value] of Object.entries(options.values)) {
|
|
235
|
+
const sanitizedKey = sanitizePropertyName(key);
|
|
236
|
+
const typeNameSuffix = sanitizeTypeName(key);
|
|
237
|
+
const propertySchema = schemaProps?.[key];
|
|
238
|
+
const fullKey = keyPrefix ? `${keyPrefix}.${key}` : key;
|
|
239
|
+
const yamlComment = options.yamlComments?.get(fullKey);
|
|
240
|
+
|
|
241
|
+
properties[sanitizedKey] = convertValueToProperty({
|
|
242
|
+
value,
|
|
243
|
+
nestedTypeName: `${options.interfaceName}${capitalizeFirst(typeNameSuffix)}`,
|
|
244
|
+
schema: propertySchema,
|
|
245
|
+
propertyName: key,
|
|
246
|
+
yamlComment,
|
|
247
|
+
yamlComments: options.yamlComments,
|
|
248
|
+
fullKey,
|
|
249
|
+
});
|
|
250
|
+
}
|
|
251
|
+
|
|
252
|
+
return {
|
|
253
|
+
name: options.interfaceName,
|
|
254
|
+
properties,
|
|
255
|
+
};
|
|
256
|
+
}
|
|
257
|
+
|
|
258
|
+
type InferencePropertyContext = {
|
|
259
|
+
value: unknown;
|
|
260
|
+
nestedTypeName: string;
|
|
261
|
+
schema?: JSONSchemaProperty;
|
|
262
|
+
propertyName?: string;
|
|
263
|
+
yamlComment?: string;
|
|
264
|
+
yamlComments?: Map<string, string>;
|
|
265
|
+
fullKey?: string;
|
|
266
|
+
};
|
|
267
|
+
|
|
268
|
+
/**
|
|
269
|
+
* Merge description from schema and YAML comments
|
|
270
|
+
*/
|
|
271
|
+
function mergeDescriptions(
|
|
272
|
+
schemaDescription: string | undefined,
|
|
273
|
+
yamlComment: string | undefined,
|
|
274
|
+
): string | undefined {
|
|
275
|
+
if (yamlComment == null || yamlComment === "") {
|
|
276
|
+
return schemaDescription;
|
|
277
|
+
}
|
|
278
|
+
return schemaDescription != null && schemaDescription !== ""
|
|
279
|
+
? `${yamlComment}\n\n${schemaDescription}`
|
|
280
|
+
: yamlComment;
|
|
281
|
+
}
|
|
282
|
+
|
|
283
|
+
/**
|
|
284
|
+
* Convert a value to a TypeProperty using JSON schema information
|
|
285
|
+
*/
|
|
286
|
+
function convertWithSchema(
|
|
287
|
+
ctx: InferencePropertyContext & { schema: JSONSchemaProperty },
|
|
288
|
+
): TypeProperty {
|
|
289
|
+
const {
|
|
290
|
+
value,
|
|
291
|
+
nestedTypeName,
|
|
292
|
+
schema,
|
|
293
|
+
propertyName,
|
|
294
|
+
yamlComment,
|
|
295
|
+
yamlComments,
|
|
296
|
+
fullKey,
|
|
297
|
+
} = ctx;
|
|
298
|
+
|
|
299
|
+
const inferredType = inferTypeFromValue(value);
|
|
300
|
+
const schemaType = jsonSchemaToTypeScript(schema);
|
|
301
|
+
|
|
302
|
+
if (
|
|
303
|
+
inferredType != null &&
|
|
304
|
+
inferredType !== "" &&
|
|
305
|
+
!typesAreCompatible(inferredType, schemaType)
|
|
306
|
+
) {
|
|
307
|
+
const propName =
|
|
308
|
+
propertyName != null && propertyName !== "" ? `'${propertyName}': ` : "";
|
|
309
|
+
console.warn(
|
|
310
|
+
` ⚠️ Type mismatch for ${propName}Schema says '${schemaType}' but value suggests '${inferredType}' (value: ${String(value).slice(0, 50)})`,
|
|
311
|
+
);
|
|
312
|
+
}
|
|
313
|
+
|
|
314
|
+
const description = mergeDescriptions(schema.description, yamlComment);
|
|
315
|
+
const defaultValue = schema.default === undefined ? value : schema.default;
|
|
316
|
+
|
|
317
|
+
// If schema defines it as an object with properties, recurse
|
|
318
|
+
const helmValueCheckForProps = HelmValueSchema.safeParse(value);
|
|
319
|
+
if (schema.properties && helmValueCheckForProps.success) {
|
|
320
|
+
const nestedInterface = convertToTypeScriptInterface({
|
|
321
|
+
values: helmValueCheckForProps.data,
|
|
322
|
+
interfaceName: nestedTypeName,
|
|
323
|
+
schema,
|
|
324
|
+
yamlComments,
|
|
325
|
+
keyPrefix: fullKey,
|
|
326
|
+
});
|
|
327
|
+
return {
|
|
328
|
+
type: nestedTypeName,
|
|
329
|
+
optional: true,
|
|
330
|
+
nested: nestedInterface,
|
|
331
|
+
description,
|
|
332
|
+
default: defaultValue,
|
|
333
|
+
};
|
|
334
|
+
}
|
|
335
|
+
|
|
336
|
+
// Handle object types without explicit properties
|
|
337
|
+
const helmValueCheckForObject = HelmValueSchema.safeParse(value);
|
|
338
|
+
if (schemaType === "object" && helmValueCheckForObject.success) {
|
|
339
|
+
const nestedInterface = convertToTypeScriptInterface({
|
|
340
|
+
values: helmValueCheckForObject.data,
|
|
341
|
+
interfaceName: nestedTypeName,
|
|
342
|
+
yamlComments,
|
|
343
|
+
keyPrefix: fullKey,
|
|
344
|
+
});
|
|
345
|
+
return {
|
|
346
|
+
type: nestedTypeName,
|
|
347
|
+
optional: true,
|
|
348
|
+
nested: nestedInterface,
|
|
349
|
+
description,
|
|
350
|
+
default: defaultValue,
|
|
351
|
+
};
|
|
352
|
+
}
|
|
353
|
+
|
|
354
|
+
return {
|
|
355
|
+
type: schemaType,
|
|
356
|
+
optional: true,
|
|
357
|
+
description,
|
|
358
|
+
default: defaultValue,
|
|
359
|
+
};
|
|
360
|
+
}
|
|
361
|
+
|
|
362
|
+
/**
|
|
363
|
+
* Infer array element type from sampled elements
|
|
364
|
+
*/
|
|
365
|
+
function inferArrayType(
|
|
366
|
+
nestedTypeName: string,
|
|
367
|
+
arrayValue: unknown[],
|
|
368
|
+
): TypeProperty {
|
|
369
|
+
if (arrayValue.length === 0) {
|
|
370
|
+
return { type: "unknown[]", optional: true };
|
|
371
|
+
}
|
|
372
|
+
|
|
373
|
+
const elementTypes = new Set<string>();
|
|
374
|
+
const elementTypeProps: TypeProperty[] = [];
|
|
375
|
+
const sampleSize = Math.min(arrayValue.length, 3);
|
|
376
|
+
|
|
377
|
+
for (let i = 0; i < sampleSize; i++) {
|
|
378
|
+
const elementType = convertValueToProperty({
|
|
379
|
+
value: arrayValue[i],
|
|
380
|
+
nestedTypeName,
|
|
381
|
+
});
|
|
382
|
+
elementTypes.add(elementType.type);
|
|
383
|
+
elementTypeProps.push(elementType);
|
|
384
|
+
}
|
|
385
|
+
|
|
386
|
+
if (elementTypes.size === 1) {
|
|
387
|
+
return inferUniformArrayType(
|
|
388
|
+
elementTypes,
|
|
389
|
+
elementTypeProps,
|
|
390
|
+
nestedTypeName,
|
|
391
|
+
);
|
|
392
|
+
}
|
|
393
|
+
|
|
394
|
+
const types = [...elementTypes].toSorted();
|
|
395
|
+
if (
|
|
396
|
+
types.length <= 3 &&
|
|
397
|
+
types.every((t) => ["string", "number", "boolean"].includes(t))
|
|
398
|
+
) {
|
|
399
|
+
return { type: `(${types.join(" | ")})[]`, optional: true };
|
|
400
|
+
}
|
|
401
|
+
|
|
402
|
+
return { type: "unknown[]", optional: true };
|
|
403
|
+
}
|
|
404
|
+
|
|
405
|
+
/**
|
|
406
|
+
* Build TypeProperty for a uniform-type array
|
|
407
|
+
*/
|
|
408
|
+
function inferUniformArrayType(
|
|
409
|
+
elementTypes: Set<string>,
|
|
410
|
+
elementTypeProps: TypeProperty[],
|
|
411
|
+
nestedTypeName: string,
|
|
412
|
+
): TypeProperty {
|
|
413
|
+
const elementType = [...elementTypes][0];
|
|
414
|
+
const elementProp = elementTypeProps[0];
|
|
415
|
+
if (elementType == null || elementType === "" || !elementProp) {
|
|
416
|
+
return { type: "unknown[]", optional: true };
|
|
417
|
+
}
|
|
418
|
+
|
|
419
|
+
if (elementProp.nested) {
|
|
420
|
+
const arrayElementTypeName = `${nestedTypeName}Element`;
|
|
421
|
+
const arrayElementInterface: TypeScriptInterface = {
|
|
422
|
+
name: arrayElementTypeName,
|
|
423
|
+
properties: elementProp.nested.properties,
|
|
424
|
+
};
|
|
425
|
+
return {
|
|
426
|
+
type: `${arrayElementTypeName}[]`,
|
|
427
|
+
optional: true,
|
|
428
|
+
nested: arrayElementInterface,
|
|
429
|
+
};
|
|
430
|
+
}
|
|
431
|
+
|
|
432
|
+
return { type: `${elementType}[]`, optional: true };
|
|
433
|
+
}
|
|
434
|
+
|
|
435
|
+
/**
|
|
436
|
+
* Infer a primitive TypeProperty from a runtime value (no schema)
|
|
437
|
+
*/
|
|
438
|
+
function inferPrimitiveType(
|
|
439
|
+
value: unknown,
|
|
440
|
+
yamlComment?: string,
|
|
441
|
+
): TypeProperty {
|
|
442
|
+
if (ActualBooleanSchema.safeParse(value).success) {
|
|
443
|
+
return {
|
|
444
|
+
type: "boolean",
|
|
445
|
+
optional: true,
|
|
446
|
+
description: yamlComment,
|
|
447
|
+
default: value,
|
|
448
|
+
};
|
|
449
|
+
}
|
|
450
|
+
|
|
451
|
+
if (ActualNumberSchema.safeParse(value).success) {
|
|
452
|
+
return {
|
|
453
|
+
type: "number",
|
|
454
|
+
optional: true,
|
|
455
|
+
description: yamlComment,
|
|
456
|
+
default: value,
|
|
457
|
+
};
|
|
458
|
+
}
|
|
459
|
+
|
|
460
|
+
if (StringBooleanSchema.safeParse(value).success) {
|
|
461
|
+
return {
|
|
462
|
+
type: "boolean",
|
|
463
|
+
optional: true,
|
|
464
|
+
description: yamlComment,
|
|
465
|
+
default: value,
|
|
466
|
+
};
|
|
467
|
+
}
|
|
468
|
+
|
|
469
|
+
const stringCheckForNumber = StringSchema.safeParse(value);
|
|
470
|
+
if (stringCheckForNumber.success) {
|
|
471
|
+
const trimmed = stringCheckForNumber.data.trim();
|
|
472
|
+
if (
|
|
473
|
+
trimmed !== "" &&
|
|
474
|
+
!Number.isNaN(Number(trimmed)) &&
|
|
475
|
+
Number.isFinite(Number(trimmed))
|
|
476
|
+
) {
|
|
477
|
+
return {
|
|
478
|
+
type: "number",
|
|
479
|
+
optional: true,
|
|
480
|
+
description: yamlComment,
|
|
481
|
+
default: value,
|
|
482
|
+
};
|
|
483
|
+
}
|
|
484
|
+
}
|
|
485
|
+
|
|
486
|
+
const stringCheckForPlain = StringSchema.safeParse(value);
|
|
487
|
+
if (stringCheckForPlain.success) {
|
|
488
|
+
if (stringCheckForPlain.data === "default") {
|
|
489
|
+
return {
|
|
490
|
+
type: "string | number | boolean",
|
|
491
|
+
optional: true,
|
|
492
|
+
description: yamlComment,
|
|
493
|
+
default: value,
|
|
494
|
+
};
|
|
495
|
+
}
|
|
496
|
+
return {
|
|
497
|
+
type: "string",
|
|
498
|
+
optional: true,
|
|
499
|
+
description: yamlComment,
|
|
500
|
+
default: value,
|
|
501
|
+
};
|
|
502
|
+
}
|
|
503
|
+
|
|
504
|
+
console.warn(
|
|
505
|
+
`Unrecognized value type for: ${String(value)}, using 'unknown'`,
|
|
506
|
+
);
|
|
507
|
+
return { type: "unknown", optional: true, description: yamlComment };
|
|
508
|
+
}
|
|
509
|
+
|
|
510
|
+
function convertValueToProperty(opts: InferencePropertyContext): TypeProperty {
|
|
511
|
+
const { value, nestedTypeName, schema, yamlComment, yamlComments, fullKey } =
|
|
512
|
+
opts;
|
|
513
|
+
|
|
514
|
+
if (schema) {
|
|
515
|
+
return convertWithSchema({ ...opts, schema });
|
|
516
|
+
}
|
|
517
|
+
|
|
518
|
+
if (
|
|
519
|
+
NullSchema.safeParse(value).success ||
|
|
520
|
+
UndefinedSchema.safeParse(value).success
|
|
521
|
+
) {
|
|
522
|
+
return { type: "unknown", optional: true };
|
|
523
|
+
}
|
|
524
|
+
|
|
525
|
+
const arrayResult = ArraySchema.safeParse(value);
|
|
526
|
+
if (arrayResult.success) {
|
|
527
|
+
return inferArrayType(nestedTypeName, arrayResult.data);
|
|
528
|
+
}
|
|
529
|
+
|
|
530
|
+
const objectResult = HelmValueSchema.safeParse(value);
|
|
531
|
+
if (objectResult.success) {
|
|
532
|
+
const nestedInterface = convertToTypeScriptInterface({
|
|
533
|
+
values: objectResult.data,
|
|
534
|
+
interfaceName: nestedTypeName,
|
|
535
|
+
yamlComments,
|
|
536
|
+
keyPrefix: fullKey,
|
|
537
|
+
});
|
|
538
|
+
return {
|
|
539
|
+
type: nestedTypeName,
|
|
540
|
+
optional: true,
|
|
541
|
+
nested: nestedInterface,
|
|
542
|
+
description: yamlComment,
|
|
543
|
+
default: value,
|
|
544
|
+
};
|
|
545
|
+
}
|
|
546
|
+
|
|
547
|
+
return inferPrimitiveType(value, yamlComment);
|
|
548
|
+
}
|
package/src/types.ts
ADDED
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
// Core type definitions for Helm chart type generation
|
|
2
|
+
|
|
3
|
+
export type ChartInfo = {
|
|
4
|
+
name: string;
|
|
5
|
+
repoUrl: string;
|
|
6
|
+
version: string;
|
|
7
|
+
chartName: string; // The actual chart name (may differ from versions.ts key)
|
|
8
|
+
};
|
|
9
|
+
|
|
10
|
+
export type JSONSchemaProperty = {
|
|
11
|
+
type?: string | string[];
|
|
12
|
+
enum?: unknown[];
|
|
13
|
+
oneOf?: JSONSchemaProperty[];
|
|
14
|
+
anyOf?: JSONSchemaProperty[];
|
|
15
|
+
items?: JSONSchemaProperty;
|
|
16
|
+
properties?: Record<string, JSONSchemaProperty>;
|
|
17
|
+
required?: string[];
|
|
18
|
+
description?: string;
|
|
19
|
+
default?: unknown;
|
|
20
|
+
};
|
|
21
|
+
|
|
22
|
+
export type TypeScriptInterface = {
|
|
23
|
+
name: string;
|
|
24
|
+
properties: Record<string, TypeProperty>;
|
|
25
|
+
/**
|
|
26
|
+
* If true, add an index signature to allow arbitrary additional properties
|
|
27
|
+
* Useful for config maps, arbitrary key-value stores, etc.
|
|
28
|
+
*/
|
|
29
|
+
allowArbitraryProps?: boolean;
|
|
30
|
+
};
|
|
31
|
+
|
|
32
|
+
export type TypeProperty = {
|
|
33
|
+
type: string;
|
|
34
|
+
optional: boolean;
|
|
35
|
+
description?: string;
|
|
36
|
+
default?: unknown;
|
|
37
|
+
nested?: TypeScriptInterface;
|
|
38
|
+
};
|
package/src/utils.ts
ADDED
|
@@ -0,0 +1,76 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Utility functions for Helm type generation
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
/**
|
|
6
|
+
* Sanitize property names for TypeScript interfaces
|
|
7
|
+
* Handles special characters, reserved keywords, and invalid syntax
|
|
8
|
+
*/
|
|
9
|
+
export function sanitizePropertyName(key: string): string {
|
|
10
|
+
// TypeScript reserved keywords that need quoting
|
|
11
|
+
const reservedKeywords = new Set([
|
|
12
|
+
"break",
|
|
13
|
+
"case",
|
|
14
|
+
"catch",
|
|
15
|
+
"class",
|
|
16
|
+
"const",
|
|
17
|
+
"continue",
|
|
18
|
+
"debugger",
|
|
19
|
+
"default",
|
|
20
|
+
"delete",
|
|
21
|
+
"do",
|
|
22
|
+
"else",
|
|
23
|
+
"enum",
|
|
24
|
+
"export",
|
|
25
|
+
"extends",
|
|
26
|
+
"false",
|
|
27
|
+
"finally",
|
|
28
|
+
"for",
|
|
29
|
+
"function",
|
|
30
|
+
"if",
|
|
31
|
+
"import",
|
|
32
|
+
"in",
|
|
33
|
+
"instanceof",
|
|
34
|
+
"new",
|
|
35
|
+
"null",
|
|
36
|
+
"return",
|
|
37
|
+
"super",
|
|
38
|
+
"switch",
|
|
39
|
+
"this",
|
|
40
|
+
"throw",
|
|
41
|
+
"true",
|
|
42
|
+
"try",
|
|
43
|
+
"typeof",
|
|
44
|
+
"var",
|
|
45
|
+
"void",
|
|
46
|
+
"while",
|
|
47
|
+
"with",
|
|
48
|
+
]);
|
|
49
|
+
|
|
50
|
+
// Check if key needs quoting
|
|
51
|
+
const needsQuoting =
|
|
52
|
+
reservedKeywords.has(key) ||
|
|
53
|
+
/[^\w$]/.test(key) || // Contains special characters
|
|
54
|
+
/^\d/.test(key); // Starts with digit
|
|
55
|
+
|
|
56
|
+
return needsQuoting ? `"${key}"` : key;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
/**
|
|
60
|
+
* Sanitize type names by removing invalid characters and normalizing
|
|
61
|
+
*/
|
|
62
|
+
export function sanitizeTypeName(key: string): string {
|
|
63
|
+
return (
|
|
64
|
+
key
|
|
65
|
+
.replaceAll(/[^a-z0-9]/gi, "") // Remove all special characters
|
|
66
|
+
.replace(/^\d+/, "") || // Remove leading digits
|
|
67
|
+
"Property"
|
|
68
|
+
); // Fallback if empty
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
/**
|
|
72
|
+
* Capitalize the first letter of a string
|
|
73
|
+
*/
|
|
74
|
+
export function capitalizeFirst(str: string): string {
|
|
75
|
+
return str.charAt(0).toUpperCase() + str.slice(1);
|
|
76
|
+
}
|