@nikovirtala/projen-constructs 0.1.4 → 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.
Files changed (35) hide show
  1. package/.jsii +223 -203
  2. package/API.md +52 -28
  3. package/lib/components/mise.js +4 -2
  4. package/lib/components/vitest.js +1 -1
  5. package/lib/config.d.ts +20 -7
  6. package/lib/config.js +28 -16
  7. package/lib/errors.d.ts +20 -0
  8. package/lib/errors.js +36 -0
  9. package/lib/projects/awscdk-construct-library-options.generated.d.ts +7 -7
  10. package/lib/projects/awscdk-construct-library-options.generated.js +1 -1
  11. package/lib/projects/{awscdk-construct-library.d.ts → awscdk-construct-library.generated.d.ts} +8 -0
  12. package/lib/projects/awscdk-construct-library.generated.js +33 -0
  13. package/lib/projects/awscdk-typescript-app-options.generated.d.ts +31 -31
  14. package/lib/projects/awscdk-typescript-app-options.generated.js +1 -1
  15. package/lib/projects/{awscdk-typescript-app.d.ts → awscdk-typescript-app.generated.d.ts} +8 -0
  16. package/lib/projects/awscdk-typescript-app.generated.js +33 -0
  17. package/lib/projects/index.d.ts +4 -4
  18. package/lib/projects/index.js +5 -5
  19. package/lib/projects/jsii-options.generated.d.ts +7 -7
  20. package/lib/projects/jsii-options.generated.js +1 -1
  21. package/lib/projects/{jsii.d.ts → jsii.generated.d.ts} +8 -0
  22. package/lib/projects/jsii.generated.js +33 -0
  23. package/lib/projects/typescript-options.generated.d.ts +7 -7
  24. package/lib/projects/typescript-options.generated.js +1 -1
  25. package/lib/projects/{typescript.d.ts → typescript.generated.d.ts} +8 -0
  26. package/lib/projects/typescript.generated.js +33 -0
  27. package/lib/projen-project-generator.d.ts +173 -0
  28. package/lib/projen-project-generator.js +509 -0
  29. package/package.json +1 -1
  30. package/lib/projects/awscdk-app-options.generated.d.ts +0 -1098
  31. package/lib/projects/awscdk-app-options.generated.js +0 -3
  32. package/lib/projects/awscdk-construct-library.js +0 -19
  33. package/lib/projects/awscdk-typescript-app.js +0 -19
  34. package/lib/projects/jsii.js +0 -19
  35. package/lib/projects/typescript.js +0 -19
@@ -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
@@ -50,7 +50,7 @@
50
50
  "publishConfig": {
51
51
  "access": "public"
52
52
  },
53
- "version": "0.1.4",
53
+ "version": "0.1.6",
54
54
  "types": "lib/index.d.ts",
55
55
  "stability": "stable",
56
56
  "jsii": {