@nikovirtala/projen-constructs 0.1.5 → 0.1.6
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/.jsii +54 -34
- package/API.md +52 -28
- package/lib/components/mise.js +4 -2
- package/lib/components/vitest.js +1 -1
- package/lib/config.d.ts +20 -7
- package/lib/config.js +28 -16
- package/lib/errors.d.ts +20 -0
- package/lib/errors.js +36 -0
- package/lib/projects/awscdk-construct-library-options.generated.d.ts +7 -7
- package/lib/projects/awscdk-construct-library-options.generated.js +1 -1
- package/lib/projects/awscdk-construct-library.generated.d.ts +8 -0
- package/lib/projects/awscdk-construct-library.generated.js +17 -4
- package/lib/projects/awscdk-typescript-app-options.generated.d.ts +7 -7
- package/lib/projects/awscdk-typescript-app-options.generated.js +1 -1
- package/lib/projects/awscdk-typescript-app.generated.d.ts +8 -0
- package/lib/projects/awscdk-typescript-app.generated.js +17 -4
- package/lib/projects/jsii-options.generated.d.ts +7 -7
- package/lib/projects/jsii-options.generated.js +1 -1
- package/lib/projects/jsii.generated.d.ts +8 -0
- package/lib/projects/jsii.generated.js +17 -4
- package/lib/projects/typescript-options.generated.d.ts +7 -7
- package/lib/projects/typescript-options.generated.js +1 -1
- package/lib/projects/typescript.generated.d.ts +8 -0
- package/lib/projects/typescript.generated.js +17 -4
- package/lib/projen-project-generator.d.ts +173 -0
- package/lib/projen-project-generator.js +509 -0
- package/package.json +1 -1
- package/lib/projen-project-class.d.ts +0 -44
- package/lib/projen-project-class.js +0 -137
|
@@ -0,0 +1,509 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.ProjenProjectGenerator = void 0;
|
|
4
|
+
const jsii_struct_builder_1 = require("@mrgrain/jsii-struct-builder");
|
|
5
|
+
const projen_1 = require("projen");
|
|
6
|
+
const errors_1 = require("./errors");
|
|
7
|
+
/**
|
|
8
|
+
* Default components applied to all generated projects
|
|
9
|
+
*/
|
|
10
|
+
const DEFAULT_COMPONENTS = [
|
|
11
|
+
{ component: require("./components/mise").Mise },
|
|
12
|
+
{
|
|
13
|
+
component: require("./components/vitest").Vitest,
|
|
14
|
+
optionsProperty: {
|
|
15
|
+
name: "vitestOptions",
|
|
16
|
+
type: "@nikovirtala/projen-constructs.VitestOptions",
|
|
17
|
+
docs: "Vitest configuration",
|
|
18
|
+
},
|
|
19
|
+
},
|
|
20
|
+
];
|
|
21
|
+
/**
|
|
22
|
+
* Buffer for building indented code with automatic line management
|
|
23
|
+
*
|
|
24
|
+
* Maintains indentation level and provides methods for adding lines,
|
|
25
|
+
* opening/closing blocks, and flushing accumulated content.
|
|
26
|
+
*/
|
|
27
|
+
class CodeBuffer {
|
|
28
|
+
/**
|
|
29
|
+
* @param indent - String to use for each indentation level (default: single space)
|
|
30
|
+
*/
|
|
31
|
+
constructor(indent = " ") {
|
|
32
|
+
this.indent = indent;
|
|
33
|
+
this.lines = [];
|
|
34
|
+
this.indentLevel = 0;
|
|
35
|
+
}
|
|
36
|
+
/**
|
|
37
|
+
* Returns all accumulated lines and resets the buffer
|
|
38
|
+
*
|
|
39
|
+
* @returns Array of code lines with proper indentation
|
|
40
|
+
*/
|
|
41
|
+
flush() {
|
|
42
|
+
const current = this.lines;
|
|
43
|
+
this.reset();
|
|
44
|
+
return current;
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* Adds a line of code at the current indentation level
|
|
48
|
+
*
|
|
49
|
+
* @param code - Code to add (optional, adds blank line if omitted)
|
|
50
|
+
*/
|
|
51
|
+
line(code) {
|
|
52
|
+
const prefix = this.indent.repeat(this.indentLevel);
|
|
53
|
+
this.lines.push((prefix + (code ?? "")).trimEnd());
|
|
54
|
+
}
|
|
55
|
+
/**
|
|
56
|
+
* Opens a new indentation block
|
|
57
|
+
*
|
|
58
|
+
* @param code - Optional code to add before increasing indent (e.g., opening brace)
|
|
59
|
+
*/
|
|
60
|
+
open(code) {
|
|
61
|
+
if (code) {
|
|
62
|
+
this.line(code);
|
|
63
|
+
}
|
|
64
|
+
this.indentLevel++;
|
|
65
|
+
}
|
|
66
|
+
/**
|
|
67
|
+
* Closes the current indentation block
|
|
68
|
+
*
|
|
69
|
+
* @param code - Optional code to add after decreasing indent (e.g., closing brace)
|
|
70
|
+
* @throws {InvalidIndentLevelError} When attempting to decrease indent below zero
|
|
71
|
+
*/
|
|
72
|
+
close(code) {
|
|
73
|
+
if (this.indentLevel === 0) {
|
|
74
|
+
throw new errors_1.InvalidIndentLevelError();
|
|
75
|
+
}
|
|
76
|
+
this.indentLevel--;
|
|
77
|
+
if (code) {
|
|
78
|
+
this.line(code);
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
/**
|
|
82
|
+
* Resets the buffer to initial state
|
|
83
|
+
*/
|
|
84
|
+
reset() {
|
|
85
|
+
this.lines = [];
|
|
86
|
+
this.indentLevel = 0;
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
/**
|
|
90
|
+
* Renders TypeScript class code from ProjenProjectGenerator options
|
|
91
|
+
*
|
|
92
|
+
* Generates a complete TypeScript class file including imports, options interface export,
|
|
93
|
+
* class declaration, and constructor with component integration.
|
|
94
|
+
*/
|
|
95
|
+
class TypeScriptClassRenderer {
|
|
96
|
+
/**
|
|
97
|
+
* @param indent - Number of spaces per indentation level (default: 4)
|
|
98
|
+
*/
|
|
99
|
+
constructor(indent = 4) {
|
|
100
|
+
this.buffer = new CodeBuffer(" ".repeat(indent));
|
|
101
|
+
}
|
|
102
|
+
/**
|
|
103
|
+
* Renders complete TypeScript class code from options
|
|
104
|
+
*
|
|
105
|
+
* Orchestrates the rendering process: derives names, extracts component configuration,
|
|
106
|
+
* builds import statements, and renders the class structure.
|
|
107
|
+
*
|
|
108
|
+
* @param options - Generator configuration
|
|
109
|
+
* @returns Complete TypeScript class code as a string
|
|
110
|
+
*/
|
|
111
|
+
render(options) {
|
|
112
|
+
this.buffer.flush();
|
|
113
|
+
/* Derive interface and type names from the class name */
|
|
114
|
+
const optionsInterface = `${options.name}Options`;
|
|
115
|
+
const baseOptionsFqn = this.getBaseOptionsFqn(options.baseClass);
|
|
116
|
+
const baseOptionsType = baseOptionsFqn.replace(/^projen\./, "");
|
|
117
|
+
/* Use provided components or fall back to defaults (Mise + Vitest) */
|
|
118
|
+
const components = options.components ?? DEFAULT_COMPONENTS;
|
|
119
|
+
/* Extract component configuration for constructor generation */
|
|
120
|
+
const { destructure, componentArray } = this.extractComponentOptions(components);
|
|
121
|
+
const imports = this.extractImports(options, optionsInterface);
|
|
122
|
+
/* Render all sections of the class file */
|
|
123
|
+
this.renderImports(imports);
|
|
124
|
+
this.buffer.line();
|
|
125
|
+
this.renderExport(optionsInterface);
|
|
126
|
+
this.buffer.line();
|
|
127
|
+
this.renderClass(options, optionsInterface, baseOptionsType, destructure, componentArray);
|
|
128
|
+
this.buffer.line();
|
|
129
|
+
return this.buffer.flush().join("\n");
|
|
130
|
+
}
|
|
131
|
+
/**
|
|
132
|
+
* Derives the fully qualified options interface name from base class
|
|
133
|
+
*
|
|
134
|
+
* Transforms base class reference into the corresponding Projen options interface name
|
|
135
|
+
* by appending "Options" suffix and prefixing with "projen." namespace.
|
|
136
|
+
*
|
|
137
|
+
* @param baseClass - Base class in format "module.ClassName"
|
|
138
|
+
* @returns Fully qualified options interface name
|
|
139
|
+
* @throws {InvalidBaseClassFormatError} When baseClass format is invalid
|
|
140
|
+
*
|
|
141
|
+
* @example
|
|
142
|
+
* getBaseOptionsFqn("typescript.TypeScriptProject") // "projen.typescript.TypeScriptProjectOptions"
|
|
143
|
+
* getBaseOptionsFqn("cdk.JsiiProject") // "projen.cdk.JsiiProjectOptions"
|
|
144
|
+
*/
|
|
145
|
+
getBaseOptionsFqn(baseClass) {
|
|
146
|
+
/* Validate base class format: must contain module.ClassName structure */
|
|
147
|
+
if (!baseClass.includes(".")) {
|
|
148
|
+
throw new errors_1.InvalidBaseClassFormatError(baseClass);
|
|
149
|
+
}
|
|
150
|
+
return `projen.${baseClass}Options`;
|
|
151
|
+
}
|
|
152
|
+
/**
|
|
153
|
+
* Extracts component configuration for constructor code generation
|
|
154
|
+
*
|
|
155
|
+
* Transforms component configurations into:
|
|
156
|
+
* 1. Destructure list: variable names to extract from options
|
|
157
|
+
* 2. Component array: code string for applyDefaults() call
|
|
158
|
+
*
|
|
159
|
+
* @param components - Component configurations
|
|
160
|
+
* @returns Object with destructure array and componentArray code string
|
|
161
|
+
*
|
|
162
|
+
* @example
|
|
163
|
+
* Input: [{ component: Vitest, optionsProperty: { name: "vitestOptions", ... } }]
|
|
164
|
+
* Output: {
|
|
165
|
+
* destructure: ["vitest", "vitestOptions"],
|
|
166
|
+
* componentArray: "[{ component: Vitest, enabled: vitest, options: vitestOptions }]"
|
|
167
|
+
* }
|
|
168
|
+
*/
|
|
169
|
+
extractComponentOptions(components) {
|
|
170
|
+
const destructure = [];
|
|
171
|
+
const componentParts = [];
|
|
172
|
+
for (const c of components) {
|
|
173
|
+
/* Convert component class name to camelCase variable name (e.g., Vitest -> vitest) */
|
|
174
|
+
const name = c.component.name.charAt(0).toLowerCase() + c.component.name.slice(1);
|
|
175
|
+
destructure.push(name);
|
|
176
|
+
/* Build component config object for applyDefaults() call */
|
|
177
|
+
const parts = [`component: ${c.component.name}`];
|
|
178
|
+
if (c.optionsProperty) {
|
|
179
|
+
/* Component has configurable options - include both enabled flag and options */
|
|
180
|
+
destructure.push(c.optionsProperty.name);
|
|
181
|
+
parts.push(`enabled: ${name}`, `options: ${c.optionsProperty.name}`);
|
|
182
|
+
}
|
|
183
|
+
else {
|
|
184
|
+
/* Component has no options - only include enabled flag */
|
|
185
|
+
parts.push(`enabled: ${name}`);
|
|
186
|
+
}
|
|
187
|
+
componentParts.push(`{ ${parts.join(", ")} }`);
|
|
188
|
+
}
|
|
189
|
+
return { destructure, componentArray: `[${componentParts.join(", ")}]` };
|
|
190
|
+
}
|
|
191
|
+
/**
|
|
192
|
+
* Builds import statements map for the generated class
|
|
193
|
+
*
|
|
194
|
+
* Collects all required imports:
|
|
195
|
+
* - Projen base module (typescript, cdk, awscdk)
|
|
196
|
+
* - Config utilities (applyDefaults, defaultOptions)
|
|
197
|
+
* - Component classes
|
|
198
|
+
* - Utility functions (deepMerge)
|
|
199
|
+
* - Generated options interface (type-only import)
|
|
200
|
+
*
|
|
201
|
+
* @param options - Generator configuration
|
|
202
|
+
* @param optionsInterface - Name of the generated options interface
|
|
203
|
+
* @returns Map of module paths to sets of imported names
|
|
204
|
+
*/
|
|
205
|
+
extractImports(options, optionsInterface) {
|
|
206
|
+
const imports = new Map();
|
|
207
|
+
/* Extract base module name from baseClass (e.g., "typescript" from "typescript.TypeScriptProject") */
|
|
208
|
+
const baseModule = options.baseClass.split(".")[0];
|
|
209
|
+
const optionsFileName = this.getOptionsFileName(optionsInterface);
|
|
210
|
+
/* Projen base module import */
|
|
211
|
+
imports.set("projen", new Set([baseModule]));
|
|
212
|
+
/* Configuration utilities import */
|
|
213
|
+
imports.set("../config", new Set(["applyDefaults", "defaultOptions"]));
|
|
214
|
+
/* Component class imports - derive module path from component class name */
|
|
215
|
+
const components = options.components ?? DEFAULT_COMPONENTS;
|
|
216
|
+
for (const c of components) {
|
|
217
|
+
const componentName = c.component.name;
|
|
218
|
+
const modulePath = `../components/${componentName.toLowerCase()}`;
|
|
219
|
+
if (!imports.has(modulePath)) {
|
|
220
|
+
imports.set(modulePath, new Set());
|
|
221
|
+
}
|
|
222
|
+
const moduleImports = imports.get(modulePath);
|
|
223
|
+
if (moduleImports) {
|
|
224
|
+
moduleImports.add(componentName);
|
|
225
|
+
}
|
|
226
|
+
}
|
|
227
|
+
/* Utility functions import */
|
|
228
|
+
imports.set("../utils", new Set(["deepMerge"]));
|
|
229
|
+
/* Generated options interface import (type-only to avoid circular dependencies) */
|
|
230
|
+
imports.set(`./${optionsFileName}.generated`, new Set([optionsInterface]));
|
|
231
|
+
return imports;
|
|
232
|
+
}
|
|
233
|
+
/**
|
|
234
|
+
* Derives the config path for accessing default options
|
|
235
|
+
*
|
|
236
|
+
* Transforms base class reference into the corresponding path in the defaultOptions
|
|
237
|
+
* configuration object exported from config.ts.
|
|
238
|
+
*
|
|
239
|
+
* @param baseClass - Base class in format "module.ClassName"
|
|
240
|
+
* @returns Config path for accessing default options
|
|
241
|
+
*
|
|
242
|
+
* @example
|
|
243
|
+
* getConfigPath("typescript.TypeScriptProject") // "defaultOptions.typescript.TypeScriptProject"
|
|
244
|
+
*/
|
|
245
|
+
getConfigPath(baseClass) {
|
|
246
|
+
return `defaultOptions.${baseClass}`;
|
|
247
|
+
}
|
|
248
|
+
/**
|
|
249
|
+
* Renders import statements in sorted order
|
|
250
|
+
*
|
|
251
|
+
* Sorts imports with external packages first, then relative imports.
|
|
252
|
+
* Uses type-only imports for generated files to avoid circular dependencies.
|
|
253
|
+
*
|
|
254
|
+
* @param imports - Map of module paths to imported names
|
|
255
|
+
*/
|
|
256
|
+
renderImports(imports) {
|
|
257
|
+
/* Sort modules: external packages first, then relative imports alphabetically */
|
|
258
|
+
const sortedModules = Array.from(imports.keys()).sort((a, b) => {
|
|
259
|
+
const aIsRelative = a.startsWith(".");
|
|
260
|
+
const bIsRelative = b.startsWith(".");
|
|
261
|
+
if (aIsRelative !== bIsRelative) {
|
|
262
|
+
return aIsRelative ? 1 : -1;
|
|
263
|
+
}
|
|
264
|
+
return a.localeCompare(b);
|
|
265
|
+
});
|
|
266
|
+
for (const mod of sortedModules) {
|
|
267
|
+
const names = Array.from(imports.get(mod) || []).sort();
|
|
268
|
+
/* Use type-only import for generated files to prevent circular dependencies */
|
|
269
|
+
const isTypeOnly = mod.includes(".generated");
|
|
270
|
+
const importStmt = isTypeOnly ? "import type" : "import";
|
|
271
|
+
this.buffer.line(`${importStmt} { ${names.join(", ")} } from "${mod}";`);
|
|
272
|
+
}
|
|
273
|
+
}
|
|
274
|
+
/**
|
|
275
|
+
* Renders re-export statement for the generated options interface
|
|
276
|
+
*
|
|
277
|
+
* @param optionsInterface - Name of the options interface to export
|
|
278
|
+
*/
|
|
279
|
+
renderExport(optionsInterface) {
|
|
280
|
+
const optionsFileName = this.getOptionsFileName(optionsInterface);
|
|
281
|
+
this.buffer.line();
|
|
282
|
+
this.buffer.line(`export { ${optionsInterface} } from "./${optionsFileName}.generated";`);
|
|
283
|
+
}
|
|
284
|
+
/**
|
|
285
|
+
* Renders the class declaration and constructor
|
|
286
|
+
*
|
|
287
|
+
* @param options - Generator configuration
|
|
288
|
+
* @param optionsInterface - Name of the options interface
|
|
289
|
+
* @param baseOptionsType - Base options type without "projen." prefix
|
|
290
|
+
* @param destructure - Variable names to destructure from options
|
|
291
|
+
* @param componentArray - Code string for component array
|
|
292
|
+
*/
|
|
293
|
+
renderClass(options, optionsInterface, baseOptionsType, destructure, componentArray) {
|
|
294
|
+
this.buffer.line();
|
|
295
|
+
this.buffer.line("/**");
|
|
296
|
+
this.buffer.line(` * ${options.name} with standard configuration and component integration`);
|
|
297
|
+
this.buffer.line(" *");
|
|
298
|
+
this.buffer.line(" * Extends Projen's base class with opinionated defaults and automatic component setup.");
|
|
299
|
+
this.buffer.line(" */");
|
|
300
|
+
this.buffer.open(`export class ${options.name} extends ${options.baseClass} {`);
|
|
301
|
+
this.renderConstructor(options, optionsInterface, baseOptionsType, destructure, componentArray);
|
|
302
|
+
this.buffer.close("}");
|
|
303
|
+
}
|
|
304
|
+
/**
|
|
305
|
+
* Renders the constructor implementation
|
|
306
|
+
*
|
|
307
|
+
* The constructor:
|
|
308
|
+
* 1. Destructures component flags and options from the options parameter
|
|
309
|
+
* 2. Merges default configuration with user-provided options
|
|
310
|
+
* 3. Calls super() with merged options
|
|
311
|
+
* 4. Applies component defaults via applyDefaults()
|
|
312
|
+
*
|
|
313
|
+
* @param options - Generator configuration
|
|
314
|
+
* @param optionsInterface - Name of the options interface
|
|
315
|
+
* @param baseOptionsType - Base options type without "projen." prefix
|
|
316
|
+
* @param destructure - Variable names to destructure from options
|
|
317
|
+
* @param componentArray - Code string for component array
|
|
318
|
+
*/
|
|
319
|
+
renderConstructor(options, optionsInterface, baseOptionsType, destructure, componentArray) {
|
|
320
|
+
const configPath = this.getConfigPath(options.baseClass);
|
|
321
|
+
this.buffer.line("/**");
|
|
322
|
+
this.buffer.line(" * @param options - Project configuration");
|
|
323
|
+
this.buffer.line(" */");
|
|
324
|
+
this.buffer.open(`constructor(options: ${optionsInterface}) {`);
|
|
325
|
+
this.buffer.line("/* Separate component configuration from base Projen options */");
|
|
326
|
+
this.buffer.line(`const { ${destructure.join(", ")}, ...baseOptions } = options;`);
|
|
327
|
+
this.buffer.line();
|
|
328
|
+
this.buffer.line("/* Merge default configuration with user options and initialize base class */");
|
|
329
|
+
this.buffer.line(`super(deepMerge<${baseOptionsType}>(${configPath}, baseOptions));`);
|
|
330
|
+
this.buffer.line();
|
|
331
|
+
this.buffer.line("/* Apply component defaults and instantiate enabled components */");
|
|
332
|
+
this.buffer.line(`applyDefaults(this, ${componentArray});`);
|
|
333
|
+
this.buffer.close("}");
|
|
334
|
+
}
|
|
335
|
+
/**
|
|
336
|
+
* Maps options interface name to file name
|
|
337
|
+
*
|
|
338
|
+
* Converts PascalCase interface names to kebab-case file names.
|
|
339
|
+
* Falls back to lowercase conversion for unmapped interfaces.
|
|
340
|
+
*
|
|
341
|
+
* @param optionsInterface - Options interface name
|
|
342
|
+
* @returns Kebab-case file name without extension
|
|
343
|
+
*
|
|
344
|
+
* @example
|
|
345
|
+
* getOptionsFileName("TypeScriptProjectOptions") // "typescript-options"
|
|
346
|
+
*/
|
|
347
|
+
getOptionsFileName(optionsInterface) {
|
|
348
|
+
const mapping = {
|
|
349
|
+
TypeScriptProjectOptions: "typescript-options",
|
|
350
|
+
JsiiProjectOptions: "jsii-options",
|
|
351
|
+
AwsCdkTypeScriptAppProjectOptions: "awscdk-typescript-app-options",
|
|
352
|
+
AwsCdkConstructLibraryProjectOptions: "awscdk-construct-library-options",
|
|
353
|
+
};
|
|
354
|
+
return mapping[optionsInterface] || optionsInterface.toLowerCase();
|
|
355
|
+
}
|
|
356
|
+
}
|
|
357
|
+
/**
|
|
358
|
+
* Projen component that generates TypeScript project classes with standard configuration
|
|
359
|
+
*
|
|
360
|
+
* This component automates the creation of project classes that extend Projen base classes
|
|
361
|
+
* with opinionated defaults and component integration. It generates both:
|
|
362
|
+
* 1. An options interface (via ProjenStruct) that extends the base Projen options
|
|
363
|
+
* 2. A project class that applies default configuration and instantiates components
|
|
364
|
+
*
|
|
365
|
+
* The generated code follows a consistent pattern:
|
|
366
|
+
* - Imports required modules and components
|
|
367
|
+
* - Re-exports the generated options interface
|
|
368
|
+
* - Defines a class extending the Projen base class
|
|
369
|
+
* - Constructor merges defaults with user options and applies components
|
|
370
|
+
*
|
|
371
|
+
* @example
|
|
372
|
+
* ```typescript
|
|
373
|
+
* new ProjenProjectGenerator(project, {
|
|
374
|
+
* name: "TypeScriptProject",
|
|
375
|
+
* baseClass: "typescript.TypeScriptProject",
|
|
376
|
+
* filePath: "src/projects/typescript.generated.ts",
|
|
377
|
+
* components: [{ component: Vitest, optionsProperty: "vitestOptions" }]
|
|
378
|
+
* });
|
|
379
|
+
* ```
|
|
380
|
+
*/
|
|
381
|
+
class ProjenProjectGenerator extends projen_1.Component {
|
|
382
|
+
/**
|
|
383
|
+
* @param project - Projen project instance
|
|
384
|
+
* @param options - Generator configuration
|
|
385
|
+
*/
|
|
386
|
+
constructor(project, options) {
|
|
387
|
+
super(project);
|
|
388
|
+
this.options = options;
|
|
389
|
+
this.renderer = new TypeScriptClassRenderer();
|
|
390
|
+
/* Generate the options interface using ProjenStruct for JSII compatibility */
|
|
391
|
+
const optionsInterface = `${options.name}Options`;
|
|
392
|
+
const baseOptionsFqn = this.renderer.getBaseOptionsFqn(options.baseClass);
|
|
393
|
+
const optionsFilePath = this.getOptionsFilePath(optionsInterface);
|
|
394
|
+
/* ProjenStruct generates a concrete TypeScript interface from Projen's options
|
|
395
|
+
* This is necessary because JSII doesn't support TypeScript utility types like Omit<>
|
|
396
|
+
* The generated interface can be consumed by other languages (Python, Java, C#, Go) */
|
|
397
|
+
const struct = new jsii_struct_builder_1.ProjenStruct(this.asTypeScriptProject(project), {
|
|
398
|
+
name: optionsInterface,
|
|
399
|
+
filePath: optionsFilePath,
|
|
400
|
+
outputFileOptions: { readonly: true },
|
|
401
|
+
})
|
|
402
|
+
.mixin(jsii_struct_builder_1.Struct.fromFqn(baseOptionsFqn))
|
|
403
|
+
.withoutDeprecated();
|
|
404
|
+
/* Remove specified options from the base interface */
|
|
405
|
+
if (options.omitOptions) {
|
|
406
|
+
struct.omit(...options.omitOptions);
|
|
407
|
+
}
|
|
408
|
+
/* Add component-derived options to the interface */
|
|
409
|
+
const components = options.components ?? DEFAULT_COMPONENTS;
|
|
410
|
+
const { PrimitiveType } = require("@jsii/spec");
|
|
411
|
+
for (const c of components) {
|
|
412
|
+
const name = c.component.name.charAt(0).toLowerCase() + c.component.name.slice(1);
|
|
413
|
+
/* Add enabled flag for the component */
|
|
414
|
+
struct.add({
|
|
415
|
+
name,
|
|
416
|
+
type: { primitive: PrimitiveType.Boolean },
|
|
417
|
+
optional: true,
|
|
418
|
+
docs: {
|
|
419
|
+
summary: `Enable ${c.component.name} component`,
|
|
420
|
+
default: "true",
|
|
421
|
+
},
|
|
422
|
+
});
|
|
423
|
+
/* Add options property if component is configurable */
|
|
424
|
+
if (c.optionsProperty) {
|
|
425
|
+
struct.add({
|
|
426
|
+
name: c.optionsProperty.name,
|
|
427
|
+
type: { fqn: c.optionsProperty.type },
|
|
428
|
+
optional: true,
|
|
429
|
+
docs: {
|
|
430
|
+
summary: c.optionsProperty.docs ?? `${c.component.name} configuration`,
|
|
431
|
+
default: `- default ${c.component.name} configuration`,
|
|
432
|
+
},
|
|
433
|
+
});
|
|
434
|
+
}
|
|
435
|
+
}
|
|
436
|
+
/* Add custom options to the interface */
|
|
437
|
+
if (options.additionalOptions) {
|
|
438
|
+
struct.add(...options.additionalOptions);
|
|
439
|
+
}
|
|
440
|
+
}
|
|
441
|
+
/**
|
|
442
|
+
* Casts project to TypeScript project type
|
|
443
|
+
*
|
|
444
|
+
* ProjenStruct requires a TypeScriptProject instance. This generator is only used
|
|
445
|
+
* within TypeScript projects, so the cast is safe in practice.
|
|
446
|
+
*
|
|
447
|
+
* Note: Type assertion is necessary here because Projen's type system doesn't provide
|
|
448
|
+
* a runtime type guard for TypeScriptProject.
|
|
449
|
+
*
|
|
450
|
+
* @param project - Projen project instance
|
|
451
|
+
* @returns TypeScript project instance
|
|
452
|
+
*/
|
|
453
|
+
asTypeScriptProject(project) {
|
|
454
|
+
return project;
|
|
455
|
+
}
|
|
456
|
+
/**
|
|
457
|
+
* Derives the file path for the generated options interface
|
|
458
|
+
*
|
|
459
|
+
* Places the options interface file in the same directory as the class file
|
|
460
|
+
* with a ".generated.ts" suffix.
|
|
461
|
+
*
|
|
462
|
+
* @param optionsInterface - Name of the options interface
|
|
463
|
+
* @returns File path for the options interface
|
|
464
|
+
* @throws {InvalidFilePathError} When filePath doesn't contain a directory separator
|
|
465
|
+
*/
|
|
466
|
+
getOptionsFilePath(optionsInterface) {
|
|
467
|
+
const lastSlash = this.options.filePath.lastIndexOf("/");
|
|
468
|
+
if (lastSlash === -1) {
|
|
469
|
+
throw new errors_1.InvalidFilePathError(this.options.filePath);
|
|
470
|
+
}
|
|
471
|
+
const dir = this.options.filePath.substring(0, lastSlash);
|
|
472
|
+
const optionsFileName = this.renderer.getOptionsFileName(optionsInterface);
|
|
473
|
+
return `${dir}/${optionsFileName}.generated.ts`;
|
|
474
|
+
}
|
|
475
|
+
/**
|
|
476
|
+
* Generates the TypeScript class file during Projen synthesis
|
|
477
|
+
*
|
|
478
|
+
* Called by Projen during the synthesis phase to generate the project class file.
|
|
479
|
+
* The file is marked as readonly to prevent manual editing.
|
|
480
|
+
*/
|
|
481
|
+
preSynthesize() {
|
|
482
|
+
const content = this.renderer.render(this.options);
|
|
483
|
+
new TypeScriptClassFile(this.project, this.options.filePath, content, {
|
|
484
|
+
readonly: this.options.readonly ?? true,
|
|
485
|
+
});
|
|
486
|
+
}
|
|
487
|
+
}
|
|
488
|
+
exports.ProjenProjectGenerator = ProjenProjectGenerator;
|
|
489
|
+
/**
|
|
490
|
+
* Text file for generated TypeScript class code
|
|
491
|
+
*
|
|
492
|
+
* Extends Projen's TextFile to add the generated file marker comment
|
|
493
|
+
* at the beginning of the file.
|
|
494
|
+
*/
|
|
495
|
+
class TypeScriptClassFile extends projen_1.TextFile {
|
|
496
|
+
/**
|
|
497
|
+
* @param project - Projen project instance
|
|
498
|
+
* @param filePath - Output file path
|
|
499
|
+
* @param content - TypeScript class code
|
|
500
|
+
* @param options - Source code options (readonly, etc.)
|
|
501
|
+
*/
|
|
502
|
+
constructor(project, filePath, content, options = {}) {
|
|
503
|
+
super(project, filePath, options);
|
|
504
|
+
/* Add generated file marker to prevent manual editing */
|
|
505
|
+
this.addLine(`// ${this.marker}`);
|
|
506
|
+
this.addLine(content);
|
|
507
|
+
}
|
|
508
|
+
}
|
|
509
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"projen-project-generator.js","sourceRoot":"","sources":["../src/projen-project-generator.ts"],"names":[],"mappings":"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","sourcesContent":["import type { Property } from \"@jsii/spec\";\nimport { ProjenStruct, Struct } from \"@mrgrain/jsii-struct-builder\";\nimport type { Project, SourceCodeOptions, typescript } from \"projen\";\nimport { Component, TextFile } from \"projen\";\nimport { InvalidBaseClassFormatError, InvalidFilePathError, InvalidIndentLevelError } from \"./errors\";\n\ntype ProjenModule = \"typescript\" | \"cdk\" | \"awscdk\";\ntype ProjenBaseClass<M extends ProjenModule> = `${M}.${string}`;\n\n/**\n * Default components applied to all generated projects\n */\nconst DEFAULT_COMPONENTS: ComponentConfig[] = [\n    { component: require(\"./components/mise\").Mise },\n    {\n        component: require(\"./components/vitest\").Vitest,\n        optionsProperty: {\n            name: \"vitestOptions\",\n            type: \"@nikovirtala/projen-constructs.VitestOptions\",\n            docs: \"Vitest configuration\",\n        },\n    },\n];\n\n/**\n * Configuration for a component to be integrated into a generated project\n *\n * @example\n * ```typescript\n * const config: ComponentConfig = {\n *   component: Vitest,\n *   optionsProperty: \"vitestOptions\"\n * };\n * ```\n */\ninterface ComponentConfig<T extends Component = Component> {\n    /**\n     * Component class constructor\n     */\n    readonly component: new (\n        project: never,\n        options?: never,\n    ) => T;\n\n    /**\n     * Options property configuration for the generated options interface\n     *\n     * When specified, adds an options property to the interface allowing\n     * users to configure the component.\n     */\n    readonly optionsProperty?: {\n        /**\n         * Name of the options property (e.g., \"vitestOptions\")\n         */\n        readonly name: string;\n        /**\n         * Fully qualified type name (e.g., \"@nikovirtala/projen-constructs.VitestOptions\")\n         */\n        readonly type: string;\n        /**\n         * Documentation summary\n         */\n        readonly docs?: string;\n    };\n}\n\n/**\n * Options for ProjenProjectGenerator component\n *\n * Configures the generation of a TypeScript project class that extends a Projen base class\n * with standard configuration and component integration.\n *\n * @example\n * ```typescript\n * new ProjenProjectGenerator(project, {\n *   name: \"TypeScriptProject\",\n *   baseClass: \"typescript.TypeScriptProject\",\n *   filePath: \"src/projects/typescript.generated.ts\",\n *   components: [{ component: Vitest, optionsProperty: \"vitestOptions\" }]\n * });\n * ```\n */\nexport interface ProjenProjectGeneratorOptions<M extends ProjenModule = ProjenModule> extends SourceCodeOptions {\n    /**\n     * Name of the generated class (e.g., \"TypeScriptProject\")\n     *\n     * The options interface will be named `${name}Options`.\n     */\n    readonly name: string;\n\n    /**\n     * Fully qualified base class to extend in format \"module.ClassName\"\n     *\n     * Must be a valid Projen class reference like \"typescript.TypeScriptProject\",\n     * \"cdk.JsiiProject\", or \"awscdk.AwsCdkTypeScriptApp\".\n     *\n     * @example \"typescript.TypeScriptProject\"\n     * @example \"cdk.JsiiProject\"\n     */\n    readonly baseClass: ProjenBaseClass<M>;\n\n    /**\n     * Output file path for the generated class\n     *\n     * Must contain a directory separator. The options interface will be generated\n     * in the same directory with a \".generated.ts\" suffix.\n     *\n     * @example \"src/projects/typescript.generated.ts\"\n     */\n    readonly filePath: string;\n\n    /**\n     * Components to integrate into the project\n     *\n     * Each component will be instantiated during project construction and can be\n     * configured via an optional options property in the generated interface.\n     *\n     * @default [{ component: Mise }, { component: Vitest, optionsProperty: { name: \"vitestOptions\", type: \"...\", docs: \"...\" } }]\n     */\n    readonly components?: ComponentConfig[];\n\n    /**\n     * Additional properties to add to the generated options interface\n     *\n     * Use this to extend the base options with custom properties specific to\n     * your project type.\n     */\n    readonly additionalOptions?: Property[];\n\n    /**\n     * Property names to omit from the base options interface\n     *\n     * Use this to hide base class options that should not be configurable\n     * in the generated project type.\n     */\n    readonly omitOptions?: string[];\n}\n\n/**\n * Buffer for building indented code with automatic line management\n *\n * Maintains indentation level and provides methods for adding lines,\n * opening/closing blocks, and flushing accumulated content.\n */\nclass CodeBuffer {\n    private lines: string[] = [];\n    private indentLevel = 0;\n\n    /**\n     * @param indent - String to use for each indentation level (default: single space)\n     */\n    constructor(private readonly indent = \" \") {}\n\n    /**\n     * Returns all accumulated lines and resets the buffer\n     *\n     * @returns Array of code lines with proper indentation\n     */\n    flush(): string[] {\n        const current = this.lines;\n        this.reset();\n        return current;\n    }\n\n    /**\n     * Adds a line of code at the current indentation level\n     *\n     * @param code - Code to add (optional, adds blank line if omitted)\n     */\n    line(code?: string) {\n        const prefix = this.indent.repeat(this.indentLevel);\n        this.lines.push((prefix + (code ?? \"\")).trimEnd());\n    }\n\n    /**\n     * Opens a new indentation block\n     *\n     * @param code - Optional code to add before increasing indent (e.g., opening brace)\n     */\n    open(code?: string) {\n        if (code) {\n            this.line(code);\n        }\n        this.indentLevel++;\n    }\n\n    /**\n     * Closes the current indentation block\n     *\n     * @param code - Optional code to add after decreasing indent (e.g., closing brace)\n     * @throws {InvalidIndentLevelError} When attempting to decrease indent below zero\n     */\n    close(code?: string) {\n        if (this.indentLevel === 0) {\n            throw new InvalidIndentLevelError();\n        }\n        this.indentLevel--;\n        if (code) {\n            this.line(code);\n        }\n    }\n\n    /**\n     * Resets the buffer to initial state\n     */\n    private reset(): void {\n        this.lines = [];\n        this.indentLevel = 0;\n    }\n}\n\n/**\n * Renders TypeScript class code from ProjenProjectGenerator options\n *\n * Generates a complete TypeScript class file including imports, options interface export,\n * class declaration, and constructor with component integration.\n */\nclass TypeScriptClassRenderer {\n    private buffer: CodeBuffer;\n\n    /**\n     * @param indent - Number of spaces per indentation level (default: 4)\n     */\n    constructor(indent = 4) {\n        this.buffer = new CodeBuffer(\" \".repeat(indent));\n    }\n\n    /**\n     * Renders complete TypeScript class code from options\n     *\n     * Orchestrates the rendering process: derives names, extracts component configuration,\n     * builds import statements, and renders the class structure.\n     *\n     * @param options - Generator configuration\n     * @returns Complete TypeScript class code as a string\n     */\n    render(options: ProjenProjectGeneratorOptions<ProjenModule>): string {\n        this.buffer.flush();\n\n        /* Derive interface and type names from the class name */\n        const optionsInterface = `${options.name}Options`;\n        const baseOptionsFqn = this.getBaseOptionsFqn(options.baseClass);\n        const baseOptionsType = baseOptionsFqn.replace(/^projen\\./, \"\");\n\n        /* Use provided components or fall back to defaults (Mise + Vitest) */\n        const components = options.components ?? DEFAULT_COMPONENTS;\n\n        /* Extract component configuration for constructor generation */\n        const { destructure, componentArray } = this.extractComponentOptions(components);\n        const imports = this.extractImports(options, optionsInterface);\n\n        /* Render all sections of the class file */\n        this.renderImports(imports);\n        this.buffer.line();\n        this.renderExport(optionsInterface);\n        this.buffer.line();\n        this.renderClass(options, optionsInterface, baseOptionsType, destructure, componentArray);\n        this.buffer.line();\n\n        return this.buffer.flush().join(\"\\n\");\n    }\n\n    /**\n     * Derives the fully qualified options interface name from base class\n     *\n     * Transforms base class reference into the corresponding Projen options interface name\n     * by appending \"Options\" suffix and prefixing with \"projen.\" namespace.\n     *\n     * @param baseClass - Base class in format \"module.ClassName\"\n     * @returns Fully qualified options interface name\n     * @throws {InvalidBaseClassFormatError} When baseClass format is invalid\n     *\n     * @example\n     * getBaseOptionsFqn(\"typescript.TypeScriptProject\") // \"projen.typescript.TypeScriptProjectOptions\"\n     * getBaseOptionsFqn(\"cdk.JsiiProject\") // \"projen.cdk.JsiiProjectOptions\"\n     */\n    getBaseOptionsFqn(baseClass: string): string {\n        /* Validate base class format: must contain module.ClassName structure */\n        if (!baseClass.includes(\".\")) {\n            throw new InvalidBaseClassFormatError(baseClass);\n        }\n\n        return `projen.${baseClass}Options`;\n    }\n\n    /**\n     * Extracts component configuration for constructor code generation\n     *\n     * Transforms component configurations into:\n     * 1. Destructure list: variable names to extract from options\n     * 2. Component array: code string for applyDefaults() call\n     *\n     * @param components - Component configurations\n     * @returns Object with destructure array and componentArray code string\n     *\n     * @example\n     * Input: [{ component: Vitest, optionsProperty: { name: \"vitestOptions\", ... } }]\n     * Output: {\n     *   destructure: [\"vitest\", \"vitestOptions\"],\n     *   componentArray: \"[{ component: Vitest, enabled: vitest, options: vitestOptions }]\"\n     * }\n     */\n    private extractComponentOptions(components: ComponentConfig[]): { destructure: string[]; componentArray: string } {\n        const destructure: string[] = [];\n        const componentParts: string[] = [];\n\n        for (const c of components) {\n            /* Convert component class name to camelCase variable name (e.g., Vitest -> vitest) */\n            const name = c.component.name.charAt(0).toLowerCase() + c.component.name.slice(1);\n            destructure.push(name);\n\n            /* Build component config object for applyDefaults() call */\n            const parts = [`component: ${c.component.name}`];\n            if (c.optionsProperty) {\n                /* Component has configurable options - include both enabled flag and options */\n                destructure.push(c.optionsProperty.name);\n                parts.push(`enabled: ${name}`, `options: ${c.optionsProperty.name}`);\n            } else {\n                /* Component has no options - only include enabled flag */\n                parts.push(`enabled: ${name}`);\n            }\n            componentParts.push(`{ ${parts.join(\", \")} }`);\n        }\n\n        return { destructure, componentArray: `[${componentParts.join(\", \")}]` };\n    }\n\n    /**\n     * Builds import statements map for the generated class\n     *\n     * Collects all required imports:\n     * - Projen base module (typescript, cdk, awscdk)\n     * - Config utilities (applyDefaults, defaultOptions)\n     * - Component classes\n     * - Utility functions (deepMerge)\n     * - Generated options interface (type-only import)\n     *\n     * @param options - Generator configuration\n     * @param optionsInterface - Name of the generated options interface\n     * @returns Map of module paths to sets of imported names\n     */\n    private extractImports(\n        options: ProjenProjectGeneratorOptions<ProjenModule>,\n        optionsInterface: string,\n    ): Map<string, Set<string>> {\n        const imports = new Map<string, Set<string>>();\n\n        /* Extract base module name from baseClass (e.g., \"typescript\" from \"typescript.TypeScriptProject\") */\n        const baseModule = options.baseClass.split(\".\")[0];\n        const optionsFileName = this.getOptionsFileName(optionsInterface);\n\n        /* Projen base module import */\n        imports.set(\"projen\", new Set([baseModule]));\n\n        /* Configuration utilities import */\n        imports.set(\"../config\", new Set([\"applyDefaults\", \"defaultOptions\"]));\n\n        /* Component class imports - derive module path from component class name */\n        const components = options.components ?? DEFAULT_COMPONENTS;\n        for (const c of components) {\n            const componentName = c.component.name;\n            const modulePath = `../components/${componentName.toLowerCase()}`;\n            if (!imports.has(modulePath)) {\n                imports.set(modulePath, new Set());\n            }\n            const moduleImports = imports.get(modulePath);\n            if (moduleImports) {\n                moduleImports.add(componentName);\n            }\n        }\n\n        /* Utility functions import */\n        imports.set(\"../utils\", new Set([\"deepMerge\"]));\n\n        /* Generated options interface import (type-only to avoid circular dependencies) */\n        imports.set(`./${optionsFileName}.generated`, new Set([optionsInterface]));\n\n        return imports;\n    }\n\n    /**\n     * Derives the config path for accessing default options\n     *\n     * Transforms base class reference into the corresponding path in the defaultOptions\n     * configuration object exported from config.ts.\n     *\n     * @param baseClass - Base class in format \"module.ClassName\"\n     * @returns Config path for accessing default options\n     *\n     * @example\n     * getConfigPath(\"typescript.TypeScriptProject\") // \"defaultOptions.typescript.TypeScriptProject\"\n     */\n    private getConfigPath(baseClass: string): string {\n        return `defaultOptions.${baseClass}`;\n    }\n\n    /**\n     * Renders import statements in sorted order\n     *\n     * Sorts imports with external packages first, then relative imports.\n     * Uses type-only imports for generated files to avoid circular dependencies.\n     *\n     * @param imports - Map of module paths to imported names\n     */\n    private renderImports(imports: Map<string, Set<string>>) {\n        /* Sort modules: external packages first, then relative imports alphabetically */\n        const sortedModules = Array.from(imports.keys()).sort((a, b) => {\n            const aIsRelative = a.startsWith(\".\");\n            const bIsRelative = b.startsWith(\".\");\n            if (aIsRelative !== bIsRelative) {\n                return aIsRelative ? 1 : -1;\n            }\n            return a.localeCompare(b);\n        });\n\n        for (const mod of sortedModules) {\n            const names = Array.from(imports.get(mod) || []).sort();\n            /* Use type-only import for generated files to prevent circular dependencies */\n            const isTypeOnly = mod.includes(\".generated\");\n            const importStmt = isTypeOnly ? \"import type\" : \"import\";\n            this.buffer.line(`${importStmt} { ${names.join(\", \")} } from \"${mod}\";`);\n        }\n    }\n\n    /**\n     * Renders re-export statement for the generated options interface\n     *\n     * @param optionsInterface - Name of the options interface to export\n     */\n    private renderExport(optionsInterface: string) {\n        const optionsFileName = this.getOptionsFileName(optionsInterface);\n        this.buffer.line();\n        this.buffer.line(`export { ${optionsInterface} } from \"./${optionsFileName}.generated\";`);\n    }\n\n    /**\n     * Renders the class declaration and constructor\n     *\n     * @param options - Generator configuration\n     * @param optionsInterface - Name of the options interface\n     * @param baseOptionsType - Base options type without \"projen.\" prefix\n     * @param destructure - Variable names to destructure from options\n     * @param componentArray - Code string for component array\n     */\n    private renderClass(\n        options: ProjenProjectGeneratorOptions<ProjenModule>,\n        optionsInterface: string,\n        baseOptionsType: string,\n        destructure: string[],\n        componentArray: string,\n    ) {\n        this.buffer.line();\n        this.buffer.line(\"/**\");\n        this.buffer.line(` * ${options.name} with standard configuration and component integration`);\n        this.buffer.line(\" *\");\n        this.buffer.line(\" * Extends Projen's base class with opinionated defaults and automatic component setup.\");\n        this.buffer.line(\" */\");\n        this.buffer.open(`export class ${options.name} extends ${options.baseClass} {`);\n        this.renderConstructor(options, optionsInterface, baseOptionsType, destructure, componentArray);\n        this.buffer.close(\"}\");\n    }\n\n    /**\n     * Renders the constructor implementation\n     *\n     * The constructor:\n     * 1. Destructures component flags and options from the options parameter\n     * 2. Merges default configuration with user-provided options\n     * 3. Calls super() with merged options\n     * 4. Applies component defaults via applyDefaults()\n     *\n     * @param options - Generator configuration\n     * @param optionsInterface - Name of the options interface\n     * @param baseOptionsType - Base options type without \"projen.\" prefix\n     * @param destructure - Variable names to destructure from options\n     * @param componentArray - Code string for component array\n     */\n    private renderConstructor(\n        options: ProjenProjectGeneratorOptions<ProjenModule>,\n        optionsInterface: string,\n        baseOptionsType: string,\n        destructure: string[],\n        componentArray: string,\n    ) {\n        const configPath = this.getConfigPath(options.baseClass);\n        this.buffer.line(\"/**\");\n        this.buffer.line(\" * @param options - Project configuration\");\n        this.buffer.line(\" */\");\n        this.buffer.open(`constructor(options: ${optionsInterface}) {`);\n        this.buffer.line(\"/* Separate component configuration from base Projen options */\");\n        this.buffer.line(`const { ${destructure.join(\", \")}, ...baseOptions } = options;`);\n        this.buffer.line();\n        this.buffer.line(\"/* Merge default configuration with user options and initialize base class */\");\n        this.buffer.line(`super(deepMerge<${baseOptionsType}>(${configPath}, baseOptions));`);\n        this.buffer.line();\n        this.buffer.line(\"/* Apply component defaults and instantiate enabled components */\");\n        this.buffer.line(`applyDefaults(this, ${componentArray});`);\n        this.buffer.close(\"}\");\n    }\n\n    /**\n     * Maps options interface name to file name\n     *\n     * Converts PascalCase interface names to kebab-case file names.\n     * Falls back to lowercase conversion for unmapped interfaces.\n     *\n     * @param optionsInterface - Options interface name\n     * @returns Kebab-case file name without extension\n     *\n     * @example\n     * getOptionsFileName(\"TypeScriptProjectOptions\") // \"typescript-options\"\n     */\n    getOptionsFileName(optionsInterface: string): string {\n        const mapping: Record<string, string> = {\n            TypeScriptProjectOptions: \"typescript-options\",\n            JsiiProjectOptions: \"jsii-options\",\n            AwsCdkTypeScriptAppProjectOptions: \"awscdk-typescript-app-options\",\n            AwsCdkConstructLibraryProjectOptions: \"awscdk-construct-library-options\",\n        };\n        return mapping[optionsInterface] || optionsInterface.toLowerCase();\n    }\n}\n\n/**\n * Projen component that generates TypeScript project classes with standard configuration\n *\n * This component automates the creation of project classes that extend Projen base classes\n * with opinionated defaults and component integration. It generates both:\n * 1. An options interface (via ProjenStruct) that extends the base Projen options\n * 2. A project class that applies default configuration and instantiates components\n *\n * The generated code follows a consistent pattern:\n * - Imports required modules and components\n * - Re-exports the generated options interface\n * - Defines a class extending the Projen base class\n * - Constructor merges defaults with user options and applies components\n *\n * @example\n * ```typescript\n * new ProjenProjectGenerator(project, {\n *   name: \"TypeScriptProject\",\n *   baseClass: \"typescript.TypeScriptProject\",\n *   filePath: \"src/projects/typescript.generated.ts\",\n *   components: [{ component: Vitest, optionsProperty: \"vitestOptions\" }]\n * });\n * ```\n */\nexport class ProjenProjectGenerator extends Component {\n    private renderer: TypeScriptClassRenderer;\n\n    /**\n     * @param project - Projen project instance\n     * @param options - Generator configuration\n     */\n    constructor(\n        project: Project,\n        private readonly options: ProjenProjectGeneratorOptions<ProjenModule>,\n    ) {\n        super(project);\n        this.renderer = new TypeScriptClassRenderer();\n\n        /* Generate the options interface using ProjenStruct for JSII compatibility */\n        const optionsInterface = `${options.name}Options`;\n        const baseOptionsFqn = this.renderer.getBaseOptionsFqn(options.baseClass);\n        const optionsFilePath = this.getOptionsFilePath(optionsInterface);\n\n        /* ProjenStruct generates a concrete TypeScript interface from Projen's options\n         * This is necessary because JSII doesn't support TypeScript utility types like Omit<>\n         * The generated interface can be consumed by other languages (Python, Java, C#, Go) */\n        const struct = new ProjenStruct(this.asTypeScriptProject(project), {\n            name: optionsInterface,\n            filePath: optionsFilePath,\n            outputFileOptions: { readonly: true },\n        })\n            .mixin(Struct.fromFqn(baseOptionsFqn))\n            .withoutDeprecated();\n\n        /* Remove specified options from the base interface */\n        if (options.omitOptions) {\n            struct.omit(...options.omitOptions);\n        }\n\n        /* Add component-derived options to the interface */\n        const components = options.components ?? DEFAULT_COMPONENTS;\n        const { PrimitiveType } = require(\"@jsii/spec\");\n\n        for (const c of components) {\n            const name = c.component.name.charAt(0).toLowerCase() + c.component.name.slice(1);\n\n            /* Add enabled flag for the component */\n            struct.add({\n                name,\n                type: { primitive: PrimitiveType.Boolean },\n                optional: true,\n                docs: {\n                    summary: `Enable ${c.component.name} component`,\n                    default: \"true\",\n                },\n            });\n\n            /* Add options property if component is configurable */\n            if (c.optionsProperty) {\n                struct.add({\n                    name: c.optionsProperty.name,\n                    type: { fqn: c.optionsProperty.type },\n                    optional: true,\n                    docs: {\n                        summary: c.optionsProperty.docs ?? `${c.component.name} configuration`,\n                        default: `- default ${c.component.name} configuration`,\n                    },\n                });\n            }\n        }\n\n        /* Add custom options to the interface */\n        if (options.additionalOptions) {\n            struct.add(...options.additionalOptions);\n        }\n    }\n\n    /**\n     * Casts project to TypeScript project type\n     *\n     * ProjenStruct requires a TypeScriptProject instance. This generator is only used\n     * within TypeScript projects, so the cast is safe in practice.\n     *\n     * Note: Type assertion is necessary here because Projen's type system doesn't provide\n     * a runtime type guard for TypeScriptProject.\n     *\n     * @param project - Projen project instance\n     * @returns TypeScript project instance\n     */\n    private asTypeScriptProject(project: Project): typescript.TypeScriptProject {\n        return project as unknown as typescript.TypeScriptProject;\n    }\n\n    /**\n     * Derives the file path for the generated options interface\n     *\n     * Places the options interface file in the same directory as the class file\n     * with a \".generated.ts\" suffix.\n     *\n     * @param optionsInterface - Name of the options interface\n     * @returns File path for the options interface\n     * @throws {InvalidFilePathError} When filePath doesn't contain a directory separator\n     */\n    private getOptionsFilePath(optionsInterface: string): string {\n        const lastSlash = this.options.filePath.lastIndexOf(\"/\");\n        if (lastSlash === -1) {\n            throw new InvalidFilePathError(this.options.filePath);\n        }\n\n        const dir = this.options.filePath.substring(0, lastSlash);\n        const optionsFileName = this.renderer.getOptionsFileName(optionsInterface);\n        return `${dir}/${optionsFileName}.generated.ts`;\n    }\n\n    /**\n     * Generates the TypeScript class file during Projen synthesis\n     *\n     * Called by Projen during the synthesis phase to generate the project class file.\n     * The file is marked as readonly to prevent manual editing.\n     */\n    preSynthesize(): void {\n        const content = this.renderer.render(this.options);\n\n        new TypeScriptClassFile(this.project, this.options.filePath, content, {\n            readonly: this.options.readonly ?? true,\n        });\n    }\n}\n\n/**\n * Text file for generated TypeScript class code\n *\n * Extends Projen's TextFile to add the generated file marker comment\n * at the beginning of the file.\n */\nclass TypeScriptClassFile extends TextFile {\n    /**\n     * @param project - Projen project instance\n     * @param filePath - Output file path\n     * @param content - TypeScript class code\n     * @param options - Source code options (readonly, etc.)\n     */\n    constructor(project: Project, filePath: string, content: string, options: SourceCodeOptions = {}) {\n        super(project, filePath, options);\n        /* Add generated file marker to prevent manual editing */\n        this.addLine(`// ${this.marker}`);\n        this.addLine(content);\n    }\n}\n"]}
|
package/package.json
CHANGED
|
@@ -1,44 +0,0 @@
|
|
|
1
|
-
import type { Project, SourceCodeOptions } from "projen";
|
|
2
|
-
import { Component } from "projen";
|
|
3
|
-
export interface ProjenProjectClassOptions extends SourceCodeOptions {
|
|
4
|
-
/**
|
|
5
|
-
* The name of the generated class
|
|
6
|
-
*/
|
|
7
|
-
readonly name: string;
|
|
8
|
-
/**
|
|
9
|
-
* The base class to extend (e.g., "typescript.TypeScriptProject")
|
|
10
|
-
*/
|
|
11
|
-
readonly baseClass: string;
|
|
12
|
-
/**
|
|
13
|
-
* The options interface name (e.g., "TypeScriptProjectOptions")
|
|
14
|
-
*/
|
|
15
|
-
readonly optionsInterface: string;
|
|
16
|
-
/**
|
|
17
|
-
* The base options type (e.g., "typescript.TypeScriptProjectOptions")
|
|
18
|
-
*/
|
|
19
|
-
readonly baseOptionsType: string;
|
|
20
|
-
/**
|
|
21
|
-
* The default config constant name (e.g., "typescriptProjectDefaultOptions")
|
|
22
|
-
*/
|
|
23
|
-
readonly defaultConfig: string;
|
|
24
|
-
/**
|
|
25
|
-
* Output file path
|
|
26
|
-
*/
|
|
27
|
-
readonly filePath: string;
|
|
28
|
-
/**
|
|
29
|
-
* Custom options to extract from constructor
|
|
30
|
-
* @default ["mise", "vitest", "vitestOptions"]
|
|
31
|
-
*/
|
|
32
|
-
readonly customOptions?: string[];
|
|
33
|
-
/**
|
|
34
|
-
* Indentation size
|
|
35
|
-
* @default 4
|
|
36
|
-
*/
|
|
37
|
-
readonly indent?: number;
|
|
38
|
-
}
|
|
39
|
-
export declare class ProjenProjectClass extends Component {
|
|
40
|
-
private readonly options;
|
|
41
|
-
private renderer;
|
|
42
|
-
constructor(project: Project, options: ProjenProjectClassOptions);
|
|
43
|
-
preSynthesize(): void;
|
|
44
|
-
}
|