@brainfish-ai/devdoc 0.1.45 → 0.1.46

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.
@@ -0,0 +1,871 @@
1
+ "use strict";
2
+ /**
3
+ * SDK Generation Module
4
+ *
5
+ * This module provides functionality for generating client SDKs from OpenAPI specifications.
6
+ * It supports multiple languages using both native generators and OpenAPI Generator.
7
+ */
8
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
9
+ if (k2 === undefined) k2 = k;
10
+ var desc = Object.getOwnPropertyDescriptor(m, k);
11
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
12
+ desc = { enumerable: true, get: function() { return m[k]; } };
13
+ }
14
+ Object.defineProperty(o, k2, desc);
15
+ }) : (function(o, m, k, k2) {
16
+ if (k2 === undefined) k2 = k;
17
+ o[k2] = m[k];
18
+ }));
19
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
20
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
21
+ }) : function(o, v) {
22
+ o["default"] = v;
23
+ });
24
+ var __importStar = (this && this.__importStar) || (function () {
25
+ var ownKeys = function(o) {
26
+ ownKeys = Object.getOwnPropertyNames || function (o) {
27
+ var ar = [];
28
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
29
+ return ar;
30
+ };
31
+ return ownKeys(o);
32
+ };
33
+ return function (mod) {
34
+ if (mod && mod.__esModule) return mod;
35
+ var result = {};
36
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
37
+ __setModuleDefault(result, mod);
38
+ return result;
39
+ };
40
+ })();
41
+ var __importDefault = (this && this.__importDefault) || function (mod) {
42
+ return (mod && mod.__esModule) ? mod : { "default": mod };
43
+ };
44
+ Object.defineProperty(exports, "__esModule", { value: true });
45
+ exports.SUPPORTED_LANGUAGES = void 0;
46
+ exports.loadSDKConfig = loadSDKConfig;
47
+ exports.initSDKConfig = initSDKConfig;
48
+ exports.getEnabledGenerators = getEnabledGenerators;
49
+ exports.getGeneratorInfo = getGeneratorInfo;
50
+ exports.isNativeGenerator = isNativeGenerator;
51
+ exports.resolveSpecPath = resolveSpecPath;
52
+ exports.validateSpec = validateSpec;
53
+ exports.generateSDKs = generateSDKs;
54
+ const path_1 = __importDefault(require("path"));
55
+ const fs_extra_1 = __importDefault(require("fs-extra"));
56
+ const child_process_1 = require("child_process");
57
+ const util_1 = require("util");
58
+ const execAsync = (0, util_1.promisify)(child_process_1.exec);
59
+ // ============================================================================
60
+ // Constants
61
+ // ============================================================================
62
+ /**
63
+ * All supported languages
64
+ */
65
+ exports.SUPPORTED_LANGUAGES = [
66
+ 'typescript',
67
+ 'python',
68
+ 'go',
69
+ 'java',
70
+ 'csharp',
71
+ 'ruby',
72
+ 'php',
73
+ 'swift',
74
+ 'kotlin',
75
+ 'rust',
76
+ ];
77
+ /**
78
+ * Generator information for each language
79
+ */
80
+ const GENERATORS = [
81
+ { language: 'typescript', displayName: 'TypeScript', native: true, requiresJava: false },
82
+ { language: 'python', displayName: 'Python', native: true, requiresJava: false },
83
+ { language: 'go', displayName: 'Go', native: true, requiresJava: false },
84
+ { language: 'java', displayName: 'Java', native: false, requiresJava: true, generatorName: 'java' },
85
+ { language: 'csharp', displayName: 'C#', native: false, requiresJava: true, generatorName: 'csharp' },
86
+ { language: 'ruby', displayName: 'Ruby', native: false, requiresJava: true, generatorName: 'ruby' },
87
+ { language: 'php', displayName: 'PHP', native: false, requiresJava: true, generatorName: 'php' },
88
+ { language: 'swift', displayName: 'Swift', native: false, requiresJava: true, generatorName: 'swift5' },
89
+ { language: 'kotlin', displayName: 'Kotlin', native: false, requiresJava: true, generatorName: 'kotlin' },
90
+ { language: 'rust', displayName: 'Rust', native: false, requiresJava: true, generatorName: 'rust' },
91
+ ];
92
+ const CONFIG_FILENAME = 'sdk.json';
93
+ // ============================================================================
94
+ // Configuration Functions
95
+ // ============================================================================
96
+ /**
97
+ * Load SDK configuration from project directory
98
+ */
99
+ async function loadSDKConfig(projectRoot) {
100
+ const configPath = path_1.default.join(projectRoot, CONFIG_FILENAME);
101
+ if (!(await fs_extra_1.default.pathExists(configPath))) {
102
+ return { config: null, error: `${CONFIG_FILENAME} not found` };
103
+ }
104
+ try {
105
+ const config = await fs_extra_1.default.readJson(configPath);
106
+ return { config };
107
+ }
108
+ catch (error) {
109
+ return {
110
+ config: null,
111
+ error: `Failed to parse ${CONFIG_FILENAME}: ${error instanceof Error ? error.message : String(error)}`,
112
+ };
113
+ }
114
+ }
115
+ /**
116
+ * Initialize SDK configuration
117
+ */
118
+ async function initSDKConfig(projectRoot, options) {
119
+ const configPath = path_1.default.join(projectRoot, CONFIG_FILENAME);
120
+ // Check if config already exists
121
+ if ((await fs_extra_1.default.pathExists(configPath)) && !options.force) {
122
+ throw new Error(`${CONFIG_FILENAME} already exists. Use --force to overwrite.`);
123
+ }
124
+ // Create language configurations
125
+ const languages = {};
126
+ for (const lang of options.languages) {
127
+ languages[lang] = {
128
+ enabled: true,
129
+ packageName: getDefaultPackageName(options.packageName, lang),
130
+ };
131
+ }
132
+ const config = {
133
+ openapi: options.openapi,
134
+ packageName: options.packageName,
135
+ output: './sdks',
136
+ languages,
137
+ };
138
+ await fs_extra_1.default.writeJson(configPath, config, { spaces: 2 });
139
+ return { configPath, config };
140
+ }
141
+ /**
142
+ * Get enabled generators from config
143
+ */
144
+ function getEnabledGenerators(config) {
145
+ return Object.entries(config.languages)
146
+ .filter(([, langConfig]) => langConfig?.enabled)
147
+ .map(([lang]) => lang);
148
+ }
149
+ /**
150
+ * Get generator information for all languages
151
+ */
152
+ function getGeneratorInfo() {
153
+ return [...GENERATORS];
154
+ }
155
+ /**
156
+ * Check if a generator is native (doesn't require Java)
157
+ */
158
+ function isNativeGenerator(language) {
159
+ const generator = GENERATORS.find((g) => g.language === language);
160
+ return generator?.native ?? false;
161
+ }
162
+ // ============================================================================
163
+ // Path Utilities
164
+ // ============================================================================
165
+ /**
166
+ * Resolve OpenAPI spec path relative to project root
167
+ */
168
+ function resolveSpecPath(projectRoot, specPath) {
169
+ if (path_1.default.isAbsolute(specPath)) {
170
+ return specPath;
171
+ }
172
+ return path_1.default.resolve(projectRoot, specPath);
173
+ }
174
+ /**
175
+ * Get default package name for a language
176
+ */
177
+ function getDefaultPackageName(baseName, language) {
178
+ const sanitized = baseName.toLowerCase().replace(/[^a-z0-9]/g, '-');
179
+ switch (language) {
180
+ case 'typescript':
181
+ return `@${sanitized}/sdk`;
182
+ case 'python':
183
+ return sanitized.replace(/-/g, '_');
184
+ case 'go':
185
+ return sanitized;
186
+ case 'java':
187
+ return `com.${sanitized.replace(/-/g, '.')}`;
188
+ case 'csharp':
189
+ return sanitized.split('-').map(capitalize).join('.');
190
+ case 'ruby':
191
+ return sanitized.replace(/-/g, '_');
192
+ case 'php':
193
+ return sanitized.split('-').map(capitalize).join('\\');
194
+ case 'swift':
195
+ return sanitized.split('-').map(capitalize).join('');
196
+ case 'kotlin':
197
+ return `com.${sanitized.replace(/-/g, '.')}`;
198
+ case 'rust':
199
+ return sanitized.replace(/-/g, '_');
200
+ default:
201
+ return sanitized;
202
+ }
203
+ }
204
+ function capitalize(str) {
205
+ return str.charAt(0).toUpperCase() + str.slice(1);
206
+ }
207
+ // ============================================================================
208
+ // Validation
209
+ // ============================================================================
210
+ /**
211
+ * Validate an OpenAPI specification
212
+ */
213
+ async function validateSpec(specPath) {
214
+ const errors = [];
215
+ const warnings = [];
216
+ // Check file exists
217
+ if (!(await fs_extra_1.default.pathExists(specPath))) {
218
+ return {
219
+ valid: false,
220
+ errors: [`OpenAPI specification not found: ${specPath}`],
221
+ warnings: [],
222
+ };
223
+ }
224
+ try {
225
+ // Read and parse the spec
226
+ const content = await fs_extra_1.default.readFile(specPath, 'utf-8');
227
+ let spec;
228
+ if (specPath.endsWith('.yaml') || specPath.endsWith('.yml')) {
229
+ const yaml = await Promise.resolve().then(() => __importStar(require('js-yaml')));
230
+ spec = yaml.load(content);
231
+ }
232
+ else {
233
+ spec = JSON.parse(content);
234
+ }
235
+ // Basic validation
236
+ if (!spec.openapi && !spec.swagger) {
237
+ errors.push('Missing "openapi" or "swagger" version field');
238
+ }
239
+ if (!spec.info) {
240
+ errors.push('Missing "info" section');
241
+ }
242
+ else {
243
+ const info = spec.info;
244
+ if (!info.title) {
245
+ warnings.push('Missing "info.title"');
246
+ }
247
+ if (!info.version) {
248
+ warnings.push('Missing "info.version"');
249
+ }
250
+ }
251
+ if (!spec.paths || Object.keys(spec.paths).length === 0) {
252
+ warnings.push('No paths defined in specification');
253
+ }
254
+ // Check for common issues
255
+ const openapiVersion = spec.openapi;
256
+ if (openapiVersion && !openapiVersion.startsWith('3.')) {
257
+ warnings.push(`OpenAPI ${openapiVersion} detected. Version 3.x is recommended.`);
258
+ }
259
+ return {
260
+ valid: errors.length === 0,
261
+ errors,
262
+ warnings,
263
+ };
264
+ }
265
+ catch (error) {
266
+ return {
267
+ valid: false,
268
+ errors: [`Failed to parse specification: ${error instanceof Error ? error.message : String(error)}`],
269
+ warnings: [],
270
+ };
271
+ }
272
+ }
273
+ // ============================================================================
274
+ // SDK Generation
275
+ // ============================================================================
276
+ /**
277
+ * Generate SDKs for configured languages
278
+ */
279
+ async function generateSDKs(projectRoot, options = {}) {
280
+ const startTime = Date.now();
281
+ const results = [];
282
+ // Load config
283
+ const { config, error } = await loadSDKConfig(projectRoot);
284
+ if (!config) {
285
+ throw new Error(error || 'SDK configuration not found');
286
+ }
287
+ // Determine which languages to generate
288
+ let languagesToGenerate = getEnabledGenerators(config);
289
+ if (options.language) {
290
+ if (!languagesToGenerate.includes(options.language)) {
291
+ throw new Error(`Language "${options.language}" is not enabled in sdk.json`);
292
+ }
293
+ languagesToGenerate = [options.language];
294
+ }
295
+ // Resolve spec path
296
+ const specPath = resolveSpecPath(projectRoot, config.openapi);
297
+ // Validate spec exists
298
+ if (!(await fs_extra_1.default.pathExists(specPath))) {
299
+ throw new Error(`OpenAPI specification not found: ${config.openapi}`);
300
+ }
301
+ // Base output directory
302
+ const baseOutputDir = options.outputDir
303
+ ? path_1.default.resolve(projectRoot, options.outputDir)
304
+ : path_1.default.resolve(projectRoot, config.output);
305
+ // Generate SDKs for each language
306
+ for (const language of languagesToGenerate) {
307
+ const langStart = Date.now();
308
+ const langConfig = config.languages[language];
309
+ const outputPath = langConfig?.output
310
+ ? path_1.default.resolve(projectRoot, langConfig.output)
311
+ : path_1.default.join(baseOutputDir, language);
312
+ try {
313
+ if (options.preview) {
314
+ // Preview mode - just report what would be generated
315
+ results.push({
316
+ language,
317
+ success: true,
318
+ outputPath,
319
+ duration: Date.now() - langStart,
320
+ });
321
+ continue;
322
+ }
323
+ // Ensure output directory exists
324
+ await fs_extra_1.default.ensureDir(outputPath);
325
+ // Generate SDK based on generator type
326
+ const generator = GENERATORS.find((g) => g.language === language);
327
+ if (generator?.native) {
328
+ await generateNativeSDK(language, specPath, outputPath, config, langConfig);
329
+ }
330
+ else if (generator?.generatorName) {
331
+ await generateOpenAPIGeneratorSDK(generator.generatorName, specPath, outputPath, config, langConfig);
332
+ }
333
+ else {
334
+ throw new Error(`No generator found for ${language}`);
335
+ }
336
+ results.push({
337
+ language,
338
+ success: true,
339
+ outputPath,
340
+ duration: Date.now() - langStart,
341
+ });
342
+ }
343
+ catch (err) {
344
+ results.push({
345
+ language,
346
+ success: false,
347
+ outputPath,
348
+ error: err instanceof Error ? err.message : String(err),
349
+ duration: Date.now() - langStart,
350
+ });
351
+ }
352
+ }
353
+ return {
354
+ results,
355
+ successCount: results.filter((r) => r.success).length,
356
+ failureCount: results.filter((r) => !r.success).length,
357
+ totalDuration: Date.now() - startTime,
358
+ };
359
+ }
360
+ /**
361
+ * Generate SDK using native generator (TypeScript, Python, Go)
362
+ */
363
+ async function generateNativeSDK(language, specPath, outputPath, config, langConfig) {
364
+ const packageName = langConfig?.packageName || getDefaultPackageName(config.packageName, language);
365
+ switch (language) {
366
+ case 'typescript':
367
+ await generateTypeScriptSDK(specPath, outputPath, packageName);
368
+ break;
369
+ case 'python':
370
+ await generatePythonSDK(specPath, outputPath, packageName);
371
+ break;
372
+ case 'go':
373
+ await generateGoSDK(specPath, outputPath, packageName);
374
+ break;
375
+ default:
376
+ throw new Error(`Native generator not implemented for ${language}`);
377
+ }
378
+ }
379
+ /**
380
+ * Generate SDK using OpenAPI Generator (Java-based)
381
+ */
382
+ async function generateOpenAPIGeneratorSDK(generatorName, specPath, outputPath, config, langConfig) {
383
+ // Check if Java is available
384
+ try {
385
+ await execAsync('java -version');
386
+ }
387
+ catch {
388
+ throw new Error('Java is required for this generator. Please install Java 11 or later.');
389
+ }
390
+ // Check if openapi-generator is available
391
+ let generatorCmd = 'openapi-generator';
392
+ try {
393
+ await execAsync('openapi-generator version');
394
+ }
395
+ catch {
396
+ // Try npx fallback
397
+ generatorCmd = 'npx @openapitools/openapi-generator-cli';
398
+ }
399
+ const additionalProps = langConfig?.additionalOptions
400
+ ? Object.entries(langConfig.additionalOptions)
401
+ .map(([k, v]) => `${k}=${v}`)
402
+ .join(',')
403
+ : '';
404
+ const cmd = [
405
+ generatorCmd,
406
+ 'generate',
407
+ '-i',
408
+ specPath,
409
+ '-g',
410
+ generatorName,
411
+ '-o',
412
+ outputPath,
413
+ additionalProps ? `--additional-properties=${additionalProps}` : '',
414
+ ]
415
+ .filter(Boolean)
416
+ .join(' ');
417
+ await execAsync(cmd);
418
+ }
419
+ // ============================================================================
420
+ // Native Generator Implementations
421
+ // ============================================================================
422
+ /**
423
+ * Generate TypeScript SDK
424
+ */
425
+ async function generateTypeScriptSDK(specPath, outputPath, packageName) {
426
+ // Read the OpenAPI spec
427
+ const content = await fs_extra_1.default.readFile(specPath, 'utf-8');
428
+ let spec;
429
+ if (specPath.endsWith('.yaml') || specPath.endsWith('.yml')) {
430
+ const yaml = await Promise.resolve().then(() => __importStar(require('js-yaml')));
431
+ spec = yaml.load(content);
432
+ }
433
+ else {
434
+ spec = JSON.parse(content);
435
+ }
436
+ const info = (spec.info || {});
437
+ const paths = (spec.paths || {});
438
+ // Generate package.json
439
+ const packageJson = {
440
+ name: packageName,
441
+ version: info.version || '1.0.0',
442
+ description: info.description || 'Generated SDK',
443
+ main: 'dist/index.js',
444
+ types: 'dist/index.d.ts',
445
+ scripts: {
446
+ build: 'tsc',
447
+ prepublishOnly: 'npm run build',
448
+ },
449
+ devDependencies: {
450
+ typescript: '^5.0.0',
451
+ '@types/node': '^20.0.0',
452
+ },
453
+ };
454
+ await fs_extra_1.default.writeJson(path_1.default.join(outputPath, 'package.json'), packageJson, { spaces: 2 });
455
+ // Generate tsconfig.json
456
+ const tsConfig = {
457
+ compilerOptions: {
458
+ target: 'ES2020',
459
+ module: 'commonjs',
460
+ declaration: true,
461
+ outDir: './dist',
462
+ strict: true,
463
+ esModuleInterop: true,
464
+ skipLibCheck: true,
465
+ forceConsistentCasingInFileNames: true,
466
+ },
467
+ include: ['src/**/*'],
468
+ exclude: ['node_modules', 'dist'],
469
+ };
470
+ await fs_extra_1.default.writeJson(path_1.default.join(outputPath, 'tsconfig.json'), tsConfig, { spaces: 2 });
471
+ // Generate source files
472
+ await fs_extra_1.default.ensureDir(path_1.default.join(outputPath, 'src'));
473
+ // Generate types
474
+ const typesContent = generateTypeScriptTypes(spec);
475
+ await fs_extra_1.default.writeFile(path_1.default.join(outputPath, 'src', 'types.ts'), typesContent);
476
+ // Generate client
477
+ const clientContent = generateTypeScriptClient(spec, paths);
478
+ await fs_extra_1.default.writeFile(path_1.default.join(outputPath, 'src', 'client.ts'), clientContent);
479
+ // Generate index
480
+ const indexContent = `export * from './types';\nexport * from './client';\n`;
481
+ await fs_extra_1.default.writeFile(path_1.default.join(outputPath, 'src', 'index.ts'), indexContent);
482
+ // Generate README
483
+ const readmeContent = `# ${packageName}
484
+
485
+ Generated TypeScript SDK for ${info.title || 'API'}.
486
+
487
+ ## Installation
488
+
489
+ \`\`\`bash
490
+ npm install ${packageName}
491
+ \`\`\`
492
+
493
+ ## Usage
494
+
495
+ \`\`\`typescript
496
+ import { ApiClient } from '${packageName}';
497
+
498
+ const client = new ApiClient({
499
+ baseUrl: 'https://api.example.com',
500
+ // apiKey: 'your-api-key',
501
+ });
502
+ \`\`\`
503
+ `;
504
+ await fs_extra_1.default.writeFile(path_1.default.join(outputPath, 'README.md'), readmeContent);
505
+ }
506
+ function generateTypeScriptTypes(spec) {
507
+ let content = '/**\n * Generated types from OpenAPI specification\n */\n\n';
508
+ const schemas = (spec.components?.schemas || {});
509
+ // If no schemas, export a placeholder to make this a valid module
510
+ if (Object.keys(schemas).length === 0) {
511
+ content += '// No schemas defined in the OpenAPI specification\n';
512
+ content += 'export type ApiResponse<T> = T;\n';
513
+ content += 'export type ApiError = { message: string; code?: number };\n';
514
+ return content;
515
+ }
516
+ for (const [name, schema] of Object.entries(schemas)) {
517
+ content += `export interface ${name} {\n`;
518
+ const properties = schema.properties;
519
+ const required = (schema.required || []);
520
+ if (properties) {
521
+ for (const [propName, propSchema] of Object.entries(properties)) {
522
+ const isRequired = required.includes(propName);
523
+ const tsType = openApiTypeToTs(propSchema);
524
+ content += ` ${propName}${isRequired ? '' : '?'}: ${tsType};\n`;
525
+ }
526
+ }
527
+ content += '}\n\n';
528
+ }
529
+ return content;
530
+ }
531
+ function openApiTypeToTs(schema) {
532
+ const type = schema.type;
533
+ const format = schema.format;
534
+ const ref = schema.$ref;
535
+ if (ref) {
536
+ return ref.split('/').pop() || 'unknown';
537
+ }
538
+ switch (type) {
539
+ case 'string':
540
+ return 'string';
541
+ case 'integer':
542
+ case 'number':
543
+ return 'number';
544
+ case 'boolean':
545
+ return 'boolean';
546
+ case 'array':
547
+ const items = schema.items;
548
+ return `${openApiTypeToTs(items || {})}[]`;
549
+ case 'object':
550
+ return 'Record<string, unknown>';
551
+ default:
552
+ return 'unknown';
553
+ }
554
+ }
555
+ function generateTypeScriptClient(spec, paths) {
556
+ const info = (spec.info || {});
557
+ const servers = (spec.servers || []);
558
+ const defaultBaseUrl = servers[0]?.url || 'https://api.example.com';
559
+ let content = `/**
560
+ * ${info.title || 'API'} Client
561
+ * ${info.description || ''}
562
+ */
563
+
564
+ export interface ClientConfig {
565
+ baseUrl?: string;
566
+ apiKey?: string;
567
+ headers?: Record<string, string>;
568
+ }
569
+
570
+ export class ApiClient {
571
+ private baseUrl: string;
572
+ private headers: Record<string, string>;
573
+
574
+ constructor(config: ClientConfig = {}) {
575
+ this.baseUrl = config.baseUrl || '${defaultBaseUrl}';
576
+ this.headers = {
577
+ 'Content-Type': 'application/json',
578
+ ...config.headers,
579
+ };
580
+
581
+ if (config.apiKey) {
582
+ this.headers['Authorization'] = \`Bearer \${config.apiKey}\`;
583
+ }
584
+ }
585
+
586
+ private async request<T>(
587
+ method: string,
588
+ path: string,
589
+ options: { body?: unknown; query?: Record<string, string> } = {}
590
+ ): Promise<T> {
591
+ let url = \`\${this.baseUrl}\${path}\`;
592
+
593
+ if (options.query) {
594
+ const params = new URLSearchParams(options.query);
595
+ url += \`?\${params.toString()}\`;
596
+ }
597
+
598
+ const response = await fetch(url, {
599
+ method,
600
+ headers: this.headers,
601
+ body: options.body ? JSON.stringify(options.body) : undefined,
602
+ });
603
+
604
+ if (!response.ok) {
605
+ throw new Error(\`HTTP \${response.status}: \${response.statusText}\`);
606
+ }
607
+
608
+ return response.json();
609
+ }
610
+ `;
611
+ // Generate methods for each path
612
+ for (const [pathUrl, pathItem] of Object.entries(paths)) {
613
+ const methods = ['get', 'post', 'put', 'patch', 'delete'];
614
+ for (const method of methods) {
615
+ const operation = pathItem[method];
616
+ if (!operation)
617
+ continue;
618
+ const operationId = operation.operationId || `${method}${pathUrl.replace(/[^a-zA-Z0-9]/g, '_')}`;
619
+ const summary = operation.summary || '';
620
+ content += `
621
+ /**
622
+ * ${summary}
623
+ */
624
+ async ${operationId}(): Promise<unknown> {
625
+ return this.request('${method.toUpperCase()}', '${pathUrl}');
626
+ }
627
+ `;
628
+ }
629
+ }
630
+ content += '}\n';
631
+ return content;
632
+ }
633
+ /**
634
+ * Generate Python SDK
635
+ */
636
+ async function generatePythonSDK(specPath, outputPath, packageName) {
637
+ // Read the OpenAPI spec
638
+ const content = await fs_extra_1.default.readFile(specPath, 'utf-8');
639
+ let spec;
640
+ if (specPath.endsWith('.yaml') || specPath.endsWith('.yml')) {
641
+ const yaml = await Promise.resolve().then(() => __importStar(require('js-yaml')));
642
+ spec = yaml.load(content);
643
+ }
644
+ else {
645
+ spec = JSON.parse(content);
646
+ }
647
+ const info = (spec.info || {});
648
+ const servers = (spec.servers || []);
649
+ const defaultBaseUrl = servers[0]?.url || 'https://api.example.com';
650
+ // Create package directory
651
+ const pkgDir = path_1.default.join(outputPath, packageName);
652
+ await fs_extra_1.default.ensureDir(pkgDir);
653
+ // Generate setup.py
654
+ const setupPy = `from setuptools import setup, find_packages
655
+
656
+ setup(
657
+ name="${packageName}",
658
+ version="${info.version || '1.0.0'}",
659
+ description="${info.description || 'Generated SDK'}",
660
+ packages=find_packages(),
661
+ python_requires=">=3.8",
662
+ install_requires=[
663
+ "httpx>=0.24.0",
664
+ ],
665
+ )
666
+ `;
667
+ await fs_extra_1.default.writeFile(path_1.default.join(outputPath, 'setup.py'), setupPy);
668
+ // Generate __init__.py
669
+ const initPy = `"""${info.title || 'API'} SDK"""
670
+
671
+ from .client import ApiClient
672
+
673
+ __all__ = ["ApiClient"]
674
+ __version__ = "${info.version || '1.0.0'}"
675
+ `;
676
+ await fs_extra_1.default.writeFile(path_1.default.join(pkgDir, '__init__.py'), initPy);
677
+ // Generate client.py
678
+ const clientPy = `"""API Client"""
679
+
680
+ import httpx
681
+ from typing import Optional, Dict, Any
682
+
683
+
684
+ class ApiClient:
685
+ """${info.title || 'API'} Client"""
686
+
687
+ def __init__(
688
+ self,
689
+ base_url: str = "${defaultBaseUrl}",
690
+ api_key: Optional[str] = None,
691
+ headers: Optional[Dict[str, str]] = None,
692
+ ):
693
+ self.base_url = base_url.rstrip("/")
694
+ self.headers = headers or {}
695
+ self.headers["Content-Type"] = "application/json"
696
+
697
+ if api_key:
698
+ self.headers["Authorization"] = f"Bearer {api_key}"
699
+
700
+ self._client = httpx.Client(headers=self.headers)
701
+
702
+ def _request(
703
+ self,
704
+ method: str,
705
+ path: str,
706
+ params: Optional[Dict[str, Any]] = None,
707
+ json: Optional[Dict[str, Any]] = None,
708
+ ) -> Any:
709
+ url = f"{self.base_url}{path}"
710
+ response = self._client.request(method, url, params=params, json=json)
711
+ response.raise_for_status()
712
+ return response.json()
713
+
714
+ def close(self):
715
+ self._client.close()
716
+
717
+ def __enter__(self):
718
+ return self
719
+
720
+ def __exit__(self, *args):
721
+ self.close()
722
+ `;
723
+ await fs_extra_1.default.writeFile(path_1.default.join(pkgDir, 'client.py'), clientPy);
724
+ // Generate README
725
+ const readme = `# ${packageName}
726
+
727
+ Generated Python SDK for ${info.title || 'API'}.
728
+
729
+ ## Installation
730
+
731
+ \`\`\`bash
732
+ pip install -e .
733
+ \`\`\`
734
+
735
+ ## Usage
736
+
737
+ \`\`\`python
738
+ from ${packageName} import ApiClient
739
+
740
+ client = ApiClient(
741
+ base_url="${defaultBaseUrl}",
742
+ api_key="your-api-key",
743
+ )
744
+ \`\`\`
745
+ `;
746
+ await fs_extra_1.default.writeFile(path_1.default.join(outputPath, 'README.md'), readme);
747
+ }
748
+ /**
749
+ * Generate Go SDK
750
+ */
751
+ async function generateGoSDK(specPath, outputPath, packageName) {
752
+ // Read the OpenAPI spec
753
+ const content = await fs_extra_1.default.readFile(specPath, 'utf-8');
754
+ let spec;
755
+ if (specPath.endsWith('.yaml') || specPath.endsWith('.yml')) {
756
+ const yaml = await Promise.resolve().then(() => __importStar(require('js-yaml')));
757
+ spec = yaml.load(content);
758
+ }
759
+ else {
760
+ spec = JSON.parse(content);
761
+ }
762
+ const info = (spec.info || {});
763
+ const servers = (spec.servers || []);
764
+ const defaultBaseUrl = servers[0]?.url || 'https://api.example.com';
765
+ // Generate go.mod
766
+ const goMod = `module ${packageName}
767
+
768
+ go 1.21
769
+ `;
770
+ await fs_extra_1.default.writeFile(path_1.default.join(outputPath, 'go.mod'), goMod);
771
+ // Generate client.go
772
+ const clientGo = `// Package ${packageName.split('/').pop()} provides a client for ${info.title || 'the API'}.
773
+ package ${packageName.split('/').pop() || 'sdk'}
774
+
775
+ import (
776
+ \t"bytes"
777
+ \t"encoding/json"
778
+ \t"fmt"
779
+ \t"io"
780
+ \t"net/http"
781
+ )
782
+
783
+ // Client represents an API client.
784
+ type Client struct {
785
+ \tBaseURL string
786
+ \tAPIKey string
787
+ \tHTTPClient *http.Client
788
+ }
789
+
790
+ // NewClient creates a new API client.
791
+ func NewClient(baseURL, apiKey string) *Client {
792
+ \tif baseURL == "" {
793
+ \t\tbaseURL = "${defaultBaseUrl}"
794
+ \t}
795
+ \treturn &Client{
796
+ \t\tBaseURL: baseURL,
797
+ \t\tAPIKey: apiKey,
798
+ \t\tHTTPClient: &http.Client{},
799
+ \t}
800
+ }
801
+
802
+ func (c *Client) doRequest(method, path string, body interface{}) ([]byte, error) {
803
+ \turl := c.BaseURL + path
804
+
805
+ \tvar reqBody io.Reader
806
+ \tif body != nil {
807
+ \t\tjsonBody, err := json.Marshal(body)
808
+ \t\tif err != nil {
809
+ \t\t\treturn nil, fmt.Errorf("failed to marshal request body: %w", err)
810
+ \t\t}
811
+ \t\treqBody = bytes.NewBuffer(jsonBody)
812
+ \t}
813
+
814
+ \treq, err := http.NewRequest(method, url, reqBody)
815
+ \tif err != nil {
816
+ \t\treturn nil, fmt.Errorf("failed to create request: %w", err)
817
+ \t}
818
+
819
+ \treq.Header.Set("Content-Type", "application/json")
820
+ \tif c.APIKey != "" {
821
+ \t\treq.Header.Set("Authorization", "Bearer "+c.APIKey)
822
+ \t}
823
+
824
+ \tresp, err := c.HTTPClient.Do(req)
825
+ \tif err != nil {
826
+ \t\treturn nil, fmt.Errorf("request failed: %w", err)
827
+ \t}
828
+ \tdefer resp.Body.Close()
829
+
830
+ \trespBody, err := io.ReadAll(resp.Body)
831
+ \tif err != nil {
832
+ \t\treturn nil, fmt.Errorf("failed to read response: %w", err)
833
+ \t}
834
+
835
+ \tif resp.StatusCode >= 400 {
836
+ \t\treturn nil, fmt.Errorf("API error %d: %s", resp.StatusCode, string(respBody))
837
+ \t}
838
+
839
+ \treturn respBody, nil
840
+ }
841
+ `;
842
+ await fs_extra_1.default.writeFile(path_1.default.join(outputPath, 'client.go'), clientGo);
843
+ // Generate README
844
+ const readme = `# ${packageName}
845
+
846
+ Generated Go SDK for ${info.title || 'API'}.
847
+
848
+ ## Installation
849
+
850
+ \`\`\`bash
851
+ go get ${packageName}
852
+ \`\`\`
853
+
854
+ ## Usage
855
+
856
+ \`\`\`go
857
+ package main
858
+
859
+ import (
860
+ \t"${packageName}"
861
+ )
862
+
863
+ func main() {
864
+ \tclient := ${packageName.split('/').pop()}.NewClient("${defaultBaseUrl}", "your-api-key")
865
+ \t// Use client...
866
+ }
867
+ \`\`\`
868
+ `;
869
+ await fs_extra_1.default.writeFile(path_1.default.join(outputPath, 'README.md'), readme);
870
+ }
871
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/sdk/index.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * SDK Generation Module\n *\n * This module provides functionality for generating client SDKs from OpenAPI specifications.\n * It supports multiple languages using both native generators and OpenAPI Generator.\n */\n\nimport path from 'path';\nimport fs from 'fs-extra';\nimport { exec, spawn } from 'child_process';\nimport { promisify } from 'util';\n\nconst execAsync = promisify(exec);\n\n// ============================================================================\n// Types\n// ============================================================================\n\n/**\n * Supported SDK languages\n */\nexport type SDKLanguage =\n  | 'typescript'\n  | 'python'\n  | 'go'\n  | 'java'\n  | 'csharp'\n  | 'ruby'\n  | 'php'\n  | 'swift'\n  | 'kotlin'\n  | 'rust';\n\n/**\n * SDK configuration file (sdk.json)\n */\nexport interface SDKConfig {\n  openapi: string;\n  packageName: string;\n  output: string;\n  languages: {\n    [key in SDKLanguage]?: LanguageConfig;\n  };\n}\n\n/**\n * Language-specific configuration\n */\nexport interface LanguageConfig {\n  enabled: boolean;\n  packageName?: string;\n  output?: string;\n  additionalOptions?: Record<string, string>;\n}\n\n/**\n * Generator information\n */\nexport interface GeneratorInfo {\n  language: SDKLanguage;\n  displayName: string;\n  native: boolean;\n  requiresJava: boolean;\n  generatorName?: string;\n}\n\n/**\n * SDK generation result for a single language\n */\nexport interface SDKGenerationResult {\n  language: SDKLanguage;\n  success: boolean;\n  outputPath: string;\n  error?: string;\n  duration: number;\n}\n\n/**\n * Overall SDK generation result\n */\nexport interface GenerateSDKsResult {\n  results: SDKGenerationResult[];\n  successCount: number;\n  failureCount: number;\n  totalDuration: number;\n}\n\n/**\n * Validation result\n */\nexport interface ValidationResult {\n  valid: boolean;\n  errors: string[];\n  warnings: string[];\n}\n\n/**\n * Options for SDK generation\n */\nexport interface GenerateOptions {\n  language?: SDKLanguage;\n  preview?: boolean;\n  outputDir?: string;\n}\n\n/**\n * Options for SDK config initialization\n */\nexport interface InitConfigOptions {\n  openapi: string;\n  packageName: string;\n  languages: SDKLanguage[];\n  force?: boolean;\n}\n\n// ============================================================================\n// Constants\n// ============================================================================\n\n/**\n * All supported languages\n */\nexport const SUPPORTED_LANGUAGES: SDKLanguage[] = [\n  'typescript',\n  'python',\n  'go',\n  'java',\n  'csharp',\n  'ruby',\n  'php',\n  'swift',\n  'kotlin',\n  'rust',\n];\n\n/**\n * Generator information for each language\n */\nconst GENERATORS: GeneratorInfo[] = [\n  { language: 'typescript', displayName: 'TypeScript', native: true, requiresJava: false },\n  { language: 'python', displayName: 'Python', native: true, requiresJava: false },\n  { language: 'go', displayName: 'Go', native: true, requiresJava: false },\n  { language: 'java', displayName: 'Java', native: false, requiresJava: true, generatorName: 'java' },\n  { language: 'csharp', displayName: 'C#', native: false, requiresJava: true, generatorName: 'csharp' },\n  { language: 'ruby', displayName: 'Ruby', native: false, requiresJava: true, generatorName: 'ruby' },\n  { language: 'php', displayName: 'PHP', native: false, requiresJava: true, generatorName: 'php' },\n  { language: 'swift', displayName: 'Swift', native: false, requiresJava: true, generatorName: 'swift5' },\n  { language: 'kotlin', displayName: 'Kotlin', native: false, requiresJava: true, generatorName: 'kotlin' },\n  { language: 'rust', displayName: 'Rust', native: false, requiresJava: true, generatorName: 'rust' },\n];\n\nconst CONFIG_FILENAME = 'sdk.json';\n\n// ============================================================================\n// Configuration Functions\n// ============================================================================\n\n/**\n * Load SDK configuration from project directory\n */\nexport async function loadSDKConfig(\n  projectRoot: string\n): Promise<{ config: SDKConfig | null; error?: string }> {\n  const configPath = path.join(projectRoot, CONFIG_FILENAME);\n\n  if (!(await fs.pathExists(configPath))) {\n    return { config: null, error: `${CONFIG_FILENAME} not found` };\n  }\n\n  try {\n    const config = await fs.readJson(configPath);\n    return { config };\n  } catch (error) {\n    return {\n      config: null,\n      error: `Failed to parse ${CONFIG_FILENAME}: ${error instanceof Error ? error.message : String(error)}`,\n    };\n  }\n}\n\n/**\n * Initialize SDK configuration\n */\nexport async function initSDKConfig(\n  projectRoot: string,\n  options: InitConfigOptions\n): Promise<{ configPath: string; config: SDKConfig }> {\n  const configPath = path.join(projectRoot, CONFIG_FILENAME);\n\n  // Check if config already exists\n  if ((await fs.pathExists(configPath)) && !options.force) {\n    throw new Error(`${CONFIG_FILENAME} already exists. Use --force to overwrite.`);\n  }\n\n  // Create language configurations\n  const languages: SDKConfig['languages'] = {};\n  for (const lang of options.languages) {\n    languages[lang] = {\n      enabled: true,\n      packageName: getDefaultPackageName(options.packageName, lang),\n    };\n  }\n\n  const config: SDKConfig = {\n    openapi: options.openapi,\n    packageName: options.packageName,\n    output: './sdks',\n    languages,\n  };\n\n  await fs.writeJson(configPath, config, { spaces: 2 });\n\n  return { configPath, config };\n}\n\n/**\n * Get enabled generators from config\n */\nexport function getEnabledGenerators(config: SDKConfig): SDKLanguage[] {\n  return Object.entries(config.languages)\n    .filter(([, langConfig]) => langConfig?.enabled)\n    .map(([lang]) => lang as SDKLanguage);\n}\n\n/**\n * Get generator information for all languages\n */\nexport function getGeneratorInfo(): GeneratorInfo[] {\n  return [...GENERATORS];\n}\n\n/**\n * Check if a generator is native (doesn't require Java)\n */\nexport function isNativeGenerator(language: SDKLanguage): boolean {\n  const generator = GENERATORS.find((g) => g.language === language);\n  return generator?.native ?? false;\n}\n\n// ============================================================================\n// Path Utilities\n// ============================================================================\n\n/**\n * Resolve OpenAPI spec path relative to project root\n */\nexport function resolveSpecPath(projectRoot: string, specPath: string): string {\n  if (path.isAbsolute(specPath)) {\n    return specPath;\n  }\n  return path.resolve(projectRoot, specPath);\n}\n\n/**\n * Get default package name for a language\n */\nfunction getDefaultPackageName(baseName: string, language: SDKLanguage): string {\n  const sanitized = baseName.toLowerCase().replace(/[^a-z0-9]/g, '-');\n\n  switch (language) {\n    case 'typescript':\n      return `@${sanitized}/sdk`;\n    case 'python':\n      return sanitized.replace(/-/g, '_');\n    case 'go':\n      return sanitized;\n    case 'java':\n      return `com.${sanitized.replace(/-/g, '.')}`;\n    case 'csharp':\n      return sanitized.split('-').map(capitalize).join('.');\n    case 'ruby':\n      return sanitized.replace(/-/g, '_');\n    case 'php':\n      return sanitized.split('-').map(capitalize).join('\\\\');\n    case 'swift':\n      return sanitized.split('-').map(capitalize).join('');\n    case 'kotlin':\n      return `com.${sanitized.replace(/-/g, '.')}`;\n    case 'rust':\n      return sanitized.replace(/-/g, '_');\n    default:\n      return sanitized;\n  }\n}\n\nfunction capitalize(str: string): string {\n  return str.charAt(0).toUpperCase() + str.slice(1);\n}\n\n// ============================================================================\n// Validation\n// ============================================================================\n\n/**\n * Validate an OpenAPI specification\n */\nexport async function validateSpec(specPath: string): Promise<ValidationResult> {\n  const errors: string[] = [];\n  const warnings: string[] = [];\n\n  // Check file exists\n  if (!(await fs.pathExists(specPath))) {\n    return {\n      valid: false,\n      errors: [`OpenAPI specification not found: ${specPath}`],\n      warnings: [],\n    };\n  }\n\n  try {\n    // Read and parse the spec\n    const content = await fs.readFile(specPath, 'utf-8');\n    let spec: Record<string, unknown>;\n\n    if (specPath.endsWith('.yaml') || specPath.endsWith('.yml')) {\n      const yaml = await import('js-yaml');\n      spec = yaml.load(content) as Record<string, unknown>;\n    } else {\n      spec = JSON.parse(content);\n    }\n\n    // Basic validation\n    if (!spec.openapi && !spec.swagger) {\n      errors.push('Missing \"openapi\" or \"swagger\" version field');\n    }\n\n    if (!spec.info) {\n      errors.push('Missing \"info\" section');\n    } else {\n      const info = spec.info as Record<string, unknown>;\n      if (!info.title) {\n        warnings.push('Missing \"info.title\"');\n      }\n      if (!info.version) {\n        warnings.push('Missing \"info.version\"');\n      }\n    }\n\n    if (!spec.paths || Object.keys(spec.paths as object).length === 0) {\n      warnings.push('No paths defined in specification');\n    }\n\n    // Check for common issues\n    const openapiVersion = spec.openapi as string;\n    if (openapiVersion && !openapiVersion.startsWith('3.')) {\n      warnings.push(`OpenAPI ${openapiVersion} detected. Version 3.x is recommended.`);\n    }\n\n    return {\n      valid: errors.length === 0,\n      errors,\n      warnings,\n    };\n  } catch (error) {\n    return {\n      valid: false,\n      errors: [`Failed to parse specification: ${error instanceof Error ? error.message : String(error)}`],\n      warnings: [],\n    };\n  }\n}\n\n// ============================================================================\n// SDK Generation\n// ============================================================================\n\n/**\n * Generate SDKs for configured languages\n */\nexport async function generateSDKs(\n  projectRoot: string,\n  options: GenerateOptions = {}\n): Promise<GenerateSDKsResult> {\n  const startTime = Date.now();\n  const results: SDKGenerationResult[] = [];\n\n  // Load config\n  const { config, error } = await loadSDKConfig(projectRoot);\n  if (!config) {\n    throw new Error(error || 'SDK configuration not found');\n  }\n\n  // Determine which languages to generate\n  let languagesToGenerate = getEnabledGenerators(config);\n\n  if (options.language) {\n    if (!languagesToGenerate.includes(options.language)) {\n      throw new Error(`Language \"${options.language}\" is not enabled in sdk.json`);\n    }\n    languagesToGenerate = [options.language];\n  }\n\n  // Resolve spec path\n  const specPath = resolveSpecPath(projectRoot, config.openapi);\n\n  // Validate spec exists\n  if (!(await fs.pathExists(specPath))) {\n    throw new Error(`OpenAPI specification not found: ${config.openapi}`);\n  }\n\n  // Base output directory\n  const baseOutputDir = options.outputDir\n    ? path.resolve(projectRoot, options.outputDir)\n    : path.resolve(projectRoot, config.output);\n\n  // Generate SDKs for each language\n  for (const language of languagesToGenerate) {\n    const langStart = Date.now();\n    const langConfig = config.languages[language];\n    const outputPath = langConfig?.output\n      ? path.resolve(projectRoot, langConfig.output)\n      : path.join(baseOutputDir, language);\n\n    try {\n      if (options.preview) {\n        // Preview mode - just report what would be generated\n        results.push({\n          language,\n          success: true,\n          outputPath,\n          duration: Date.now() - langStart,\n        });\n        continue;\n      }\n\n      // Ensure output directory exists\n      await fs.ensureDir(outputPath);\n\n      // Generate SDK based on generator type\n      const generator = GENERATORS.find((g) => g.language === language);\n\n      if (generator?.native) {\n        await generateNativeSDK(language, specPath, outputPath, config, langConfig);\n      } else if (generator?.generatorName) {\n        await generateOpenAPIGeneratorSDK(generator.generatorName, specPath, outputPath, config, langConfig);\n      } else {\n        throw new Error(`No generator found for ${language}`);\n      }\n\n      results.push({\n        language,\n        success: true,\n        outputPath,\n        duration: Date.now() - langStart,\n      });\n    } catch (err) {\n      results.push({\n        language,\n        success: false,\n        outputPath,\n        error: err instanceof Error ? err.message : String(err),\n        duration: Date.now() - langStart,\n      });\n    }\n  }\n\n  return {\n    results,\n    successCount: results.filter((r) => r.success).length,\n    failureCount: results.filter((r) => !r.success).length,\n    totalDuration: Date.now() - startTime,\n  };\n}\n\n/**\n * Generate SDK using native generator (TypeScript, Python, Go)\n */\nasync function generateNativeSDK(\n  language: SDKLanguage,\n  specPath: string,\n  outputPath: string,\n  config: SDKConfig,\n  langConfig?: LanguageConfig\n): Promise<void> {\n  const packageName = langConfig?.packageName || getDefaultPackageName(config.packageName, language);\n\n  switch (language) {\n    case 'typescript':\n      await generateTypeScriptSDK(specPath, outputPath, packageName);\n      break;\n    case 'python':\n      await generatePythonSDK(specPath, outputPath, packageName);\n      break;\n    case 'go':\n      await generateGoSDK(specPath, outputPath, packageName);\n      break;\n    default:\n      throw new Error(`Native generator not implemented for ${language}`);\n  }\n}\n\n/**\n * Generate SDK using OpenAPI Generator (Java-based)\n */\nasync function generateOpenAPIGeneratorSDK(\n  generatorName: string,\n  specPath: string,\n  outputPath: string,\n  config: SDKConfig,\n  langConfig?: LanguageConfig\n): Promise<void> {\n  // Check if Java is available\n  try {\n    await execAsync('java -version');\n  } catch {\n    throw new Error('Java is required for this generator. Please install Java 11 or later.');\n  }\n\n  // Check if openapi-generator is available\n  let generatorCmd = 'openapi-generator';\n  try {\n    await execAsync('openapi-generator version');\n  } catch {\n    // Try npx fallback\n    generatorCmd = 'npx @openapitools/openapi-generator-cli';\n  }\n\n  const additionalProps = langConfig?.additionalOptions\n    ? Object.entries(langConfig.additionalOptions)\n        .map(([k, v]) => `${k}=${v}`)\n        .join(',')\n    : '';\n\n  const cmd = [\n    generatorCmd,\n    'generate',\n    '-i',\n    specPath,\n    '-g',\n    generatorName,\n    '-o',\n    outputPath,\n    additionalProps ? `--additional-properties=${additionalProps}` : '',\n  ]\n    .filter(Boolean)\n    .join(' ');\n\n  await execAsync(cmd);\n}\n\n// ============================================================================\n// Native Generator Implementations\n// ============================================================================\n\n/**\n * Generate TypeScript SDK\n */\nasync function generateTypeScriptSDK(\n  specPath: string,\n  outputPath: string,\n  packageName: string\n): Promise<void> {\n  // Read the OpenAPI spec\n  const content = await fs.readFile(specPath, 'utf-8');\n  let spec: Record<string, unknown>;\n\n  if (specPath.endsWith('.yaml') || specPath.endsWith('.yml')) {\n    const yaml = await import('js-yaml');\n    spec = yaml.load(content) as Record<string, unknown>;\n  } else {\n    spec = JSON.parse(content);\n  }\n\n  const info = (spec.info || {}) as Record<string, unknown>;\n  const paths = (spec.paths || {}) as Record<string, unknown>;\n\n  // Generate package.json\n  const packageJson = {\n    name: packageName,\n    version: (info.version as string) || '1.0.0',\n    description: (info.description as string) || 'Generated SDK',\n    main: 'dist/index.js',\n    types: 'dist/index.d.ts',\n    scripts: {\n      build: 'tsc',\n      prepublishOnly: 'npm run build',\n    },\n    devDependencies: {\n      typescript: '^5.0.0',\n      '@types/node': '^20.0.0',\n    },\n  };\n\n  await fs.writeJson(path.join(outputPath, 'package.json'), packageJson, { spaces: 2 });\n\n  // Generate tsconfig.json\n  const tsConfig = {\n    compilerOptions: {\n      target: 'ES2020',\n      module: 'commonjs',\n      declaration: true,\n      outDir: './dist',\n      strict: true,\n      esModuleInterop: true,\n      skipLibCheck: true,\n      forceConsistentCasingInFileNames: true,\n    },\n    include: ['src/**/*'],\n    exclude: ['node_modules', 'dist'],\n  };\n\n  await fs.writeJson(path.join(outputPath, 'tsconfig.json'), tsConfig, { spaces: 2 });\n\n  // Generate source files\n  await fs.ensureDir(path.join(outputPath, 'src'));\n\n  // Generate types\n  const typesContent = generateTypeScriptTypes(spec);\n  await fs.writeFile(path.join(outputPath, 'src', 'types.ts'), typesContent);\n\n  // Generate client\n  const clientContent = generateTypeScriptClient(spec, paths);\n  await fs.writeFile(path.join(outputPath, 'src', 'client.ts'), clientContent);\n\n  // Generate index\n  const indexContent = `export * from './types';\\nexport * from './client';\\n`;\n  await fs.writeFile(path.join(outputPath, 'src', 'index.ts'), indexContent);\n\n  // Generate README\n  const readmeContent = `# ${packageName}\n\nGenerated TypeScript SDK for ${(info.title as string) || 'API'}.\n\n## Installation\n\n\\`\\`\\`bash\nnpm install ${packageName}\n\\`\\`\\`\n\n## Usage\n\n\\`\\`\\`typescript\nimport { ApiClient } from '${packageName}';\n\nconst client = new ApiClient({\n  baseUrl: 'https://api.example.com',\n  // apiKey: 'your-api-key',\n});\n\\`\\`\\`\n`;\n\n  await fs.writeFile(path.join(outputPath, 'README.md'), readmeContent);\n}\n\nfunction generateTypeScriptTypes(spec: Record<string, unknown>): string {\n  let content = '/**\\n * Generated types from OpenAPI specification\\n */\\n\\n';\n\n  const schemas = ((spec.components as Record<string, unknown>)?.schemas || {}) as Record<string, unknown>;\n\n  // If no schemas, export a placeholder to make this a valid module\n  if (Object.keys(schemas).length === 0) {\n    content += '// No schemas defined in the OpenAPI specification\\n';\n    content += 'export type ApiResponse<T> = T;\\n';\n    content += 'export type ApiError = { message: string; code?: number };\\n';\n    return content;\n  }\n\n  for (const [name, schema] of Object.entries(schemas)) {\n    content += `export interface ${name} {\\n`;\n    const properties = (schema as Record<string, unknown>).properties as Record<string, unknown> | undefined;\n    const required = ((schema as Record<string, unknown>).required || []) as string[];\n\n    if (properties) {\n      for (const [propName, propSchema] of Object.entries(properties)) {\n        const isRequired = required.includes(propName);\n        const tsType = openApiTypeToTs(propSchema as Record<string, unknown>);\n        content += `  ${propName}${isRequired ? '' : '?'}: ${tsType};\\n`;\n      }\n    }\n\n    content += '}\\n\\n';\n  }\n\n  return content;\n}\n\nfunction openApiTypeToTs(schema: Record<string, unknown>): string {\n  const type = schema.type as string;\n  const format = schema.format as string;\n  const ref = schema.$ref as string;\n\n  if (ref) {\n    return ref.split('/').pop() || 'unknown';\n  }\n\n  switch (type) {\n    case 'string':\n      return 'string';\n    case 'integer':\n    case 'number':\n      return 'number';\n    case 'boolean':\n      return 'boolean';\n    case 'array':\n      const items = schema.items as Record<string, unknown>;\n      return `${openApiTypeToTs(items || {})}[]`;\n    case 'object':\n      return 'Record<string, unknown>';\n    default:\n      return 'unknown';\n  }\n}\n\nfunction generateTypeScriptClient(spec: Record<string, unknown>, paths: Record<string, unknown>): string {\n  const info = (spec.info || {}) as Record<string, unknown>;\n  const servers = (spec.servers || []) as { url: string }[];\n  const defaultBaseUrl = servers[0]?.url || 'https://api.example.com';\n\n  let content = `/**\n * ${(info.title as string) || 'API'} Client\n * ${(info.description as string) || ''}\n */\n\nexport interface ClientConfig {\n  baseUrl?: string;\n  apiKey?: string;\n  headers?: Record<string, string>;\n}\n\nexport class ApiClient {\n  private baseUrl: string;\n  private headers: Record<string, string>;\n\n  constructor(config: ClientConfig = {}) {\n    this.baseUrl = config.baseUrl || '${defaultBaseUrl}';\n    this.headers = {\n      'Content-Type': 'application/json',\n      ...config.headers,\n    };\n\n    if (config.apiKey) {\n      this.headers['Authorization'] = \\`Bearer \\${config.apiKey}\\`;\n    }\n  }\n\n  private async request<T>(\n    method: string,\n    path: string,\n    options: { body?: unknown; query?: Record<string, string> } = {}\n  ): Promise<T> {\n    let url = \\`\\${this.baseUrl}\\${path}\\`;\n\n    if (options.query) {\n      const params = new URLSearchParams(options.query);\n      url += \\`?\\${params.toString()}\\`;\n    }\n\n    const response = await fetch(url, {\n      method,\n      headers: this.headers,\n      body: options.body ? JSON.stringify(options.body) : undefined,\n    });\n\n    if (!response.ok) {\n      throw new Error(\\`HTTP \\${response.status}: \\${response.statusText}\\`);\n    }\n\n    return response.json();\n  }\n`;\n\n  // Generate methods for each path\n  for (const [pathUrl, pathItem] of Object.entries(paths)) {\n    const methods = ['get', 'post', 'put', 'patch', 'delete'] as const;\n\n    for (const method of methods) {\n      const operation = (pathItem as Record<string, unknown>)[method] as Record<string, unknown> | undefined;\n      if (!operation) continue;\n\n      const operationId = (operation.operationId as string) || `${method}${pathUrl.replace(/[^a-zA-Z0-9]/g, '_')}`;\n      const summary = (operation.summary as string) || '';\n\n      content += `\n  /**\n   * ${summary}\n   */\n  async ${operationId}(): Promise<unknown> {\n    return this.request('${method.toUpperCase()}', '${pathUrl}');\n  }\n`;\n    }\n  }\n\n  content += '}\\n';\n\n  return content;\n}\n\n/**\n * Generate Python SDK\n */\nasync function generatePythonSDK(\n  specPath: string,\n  outputPath: string,\n  packageName: string\n): Promise<void> {\n  // Read the OpenAPI spec\n  const content = await fs.readFile(specPath, 'utf-8');\n  let spec: Record<string, unknown>;\n\n  if (specPath.endsWith('.yaml') || specPath.endsWith('.yml')) {\n    const yaml = await import('js-yaml');\n    spec = yaml.load(content) as Record<string, unknown>;\n  } else {\n    spec = JSON.parse(content);\n  }\n\n  const info = (spec.info || {}) as Record<string, unknown>;\n  const servers = (spec.servers || []) as { url: string }[];\n  const defaultBaseUrl = servers[0]?.url || 'https://api.example.com';\n\n  // Create package directory\n  const pkgDir = path.join(outputPath, packageName);\n  await fs.ensureDir(pkgDir);\n\n  // Generate setup.py\n  const setupPy = `from setuptools import setup, find_packages\n\nsetup(\n    name=\"${packageName}\",\n    version=\"${(info.version as string) || '1.0.0'}\",\n    description=\"${(info.description as string) || 'Generated SDK'}\",\n    packages=find_packages(),\n    python_requires=\">=3.8\",\n    install_requires=[\n        \"httpx>=0.24.0\",\n    ],\n)\n`;\n\n  await fs.writeFile(path.join(outputPath, 'setup.py'), setupPy);\n\n  // Generate __init__.py\n  const initPy = `\"\"\"${(info.title as string) || 'API'} SDK\"\"\"\n\nfrom .client import ApiClient\n\n__all__ = [\"ApiClient\"]\n__version__ = \"${(info.version as string) || '1.0.0'}\"\n`;\n\n  await fs.writeFile(path.join(pkgDir, '__init__.py'), initPy);\n\n  // Generate client.py\n  const clientPy = `\"\"\"API Client\"\"\"\n\nimport httpx\nfrom typing import Optional, Dict, Any\n\n\nclass ApiClient:\n    \"\"\"${(info.title as string) || 'API'} Client\"\"\"\n\n    def __init__(\n        self,\n        base_url: str = \"${defaultBaseUrl}\",\n        api_key: Optional[str] = None,\n        headers: Optional[Dict[str, str]] = None,\n    ):\n        self.base_url = base_url.rstrip(\"/\")\n        self.headers = headers or {}\n        self.headers[\"Content-Type\"] = \"application/json\"\n\n        if api_key:\n            self.headers[\"Authorization\"] = f\"Bearer {api_key}\"\n\n        self._client = httpx.Client(headers=self.headers)\n\n    def _request(\n        self,\n        method: str,\n        path: str,\n        params: Optional[Dict[str, Any]] = None,\n        json: Optional[Dict[str, Any]] = None,\n    ) -> Any:\n        url = f\"{self.base_url}{path}\"\n        response = self._client.request(method, url, params=params, json=json)\n        response.raise_for_status()\n        return response.json()\n\n    def close(self):\n        self._client.close()\n\n    def __enter__(self):\n        return self\n\n    def __exit__(self, *args):\n        self.close()\n`;\n\n  await fs.writeFile(path.join(pkgDir, 'client.py'), clientPy);\n\n  // Generate README\n  const readme = `# ${packageName}\n\nGenerated Python SDK for ${(info.title as string) || 'API'}.\n\n## Installation\n\n\\`\\`\\`bash\npip install -e .\n\\`\\`\\`\n\n## Usage\n\n\\`\\`\\`python\nfrom ${packageName} import ApiClient\n\nclient = ApiClient(\n    base_url=\"${defaultBaseUrl}\",\n    api_key=\"your-api-key\",\n)\n\\`\\`\\`\n`;\n\n  await fs.writeFile(path.join(outputPath, 'README.md'), readme);\n}\n\n/**\n * Generate Go SDK\n */\nasync function generateGoSDK(\n  specPath: string,\n  outputPath: string,\n  packageName: string\n): Promise<void> {\n  // Read the OpenAPI spec\n  const content = await fs.readFile(specPath, 'utf-8');\n  let spec: Record<string, unknown>;\n\n  if (specPath.endsWith('.yaml') || specPath.endsWith('.yml')) {\n    const yaml = await import('js-yaml');\n    spec = yaml.load(content) as Record<string, unknown>;\n  } else {\n    spec = JSON.parse(content);\n  }\n\n  const info = (spec.info || {}) as Record<string, unknown>;\n  const servers = (spec.servers || []) as { url: string }[];\n  const defaultBaseUrl = servers[0]?.url || 'https://api.example.com';\n\n  // Generate go.mod\n  const goMod = `module ${packageName}\n\ngo 1.21\n`;\n\n  await fs.writeFile(path.join(outputPath, 'go.mod'), goMod);\n\n  // Generate client.go\n  const clientGo = `// Package ${packageName.split('/').pop()} provides a client for ${(info.title as string) || 'the API'}.\npackage ${packageName.split('/').pop() || 'sdk'}\n\nimport (\n\\t\"bytes\"\n\\t\"encoding/json\"\n\\t\"fmt\"\n\\t\"io\"\n\\t\"net/http\"\n)\n\n// Client represents an API client.\ntype Client struct {\n\\tBaseURL    string\n\\tAPIKey     string\n\\tHTTPClient *http.Client\n}\n\n// NewClient creates a new API client.\nfunc NewClient(baseURL, apiKey string) *Client {\n\\tif baseURL == \"\" {\n\\t\\tbaseURL = \"${defaultBaseUrl}\"\n\\t}\n\\treturn &Client{\n\\t\\tBaseURL:    baseURL,\n\\t\\tAPIKey:     apiKey,\n\\t\\tHTTPClient: &http.Client{},\n\\t}\n}\n\nfunc (c *Client) doRequest(method, path string, body interface{}) ([]byte, error) {\n\\turl := c.BaseURL + path\n\n\\tvar reqBody io.Reader\n\\tif body != nil {\n\\t\\tjsonBody, err := json.Marshal(body)\n\\t\\tif err != nil {\n\\t\\t\\treturn nil, fmt.Errorf(\"failed to marshal request body: %w\", err)\n\\t\\t}\n\\t\\treqBody = bytes.NewBuffer(jsonBody)\n\\t}\n\n\\treq, err := http.NewRequest(method, url, reqBody)\n\\tif err != nil {\n\\t\\treturn nil, fmt.Errorf(\"failed to create request: %w\", err)\n\\t}\n\n\\treq.Header.Set(\"Content-Type\", \"application/json\")\n\\tif c.APIKey != \"\" {\n\\t\\treq.Header.Set(\"Authorization\", \"Bearer \"+c.APIKey)\n\\t}\n\n\\tresp, err := c.HTTPClient.Do(req)\n\\tif err != nil {\n\\t\\treturn nil, fmt.Errorf(\"request failed: %w\", err)\n\\t}\n\\tdefer resp.Body.Close()\n\n\\trespBody, err := io.ReadAll(resp.Body)\n\\tif err != nil {\n\\t\\treturn nil, fmt.Errorf(\"failed to read response: %w\", err)\n\\t}\n\n\\tif resp.StatusCode >= 400 {\n\\t\\treturn nil, fmt.Errorf(\"API error %d: %s\", resp.StatusCode, string(respBody))\n\\t}\n\n\\treturn respBody, nil\n}\n`;\n\n  await fs.writeFile(path.join(outputPath, 'client.go'), clientGo);\n\n  // Generate README\n  const readme = `# ${packageName}\n\nGenerated Go SDK for ${(info.title as string) || 'API'}.\n\n## Installation\n\n\\`\\`\\`bash\ngo get ${packageName}\n\\`\\`\\`\n\n## Usage\n\n\\`\\`\\`go\npackage main\n\nimport (\n\\t\"${packageName}\"\n)\n\nfunc main() {\n\\tclient := ${packageName.split('/').pop()}.NewClient(\"${defaultBaseUrl}\", \"your-api-key\")\n\\t// Use client...\n}\n\\`\\`\\`\n`;\n\n  await fs.writeFile(path.join(outputPath, 'README.md'), readme);\n}\n"]}