@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.
- package/README.md +45 -2
- package/dist/cli/commands/create.js +25 -8
- package/dist/cli/commands/sdk.d.ts +30 -0
- package/dist/cli/commands/sdk.js +365 -0
- package/dist/cli/index.js +28 -1
- package/dist/sdk/index.d.ts +126 -0
- package/dist/sdk/index.js +871 -0
- package/package.json +7 -3
- package/renderer/app/api/collections/route.js +38 -9
- package/renderer/components/docs-viewer/agent/agent-chat.js +146 -109
- package/renderer/components/docs-viewer/index.js +21 -3
|
@@ -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"]}
|