@gopowerteam/request-generate 0.2.2 → 0.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (37) hide show
  1. package/bin/download.mts +2 -2
  2. package/bin/generate.mts +1 -1
  3. package/bin/index.mts +2 -2
  4. package/dist/chunk-C0xms8kb.cjs +34 -0
  5. package/dist/index.cjs +1105 -0
  6. package/dist/index.d.cts +739 -0
  7. package/dist/index.d.mts +739 -0
  8. package/dist/index.mjs +937 -1040
  9. package/dist/templates/{partials → templates/partials}/export-service-namespace.hbs +1 -1
  10. package/dist/vite-plugin/index.cjs +238 -0
  11. package/dist/vite-plugin/index.d.cts +13 -0
  12. package/dist/vite-plugin/index.d.mts +14 -0
  13. package/dist/vite-plugin/index.mjs +118 -116
  14. package/package.json +29 -29
  15. package/dist/chunk-WF3XBEPN.mjs +0 -18
  16. package/dist/chunk-XXPGZHWZ.js +0 -10
  17. package/dist/index.d.ts +0 -165
  18. package/dist/index.js +0 -1203
  19. package/dist/vite-plugin/index.d.ts +0 -13
  20. package/dist/vite-plugin/index.js +0 -231
  21. /package/dist/templates/{export-model.hbs → templates/export-model.hbs} +0 -0
  22. /package/dist/templates/{export-service.hbs → templates/export-service.hbs} +0 -0
  23. /package/dist/templates/{partials → templates/partials}/export-description.hbs +0 -0
  24. /package/dist/templates/{partials → templates/partials}/export-header.hbs +0 -0
  25. /package/dist/templates/{partials → templates/partials}/export-model-field.hbs +0 -0
  26. /package/dist/templates/{partials → templates/partials}/export-model-import.hbs +0 -0
  27. /package/dist/templates/{partials → templates/partials}/export-model-type.hbs +0 -0
  28. /package/dist/templates/{partials → templates/partials}/export-operation-params-body.hbs +0 -0
  29. /package/dist/templates/{partials → templates/partials}/export-operation-params-path.hbs +0 -0
  30. /package/dist/templates/{partials → templates/partials}/export-operation-params-query.hbs +0 -0
  31. /package/dist/templates/{partials → templates/partials}/export-operation-response.hbs +0 -0
  32. /package/dist/templates/{partials → templates/partials}/export-schema-type.hbs +0 -0
  33. /package/dist/templates/{partials → templates/partials}/export-service-class.hbs +0 -0
  34. /package/dist/templates/{partials → templates/partials}/export-service-import.hbs +0 -0
  35. /package/dist/templates/{partials → templates/partials}/export-service-namespace-type.hbs +0 -0
  36. /package/dist/templates/{partials → templates/partials}/export-service-operation.hbs +0 -0
  37. /package/dist/templates/{partials → templates/partials}/is-required.hbs +0 -0
package/dist/index.cjs ADDED
@@ -0,0 +1,1105 @@
1
+ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
2
+ const require_chunk = require('./chunk-C0xms8kb.cjs');
3
+ let node_fs = require("node:fs");
4
+ node_fs = require_chunk.__toESM(node_fs);
5
+ let node_path = require("node:path");
6
+ node_path = require_chunk.__toESM(node_path);
7
+ let node_process = require("node:process");
8
+ node_process = require_chunk.__toESM(node_process);
9
+ let chalk = require("chalk");
10
+ chalk = require_chunk.__toESM(chalk);
11
+ let ora = require("ora");
12
+ ora = require_chunk.__toESM(ora);
13
+ let handlebars = require("handlebars");
14
+ handlebars = require_chunk.__toESM(handlebars);
15
+ let _apidevtools_swagger_parser = require("@apidevtools/swagger-parser");
16
+ _apidevtools_swagger_parser = require_chunk.__toESM(_apidevtools_swagger_parser);
17
+ let node_crypto = require("node:crypto");
18
+ node_crypto = require_chunk.__toESM(node_crypto);
19
+ let rimraf = require("rimraf");
20
+ rimraf = require_chunk.__toESM(rimraf);
21
+
22
+ //#region src/utils/get-services-options.ts
23
+ /**
24
+ * 创建单服务配置项
25
+ * @param options
26
+ * @param name
27
+ * @param application
28
+ */
29
+ function createOptions(options, name, application) {
30
+ const { service, openapi } = (() => {
31
+ if (!application) return {
32
+ service: "",
33
+ openapi: options.openapi
34
+ };
35
+ if (typeof application === "string") return {
36
+ service: application,
37
+ openapi: options.openapi
38
+ };
39
+ else return {
40
+ service: application.key,
41
+ openapi: application.openapi
42
+ };
43
+ })();
44
+ return {
45
+ name,
46
+ application: service,
47
+ input: `${options.gateway}/${service}/${openapi}`.replace(/\/{2,3}/g, "/"),
48
+ output: name ? node_path.default.join(options.output, name) : options.output,
49
+ exportModels: options.exportModels
50
+ };
51
+ }
52
+ /**
53
+ * 创建服务项
54
+ * @param options
55
+ * @returns GenerateApplicationOptions[]
56
+ */
57
+ function generateServiceOptions(options) {
58
+ if (options.applications && Object.keys(options.applications).length) return Object.entries(options.applications).map(([name, application]) => createOptions(options, name, application));
59
+ else return [createOptions(options, "default")];
60
+ }
61
+
62
+ //#endregion
63
+ //#region src/download/index.ts
64
+ const DefaultDownloadDir = ".request";
65
+ var Download = class Download {
66
+ static options;
67
+ static async startup(options) {
68
+ const applicationOptions = generateServiceOptions(options);
69
+ if (!node_fs.existsSync(DefaultDownloadDir)) node_fs.mkdirSync(DefaultDownloadDir);
70
+ await Promise.all(applicationOptions.map((options) => Download.downloadOpenAPIFile(options)));
71
+ }
72
+ static async downloadOpenAPIFile(option) {
73
+ const data = await (await fetch(option.input)).json();
74
+ const filePath = node_path.join(".request", `${option.name}.json`);
75
+ node_fs.writeFileSync(filePath, JSON.stringify(data, null, 2));
76
+ return data;
77
+ }
78
+ };
79
+
80
+ //#endregion
81
+ //#region src/config/enum.config.ts
82
+ let OpenAPIVersion = /* @__PURE__ */ function(OpenAPIVersion) {
83
+ OpenAPIVersion[OpenAPIVersion["V2"] = 2] = "V2";
84
+ OpenAPIVersion[OpenAPIVersion["V3"] = 3] = "V3";
85
+ return OpenAPIVersion;
86
+ }({});
87
+
88
+ //#endregion
89
+ //#region src/utils/get-camel-name.ts
90
+ function getCamelName(value) {
91
+ return value.replace(/^[^a-z]+/gi, "").replace(/\W+/g, "_").replace(/[^\w-]+/g, "_").replace(/^\S/, (s) => s.toUpperCase()).replace(/_[a-z]/gi, (s) => s.toUpperCase()).replace(/_/g, "").trim();
92
+ }
93
+
94
+ //#endregion
95
+ //#region src/entities/model.ts
96
+ var Model = class {
97
+ name;
98
+ fields;
99
+ imports;
100
+ constructor(name) {
101
+ this.name = name;
102
+ }
103
+ };
104
+
105
+ //#endregion
106
+ //#region src/entities/field.ts
107
+ var Field = class {
108
+ name;
109
+ required;
110
+ type;
111
+ ref;
112
+ enums;
113
+ description;
114
+ imports;
115
+ constructor(name, required) {
116
+ this.name = name;
117
+ this.required = required;
118
+ }
119
+ };
120
+
121
+ //#endregion
122
+ //#region src/utils/get-built-in-type.ts
123
+ const TYPE_MAPPINGS$1 = new Map([["MapStringObject", "Record<string, any>"]]);
124
+ /**
125
+ * 获取对应类型
126
+ */
127
+ function getBuiltInType(ref) {
128
+ return TYPE_MAPPINGS$1.get(ref);
129
+ }
130
+
131
+ //#endregion
132
+ //#region src/utils/get-mapped-type.ts
133
+ const TYPE_MAPPINGS = new Map([
134
+ ["file", "binary"],
135
+ ["any", "any"],
136
+ ["object", "any"],
137
+ ["array", "any[]"],
138
+ ["boolean", "boolean"],
139
+ ["byte", "number"],
140
+ ["int", "number"],
141
+ ["integer", "number"],
142
+ ["float", "number"],
143
+ ["double", "number"],
144
+ ["short", "number"],
145
+ ["long", "number"],
146
+ ["number", "number"],
147
+ ["char", "string"],
148
+ ["date", "string"],
149
+ ["date-time", "string"],
150
+ ["password", "string"],
151
+ ["string", "string"],
152
+ ["void", "void"],
153
+ ["null", "null"]
154
+ ]);
155
+ /**
156
+ * 获取对应类型
157
+ */
158
+ function getMappedType(type = "object", format) {
159
+ if (format === "binary") return "binary";
160
+ return TYPE_MAPPINGS.get(type) || "any";
161
+ }
162
+
163
+ //#endregion
164
+ //#region src/parse/v2/strip-namespace.ts
165
+ /**
166
+ * Strip (OpenAPI) namespaces fom values.
167
+ * @param value
168
+ */
169
+ function stripNamespace$1(value) {
170
+ return value.trim().replace(/^#\/definitions\//, "").replace(/^#\/parameters\//, "").replace(/^#\/responses\//, "").replace(/^#\/securityDefinitions\//, "");
171
+ }
172
+
173
+ //#endregion
174
+ //#region src/parse/v2/parse-schema-type.ts
175
+ function parseSchemaType$1(schema) {
176
+ if ("$ref" in schema && schema.$ref) {
177
+ const ref = getCamelName(stripNamespace$1(schema.$ref));
178
+ return {
179
+ type: "any",
180
+ ref,
181
+ imports: [ref]
182
+ };
183
+ }
184
+ if (!("$ref" in schema) && schema.type !== "array" && typeof schema.type === "string" && !schema.allOf && !schema.anyOf && !schema.oneOf) return {
185
+ type: getMappedType(schema.type || "any"),
186
+ ref: void 0,
187
+ enums: schema.enum
188
+ };
189
+ if (!("$ref" in schema) && schema.type === "array" && schema.items && "$ref" in schema.items && schema.items.$ref) {
190
+ const ref = getCamelName(stripNamespace$1(schema.items.$ref));
191
+ const type = getBuiltInType(ref);
192
+ return {
193
+ type: "any[]",
194
+ ref: `${type ?? ref}[]`,
195
+ imports: type ? void 0 : [ref]
196
+ };
197
+ }
198
+ if (!("$ref" in schema) && schema.type === "array" && schema.items && !("$ref" in schema.items)) return {
199
+ type: `${getMappedType(schema.items.type)}[]`,
200
+ ref: void 0
201
+ };
202
+ if (!("$ref" in schema) && schema.type === "object") return {
203
+ type: getMappedType(schema.type),
204
+ ref: "any"
205
+ };
206
+ throw new Error("无法解析相应的schema");
207
+ }
208
+
209
+ //#endregion
210
+ //#region src/parse/v2/parse-field.ts
211
+ function parseField$1(name, properties, required = false) {
212
+ const field = new Field(name, required);
213
+ const { type, ref, imports, enums } = parseSchemaType$1(properties);
214
+ field.type = type;
215
+ field.ref = ref;
216
+ field.imports = imports;
217
+ field.enums = enums;
218
+ if (!("$ref" in properties)) field.description = properties.description;
219
+ return field;
220
+ }
221
+
222
+ //#endregion
223
+ //#region src/parse/v2/parse-fields.ts
224
+ function parseFields$1(schema) {
225
+ return Object.entries(schema.properties || {}).map(([name, property]) => {
226
+ var _schema$required;
227
+ return parseField$1(name, property, (_schema$required = schema.required) === null || _schema$required === void 0 ? void 0 : _schema$required.includes(name));
228
+ });
229
+ }
230
+
231
+ //#endregion
232
+ //#region src/parse/v2/parse-model.ts
233
+ function parseModel$1(name, definition) {
234
+ const model = new Model(name);
235
+ model.fields = parseFields$1(definition);
236
+ const imports = model.fields.filter((field) => field.imports).reduce((r, m) => [...r, ...m.imports || []], []).filter((m) => m !== name);
237
+ model.imports = Array.from(new Set(imports));
238
+ return model;
239
+ }
240
+
241
+ //#endregion
242
+ //#region src/parse/v2/parse-models.ts
243
+ function parseModels$1(document) {
244
+ const models = [];
245
+ if (document.definitions) {
246
+ for (const definitionName in document.definitions) if (Object.getOwnPropertyNames(document.definitions).includes(definitionName)) {
247
+ const definition = document.definitions[definitionName];
248
+ const model = parseModel$1(getCamelName(definitionName), definition);
249
+ models.push(model);
250
+ }
251
+ }
252
+ return models;
253
+ }
254
+
255
+ //#endregion
256
+ //#region src/entities/service.ts
257
+ var Service = class {
258
+ constructor(name) {
259
+ var _Generate$options;
260
+ this.name = name;
261
+ this.responseType = ((_Generate$options = Generate.options) === null || _Generate$options === void 0 || (_Generate$options = _Generate$options.exportServices) === null || _Generate$options === void 0 ? void 0 : _Generate$options.responseType) || "promise";
262
+ }
263
+ name;
264
+ application;
265
+ imports = [];
266
+ operations = [];
267
+ responseType;
268
+ };
269
+
270
+ //#endregion
271
+ //#region src/utils/get-service-name.ts
272
+ /**
273
+ * 获取服务名称
274
+ * @param path
275
+ * @param method
276
+ * @param operationObject
277
+ * @param tags
278
+ * @returns 服务名称
279
+ */
280
+ function getServiceName(path, method, operationObject, tags) {
281
+ var _Generate$options;
282
+ const resolve = (_Generate$options = Generate.options) === null || _Generate$options === void 0 || (_Generate$options = _Generate$options.exportServices) === null || _Generate$options === void 0 ? void 0 : _Generate$options.serviceResolve;
283
+ if (resolve) return resolve({
284
+ path,
285
+ method,
286
+ object: operationObject,
287
+ tags
288
+ });
289
+ else {
290
+ var _operationObject$tags;
291
+ return ((_operationObject$tags = operationObject.tags) === null || _operationObject$tags === void 0 ? void 0 : _operationObject$tags.map((tag) => getCamelName(tag))) || "Default";
292
+ }
293
+ }
294
+
295
+ //#endregion
296
+ //#region src/entities/operation.ts
297
+ var Operation = class {
298
+ name;
299
+ method;
300
+ path;
301
+ description;
302
+ responseRef;
303
+ responseType;
304
+ parametersPath = [];
305
+ parametersQuery = [];
306
+ parametersBody;
307
+ imports = [];
308
+ constructor(name, method, path) {
309
+ var _Generate$options;
310
+ this.name = name;
311
+ this.method = method;
312
+ this.path = path;
313
+ this.responseType = ((_Generate$options = Generate.options) === null || _Generate$options === void 0 || (_Generate$options = _Generate$options.exportServices) === null || _Generate$options === void 0 ? void 0 : _Generate$options.responseType) || "promise";
314
+ }
315
+ };
316
+
317
+ //#endregion
318
+ //#region src/utils/get-operation-name.ts
319
+ /**
320
+ * 获取服务名称
321
+ * @param path
322
+ * @param method
323
+ * @param operationObject
324
+ * @returns string
325
+ */
326
+ function getOperationName(path, method, operationObject) {
327
+ var _Generate$options;
328
+ const resolve = (_Generate$options = Generate.options) === null || _Generate$options === void 0 || (_Generate$options = _Generate$options.exportServices) === null || _Generate$options === void 0 ? void 0 : _Generate$options.operationResolve;
329
+ if (resolve) return resolve({
330
+ path,
331
+ method,
332
+ object: operationObject
333
+ });
334
+ else return operationObject.operationId || "";
335
+ }
336
+
337
+ //#endregion
338
+ //#region src/entities/operation-parameter.ts
339
+ var OperationParameter = class {
340
+ in;
341
+ name;
342
+ type;
343
+ ref;
344
+ enums;
345
+ description;
346
+ required;
347
+ imports = [];
348
+ };
349
+
350
+ //#endregion
351
+ //#region src/parse/v2/parse-parameters-body.ts
352
+ function parseParametersBody$1(parameters) {
353
+ const requestBody = parameters.find((parameter) => !("$ref" in parameter) && parameter.in === "body");
354
+ if (requestBody && "schema" in requestBody) {
355
+ const { type, ref, imports } = parseSchemaType$1(requestBody.schema);
356
+ const parameter = new OperationParameter();
357
+ parameter.name = "requestBody";
358
+ parameter.in = "body";
359
+ parameter.type = type;
360
+ parameter.ref = ref;
361
+ parameter.imports = imports || [];
362
+ return parameter;
363
+ }
364
+ }
365
+
366
+ //#endregion
367
+ //#region src/parse/v2/parse-parameters-path.ts
368
+ function parseParametersPath$1(parameters) {
369
+ return parameters.reduce((r, p) => {
370
+ if (!("$ref" in p) && p.in === "path") {
371
+ const { type, ref, imports, enums } = parseSchemaType$1(p);
372
+ const parameter = new OperationParameter();
373
+ parameter.name = p.name;
374
+ parameter.description = p.description;
375
+ parameter.in = "path";
376
+ parameter.type = type;
377
+ parameter.ref = ref;
378
+ parameter.imports = imports || [];
379
+ parameter.enums = enums;
380
+ r.push(parameter);
381
+ }
382
+ return r;
383
+ }, []);
384
+ }
385
+
386
+ //#endregion
387
+ //#region src/parse/v2/parse-parameters-query.ts
388
+ function parseParametersQuery$1(parameters) {
389
+ var _Generate$options;
390
+ const excludeParams = (_Generate$options = Generate.options) === null || _Generate$options === void 0 || (_Generate$options = _Generate$options.exportServices) === null || _Generate$options === void 0 ? void 0 : _Generate$options.excludeQueryParams;
391
+ return parameters.reduce((r, p) => {
392
+ if (!("$ref" in p) && p.in === "query" && !(excludeParams && excludeParams.includes(p.name.split(".")[0]))) {
393
+ const { type, ref, imports, enums } = parseSchemaType$1(p);
394
+ const parameter = new OperationParameter();
395
+ parameter.name = p.name;
396
+ parameter.description = p.description;
397
+ parameter.in = "query";
398
+ parameter.type = type;
399
+ parameter.ref = ref;
400
+ parameter.required = p.required;
401
+ parameter.imports = imports || [];
402
+ parameter.enums = enums;
403
+ if (p.name.includes(".") && !p.name.startsWith(".") && !p.name.endsWith(".")) parameter.name = `"${p.name}"`;
404
+ r.push(parameter);
405
+ }
406
+ return r;
407
+ }, []);
408
+ }
409
+
410
+ //#endregion
411
+ //#region src/parse/v2/parse-operation.ts
412
+ function parseOperation$1(path, method, operationObject) {
413
+ var _operation$parameters;
414
+ const operation = new Operation(getOperationName(path, method, operationObject), method, path);
415
+ operation.description = operationObject.summary || operation.description;
416
+ if (operationObject.parameters) {
417
+ operation.parametersBody = parseParametersBody$1(operationObject.parameters);
418
+ operation.parametersPath = parseParametersPath$1(operationObject.parameters);
419
+ operation.parametersQuery = parseParametersQuery$1(operationObject.parameters);
420
+ }
421
+ const responseSchema = parseResponseType$1(operationObject.responses);
422
+ operation.imports = Array.from(new Set([
423
+ ...((_operation$parameters = operation.parametersBody) === null || _operation$parameters === void 0 ? void 0 : _operation$parameters.imports) || [],
424
+ ...operation.parametersPath.flatMap((p) => p.imports) || [],
425
+ ...operation.parametersQuery.flatMap((p) => p.imports) || [],
426
+ ...(responseSchema === null || responseSchema === void 0 ? void 0 : responseSchema.imports) || []
427
+ ]));
428
+ operation.responseRef = (responseSchema === null || responseSchema === void 0 ? void 0 : responseSchema.ref) || "void";
429
+ return operation;
430
+ }
431
+ function parseResponseType$1(responses) {
432
+ const response = responses === null || responses === void 0 ? void 0 : responses["200"];
433
+ if (response && "$ref" in response) return parseSchemaType$1(response);
434
+ if (response && "schema" in response && (response === null || response === void 0 ? void 0 : response.schema)) return parseSchemaType$1(response === null || response === void 0 ? void 0 : response.schema);
435
+ return {
436
+ type: "void",
437
+ ref: "void",
438
+ imports: []
439
+ };
440
+ }
441
+
442
+ //#endregion
443
+ //#region src/parse/v2/parse-service.ts
444
+ function parseService$1(path, method, operationObject, tags, services) {
445
+ const toNames = (name) => Array.isArray(name) ? name : [name];
446
+ const names = toNames(getServiceName(path, method, operationObject, tags || []));
447
+ const operation = parseOperation$1(path, method, operationObject);
448
+ names.forEach((name) => {
449
+ let service = services.find((service) => service.name === name);
450
+ if (!service) {
451
+ service = new Service(name);
452
+ services.push(service);
453
+ }
454
+ if (service) service.operations.push(operation);
455
+ service.operations.flatMap((operation) => operation.imports).forEach((model) => {
456
+ if (service && !service.imports.includes(model)) service.imports.push(model);
457
+ });
458
+ });
459
+ }
460
+
461
+ //#endregion
462
+ //#region src/parse/v2/parse-services.ts
463
+ function parseServices$1(document) {
464
+ const services = [];
465
+ Object.entries(document.paths).forEach(([path, pathObject]) => {
466
+ if (pathObject) Object.entries(pathObject).forEach(([method, operationObject]) => {
467
+ parseService$1(path, method, operationObject, document.tags || [], services);
468
+ });
469
+ });
470
+ return services;
471
+ }
472
+
473
+ //#endregion
474
+ //#region src/parse/v2/index.ts
475
+ function parseV2(document) {
476
+ return {
477
+ models: parseModels$1(document),
478
+ services: parseServices$1(document)
479
+ };
480
+ }
481
+
482
+ //#endregion
483
+ //#region src/parse/v3/strip-namespace.ts
484
+ /**
485
+ * Strip (OpenAPI) namespaces fom values.
486
+ * @param value
487
+ */
488
+ function stripNamespace(value) {
489
+ return value.trim().replace(/^#\/components\/schemas\//, "").replace(/^#\/components\/responses\//, "").replace(/^#\/components\/parameters\//, "").replace(/^#\/components\/examples\//, "").replace(/^#\/components\/requestBodies\//, "").replace(/^#\/components\/headers\//, "").replace(/^#\/components\/securitySchemes\//, "").replace(/^#\/components\/links\//, "").replace(/^#\/components\/callbacks\//, "");
490
+ }
491
+
492
+ //#endregion
493
+ //#region src/parse/v3/parse-schema-type.ts
494
+ function parseSchemaType(schema) {
495
+ if ("$ref" in schema) {
496
+ const ref = getCamelName(stripNamespace(schema.$ref));
497
+ return {
498
+ type: "any",
499
+ ref,
500
+ imports: [ref]
501
+ };
502
+ }
503
+ if (!("$ref" in schema) && schema.type !== "array" && !schema.allOf && !schema.anyOf && !schema.oneOf) return {
504
+ type: getMappedType(schema.type || "any"),
505
+ ref: void 0,
506
+ enums: schema.enum
507
+ };
508
+ if (schema.type === "array" && "$ref" in schema.items) {
509
+ const ref = getCamelName(stripNamespace(schema.items.$ref));
510
+ const type = getBuiltInType(ref);
511
+ return {
512
+ type: "any[]",
513
+ ref: `${type ?? ref}[]`,
514
+ imports: type ? void 0 : [ref]
515
+ };
516
+ }
517
+ if (schema.type === "array" && !("$ref" in schema.items)) return {
518
+ type: `${getMappedType(schema.items.type)}[]`,
519
+ ref: void 0
520
+ };
521
+ if (schema.allOf || schema.anyOf || schema.oneOf) {
522
+ const ofSchema = schema.allOf || schema.anyOf || schema.oneOf;
523
+ const ofSchemaArray = ofSchema === null || ofSchema === void 0 ? void 0 : ofSchema.map((s) => parseSchemaType(s));
524
+ if (ofSchemaArray) {
525
+ const hasRef = ofSchemaArray.some((s) => s.ref);
526
+ return {
527
+ type: hasRef ? "any" : ofSchemaArray.map((s) => s.type).join("|"),
528
+ ref: hasRef ? ofSchemaArray.map((s) => s.ref && getCamelName(s.ref) || s.type).join("|") : void 0,
529
+ imports: hasRef ? ofSchemaArray.reduce((r, s) => (s.ref && !r.includes(s.ref) && r.push(getCamelName(s.ref)), r), []) : void 0
530
+ };
531
+ }
532
+ }
533
+ throw new Error("无法解析相应的schema");
534
+ }
535
+
536
+ //#endregion
537
+ //#region src/parse/v3/parse-field.ts
538
+ function parseField(name, properties, required = false) {
539
+ const field = new Field(name, required);
540
+ const { type, ref, imports, enums } = parseSchemaType(properties);
541
+ field.type = type;
542
+ field.ref = ref;
543
+ field.imports = imports;
544
+ field.enums = enums;
545
+ if (!("$ref" in properties)) field.description = properties.description;
546
+ return field;
547
+ }
548
+
549
+ //#endregion
550
+ //#region src/parse/v3/parse-fields.ts
551
+ function parseFields(schema) {
552
+ return Object.entries(schema.properties || {}).map(([name, property]) => {
553
+ var _schema$required;
554
+ return parseField(name, property, (_schema$required = schema.required) === null || _schema$required === void 0 ? void 0 : _schema$required.includes(name));
555
+ });
556
+ }
557
+
558
+ //#endregion
559
+ //#region src/parse/v3/parse-model.ts
560
+ function parseModel(name, definition) {
561
+ const model = new Model(name);
562
+ model.fields = parseFields(definition);
563
+ const imports = model.fields.filter((field) => field.imports).reduce((r, m) => [...r, ...m.imports || []], []).filter((m) => m !== name);
564
+ model.imports = Array.from(new Set(imports));
565
+ return model;
566
+ }
567
+
568
+ //#endregion
569
+ //#region src/parse/v3/parse-models.ts
570
+ function parseModels(document) {
571
+ const models = [];
572
+ if (document.components) for (const definitionName in document.components.schemas) {
573
+ const { schemas } = document.components;
574
+ if (Object.getOwnPropertyNames(schemas).includes(definitionName)) {
575
+ const definition = document.components.schemas[definitionName];
576
+ const model = parseModel(getCamelName(definitionName), definition);
577
+ models.push(model);
578
+ }
579
+ }
580
+ return models;
581
+ }
582
+
583
+ //#endregion
584
+ //#region src/parse/v3/parse-parameters-body.ts
585
+ function parseParametersBody(requestBody) {
586
+ const { type, ref, imports } = parseBodyType(requestBody);
587
+ const parameter = new OperationParameter();
588
+ parameter.name = "requestBody";
589
+ parameter.in = "body";
590
+ parameter.type = type;
591
+ parameter.ref = ref;
592
+ parameter.imports = imports || [];
593
+ return parameter;
594
+ }
595
+ function parseBodyType(requestBody) {
596
+ var _requestBody$content$;
597
+ if ("$ref" in requestBody) return parseSchemaType(requestBody);
598
+ if ("content" in requestBody && (requestBody === null || requestBody === void 0 || (_requestBody$content$ = requestBody.content["application/json"]) === null || _requestBody$content$ === void 0 ? void 0 : _requestBody$content$.schema)) {
599
+ var _requestBody$content$2;
600
+ return parseSchemaType(requestBody === null || requestBody === void 0 || (_requestBody$content$2 = requestBody.content["application/json"]) === null || _requestBody$content$2 === void 0 ? void 0 : _requestBody$content$2.schema);
601
+ }
602
+ throw new Error("无法解析RequestBody Schema");
603
+ }
604
+
605
+ //#endregion
606
+ //#region src/parse/v3/parse-parameters-path.ts
607
+ function parseParametersPath(parameters) {
608
+ return parameters.reduce((r, p) => {
609
+ if (!("$ref" in p) && p.in === "path" && p.schema) {
610
+ const { type, ref, imports, enums } = parseSchemaType(p.schema);
611
+ const parameter = new OperationParameter();
612
+ parameter.name = p.name;
613
+ parameter.description = p.description;
614
+ parameter.in = "path";
615
+ parameter.type = type;
616
+ parameter.ref = ref;
617
+ parameter.imports = imports || [];
618
+ parameter.enums = enums;
619
+ r.push(parameter);
620
+ }
621
+ return r;
622
+ }, []);
623
+ }
624
+
625
+ //#endregion
626
+ //#region src/parse/v3/parse-parameters-query.ts
627
+ function parseParametersQuery(parameters) {
628
+ var _Generate$options;
629
+ const excludeParams = (_Generate$options = Generate.options) === null || _Generate$options === void 0 || (_Generate$options = _Generate$options.exportServices) === null || _Generate$options === void 0 ? void 0 : _Generate$options.excludeQueryParams;
630
+ return parameters.reduce((r, p) => {
631
+ if (!("$ref" in p) && p.in === "query" && p.schema && !(excludeParams && excludeParams.includes(p.name))) {
632
+ const { type, ref, imports, enums } = parseSchemaType(p.schema);
633
+ const parameter = new OperationParameter();
634
+ parameter.name = p.name;
635
+ parameter.description = p.description;
636
+ parameter.in = "query";
637
+ parameter.type = type;
638
+ parameter.ref = ref;
639
+ parameter.required = p.required;
640
+ parameter.imports = imports || [];
641
+ parameter.enums = enums;
642
+ if (p.name.includes(".") && !p.name.startsWith(".") && !p.name.endsWith(".")) parameter.name = `"${p.name}"`;
643
+ r.push(parameter);
644
+ }
645
+ return r;
646
+ }, []);
647
+ }
648
+
649
+ //#endregion
650
+ //#region src/parse/v3/parse-operation.ts
651
+ function parseOperation(path, method, operationObject) {
652
+ var _operation$parameters;
653
+ const operation = new Operation(getOperationName(path, method, operationObject), method, path);
654
+ operation.description = operationObject.summary;
655
+ if (operationObject.requestBody) operation.parametersBody = parseParametersBody(operationObject.requestBody);
656
+ if (operationObject.parameters) {
657
+ operation.parametersPath = parseParametersPath(operationObject.parameters);
658
+ operation.parametersQuery = parseParametersQuery(operationObject.parameters);
659
+ }
660
+ const responseSchema = parseResponseType(operationObject.responses);
661
+ operation.imports = Array.from(new Set([
662
+ ...((_operation$parameters = operation.parametersBody) === null || _operation$parameters === void 0 ? void 0 : _operation$parameters.imports) || [],
663
+ ...operation.parametersPath.flatMap((p) => p.imports) || [],
664
+ ...operation.parametersQuery.flatMap((p) => p.imports) || [],
665
+ ...(responseSchema === null || responseSchema === void 0 ? void 0 : responseSchema.imports) || []
666
+ ]));
667
+ operation.responseRef = (responseSchema === null || responseSchema === void 0 ? void 0 : responseSchema.ref) || (responseSchema === null || responseSchema === void 0 ? void 0 : responseSchema.type) || "void";
668
+ return operation;
669
+ }
670
+ function parseResponseType(responses) {
671
+ const response = responses === null || responses === void 0 ? void 0 : responses["200"];
672
+ const medias = ["*/*", "application/json"];
673
+ if (response && "$ref" in response) return parseSchemaType(responses);
674
+ if (response && "content" in response && medias.some((media) => {
675
+ var _response$content;
676
+ return !!(response === null || response === void 0 || (_response$content = response.content) === null || _response$content === void 0 || (_response$content = _response$content[media]) === null || _response$content === void 0 ? void 0 : _response$content.schema);
677
+ })) {
678
+ var _response$content3;
679
+ const mediaType = medias.find((media) => {
680
+ var _response$content2;
681
+ return !!(response === null || response === void 0 || (_response$content2 = response.content) === null || _response$content2 === void 0 || (_response$content2 = _response$content2[media]) === null || _response$content2 === void 0 ? void 0 : _response$content2.schema);
682
+ });
683
+ return parseSchemaType(response === null || response === void 0 || (_response$content3 = response.content) === null || _response$content3 === void 0 || (_response$content3 = _response$content3[mediaType]) === null || _response$content3 === void 0 ? void 0 : _response$content3.schema);
684
+ }
685
+ return {
686
+ type: "void",
687
+ ref: "void",
688
+ imports: []
689
+ };
690
+ }
691
+
692
+ //#endregion
693
+ //#region src/parse/v3/parse-service.ts
694
+ function parseService(path, method, operationObject, tags, services) {
695
+ const toNames = (name) => Array.isArray(name) ? name : [name];
696
+ const names = toNames(getServiceName(path, method, operationObject, tags || []));
697
+ const operation = parseOperation(path, method, operationObject);
698
+ names.forEach((name) => {
699
+ let service = services.find((service) => service.name === name);
700
+ if (!service) {
701
+ service = new Service(name);
702
+ services.push(service);
703
+ }
704
+ if (service) service.operations.push(operation);
705
+ service.operations.flatMap((operation) => operation.imports).forEach((model) => {
706
+ if (service && !service.imports.includes(model)) service.imports.push(model);
707
+ });
708
+ });
709
+ }
710
+
711
+ //#endregion
712
+ //#region src/parse/v3/parse-services.ts
713
+ function parseServices(document) {
714
+ const services = [];
715
+ Object.entries(document.paths).forEach(([path, pathObject]) => {
716
+ if (pathObject) Object.entries(pathObject).forEach(([method, operationObject]) => {
717
+ parseService(path, method, operationObject, document.tags || [], services);
718
+ });
719
+ });
720
+ return services;
721
+ }
722
+
723
+ //#endregion
724
+ //#region src/parse/v3/index.ts
725
+ function parseV3(document) {
726
+ return {
727
+ models: parseModels(document),
728
+ services: parseServices(document)
729
+ };
730
+ }
731
+
732
+ //#endregion
733
+ //#region src/progress.ts
734
+ const progressMaps = /* @__PURE__ */ new Map();
735
+ function createProgress(name) {
736
+ const spinner = (0, ora.default)(formatProgressText(name)).start();
737
+ progressMaps.set(name, {
738
+ spinner,
739
+ model: {
740
+ total: 0,
741
+ value: 0
742
+ },
743
+ service: {
744
+ total: 0,
745
+ value: 0
746
+ }
747
+ });
748
+ }
749
+ function startProgress(name, options) {
750
+ const progress = progressMaps.get(name);
751
+ if (progress) {
752
+ progress.model = {
753
+ total: options.models,
754
+ value: 0
755
+ };
756
+ progress.service = {
757
+ total: options.services,
758
+ value: 0
759
+ };
760
+ updateProgressText(name, progress);
761
+ }
762
+ }
763
+ function formatProgressText(name, progress) {
764
+ const toStateText = (text) => `${chalk.default.green(text)}`;
765
+ const toNameText = (text) => `${chalk.default.cyan(text).padEnd(20, " ")}`;
766
+ const toModelText = (text) => `${chalk.default.greenBright("Model")}:${chalk.default.gray(text).padEnd(10, " ")}`;
767
+ const toServiceText = (text) => `${chalk.default.greenBright("Service")}:${chalk.default.gray(text).padEnd(10, " ")}`;
768
+ const isFinish = (progress === null || progress === void 0 ? void 0 : progress.model.value) === (progress === null || progress === void 0 ? void 0 : progress.model.total) && (progress === null || progress === void 0 ? void 0 : progress.service.value) === (progress === null || progress === void 0 ? void 0 : progress.service.total);
769
+ if (progress) return `${toStateText(isFinish ? "生成完成" : "生成中")} | ${toNameText(name)} | ${toModelText(`${progress.model.value} / ${progress.model.total}`)} | ${toServiceText(`${progress.service.value} / ${progress.service.total}`)}`;
770
+ else return `${toStateText("开始请求")} | ${toNameText(name)}`;
771
+ }
772
+ function updateProgressText(name, progress) {
773
+ if ((progress === null || progress === void 0 ? void 0 : progress.model.value) === (progress === null || progress === void 0 ? void 0 : progress.model.total) && (progress === null || progress === void 0 ? void 0 : progress.service.value) === (progress === null || progress === void 0 ? void 0 : progress.service.total)) progress.spinner.succeed(formatProgressText(name, progress));
774
+ else progress.spinner.text = formatProgressText(name, progress);
775
+ }
776
+ function updateProgress(name, type) {
777
+ const progress = progressMaps.get(name);
778
+ if (!progress) return;
779
+ progress[type].value++;
780
+ updateProgressText(name, progress);
781
+ }
782
+
783
+ //#endregion
784
+ //#region src/template-helpers/equal.helper.ts
785
+ const equalHelper = {
786
+ name: "equal",
787
+ fn(v1, v2, options) {
788
+ if (v1 === v2) return options.fn(this);
789
+ else return options.inverse(this);
790
+ }
791
+ };
792
+
793
+ //#endregion
794
+ //#region src/template-helpers/include-query-params.ts
795
+ const includeQueryParams = {
796
+ name: "include-query-params",
797
+ fn(operations, options) {
798
+ if (operations.some((x) => x.parametersQuery.length > 0)) return options.fn(this);
799
+ else return options.inverse(this);
800
+ }
801
+ };
802
+
803
+ //#endregion
804
+ //#region src/template-helpers/is-array.helper.ts
805
+ const isArrayHelper = {
806
+ name: "is-array",
807
+ fn(v1, options) {
808
+ if (Array.isArray(v1)) return options.fn(this);
809
+ else return options.inverse(this);
810
+ }
811
+ };
812
+
813
+ //#endregion
814
+ //#region src/template-helpers/to-upper.helper.ts
815
+ const toUpperHelper = {
816
+ name: "to-upper",
817
+ fn(v1, onlyFirst) {
818
+ if (onlyFirst) return v1.replace(/^\S/, (s) => {
819
+ return s.toUpperCase();
820
+ });
821
+ else return v1.toUpperCase();
822
+ }
823
+ };
824
+
825
+ //#endregion
826
+ //#region src/template.ts
827
+ function registerHandlebarTemplates() {
828
+ registerHandlebarPartials();
829
+ registerHandlebarHelpers();
830
+ }
831
+ function registerHandlebarPartials() {
832
+ registerHandlebarPartial("is-required");
833
+ registerHandlebarPartial("export-header");
834
+ registerHandlebarPartial("export-description");
835
+ registerHandlebarPartial("export-model-import");
836
+ registerHandlebarPartial("export-model-type");
837
+ registerHandlebarPartial("export-model-field");
838
+ registerHandlebarPartial("export-schema-type");
839
+ registerHandlebarPartial("export-service-import");
840
+ registerHandlebarPartial("export-service-class");
841
+ registerHandlebarPartial("export-service-namespace");
842
+ registerHandlebarPartial("export-service-namespace-type");
843
+ registerHandlebarPartial("export-service-operation");
844
+ registerHandlebarPartial("export-operation-params-path");
845
+ registerHandlebarPartial("export-operation-params-query");
846
+ registerHandlebarPartial("export-operation-params-body");
847
+ registerHandlebarPartial("export-operation-response");
848
+ }
849
+ function registerHandlebarHelpers() {
850
+ registerHandlebarHelper(equalHelper);
851
+ registerHandlebarHelper(isArrayHelper);
852
+ registerHandlebarHelper(toUpperHelper);
853
+ registerHandlebarHelper(includeQueryParams);
854
+ }
855
+ function registerHandlebarHelper(helper) {
856
+ handlebars.default.registerHelper(helper.name, helper.fn);
857
+ }
858
+ /**
859
+ * 注册Handlebar模板
860
+ */
861
+ function registerHandlebarPartial(input) {
862
+ const template = loadHandlebarTemplate(`partials/${input}`);
863
+ handlebars.default.registerPartial(input, template);
864
+ }
865
+ /**
866
+ * 加载模板
867
+ * @returns 模板内容
868
+ */
869
+ function loadHandlebarTemplate(input) {
870
+ const templatePath = node_path.resolve(__dirname, "templates", `${input}.hbs`);
871
+ return node_fs.readFileSync(templatePath, "utf-8");
872
+ }
873
+
874
+ //#endregion
875
+ //#region src/utils/get-openapi-document.ts
876
+ /**
877
+ * Load and parse te open api spec. If the file extension is ".yml" or ".yaml"
878
+ * we will try to parse the file as a YAML spec, otherwise we will fall back
879
+ * on parsing the file as JSON.
880
+ * @param input Path or url
881
+ */
882
+ async function getOpenApiDocument(input) {
883
+ return await _apidevtools_swagger_parser.default.parse(input);
884
+ }
885
+
886
+ //#endregion
887
+ //#region src/utils/get-openapi-version.ts
888
+ /**
889
+ * 获取OpenAPI版本
890
+ * @param document
891
+ * @returns OpenAPI版本
892
+ */
893
+ function getOpenAPIVersion(document) {
894
+ const version = (document === null || document === void 0 ? void 0 : document.swagger) || (document === null || document === void 0 ? void 0 : document.openapi);
895
+ if (typeof version === "string") {
896
+ const v = Number.parseInt(version.charAt(0));
897
+ if (Object.values(OpenAPIVersion).includes(v)) return v;
898
+ }
899
+ throw new Error(`无法识别的OPENAPI版本: "${String(version)}"`);
900
+ }
901
+
902
+ //#endregion
903
+ //#region src/generate/write-config.ts
904
+ const ConfigFilePath = node_path.default.resolve(".request", ".config.json");
905
+ /**
906
+ * 生成应用配置文件
907
+ */
908
+ function updateOptionsFromLocalConfig(options, output) {
909
+ const config = readLocalConfig();
910
+ const toUpdateOptions = /* @__PURE__ */ new Map();
911
+ options = options.filter((item) => {
912
+ const md5 = isNeedUpdate(item.name, config, output);
913
+ if (md5) {
914
+ toUpdateOptions.set(item.name, md5);
915
+ return true;
916
+ } else return false;
917
+ });
918
+ options.forEach((item) => {
919
+ item.input = node_path.default.resolve(".request", `${item.name}.json`);
920
+ });
921
+ writeLocalConfig(toUpdateOptions, config);
922
+ return options;
923
+ }
924
+ /**
925
+ * 比较文件的MD5值判断文件是否更新
926
+ */
927
+ function isNeedUpdate(name, config, output) {
928
+ const file = node_path.default.resolve(".request", `${name}.json`);
929
+ const outputDir = node_path.default.resolve(output);
930
+ if (!node_fs.default.existsSync(file)) throw new Error(`未找到相应的配置文件: ${file}`);
931
+ const data = node_fs.default.readFileSync(file);
932
+ const md5 = node_crypto.default.createHash("md5").update(data.toString()).digest("hex");
933
+ if (!node_fs.default.existsSync(node_path.default.join(outputDir, name))) return md5;
934
+ const configItem = config.find((x) => x.name === name);
935
+ if (!configItem || configItem.md5 !== md5) return md5;
936
+ }
937
+ function readLocalConfig() {
938
+ if (!node_fs.default.existsSync(ConfigFilePath)) return [];
939
+ const data = node_fs.default.readFileSync(ConfigFilePath, "utf-8");
940
+ return JSON.parse(data);
941
+ }
942
+ function writeLocalConfig(toUpdateOptions, config) {
943
+ toUpdateOptions.forEach((md5, name) => {
944
+ const configItem = config.find((x) => x.name === name);
945
+ if (configItem) {
946
+ configItem.md5 = md5;
947
+ configItem.updatedAt = (/* @__PURE__ */ new Date()).toISOString();
948
+ } else config.push({
949
+ name,
950
+ md5,
951
+ updatedAt: (/* @__PURE__ */ new Date()).toISOString()
952
+ });
953
+ });
954
+ node_fs.default.writeFileSync(ConfigFilePath, JSON.stringify(config, null, 2));
955
+ }
956
+
957
+ //#endregion
958
+ //#region src/generate/write-file.ts
959
+ /**
960
+ *
961
+ * @param output
962
+ * @param content
963
+ */
964
+ function writeFile(output, content) {
965
+ node_fs.writeFileSync(output, content, "utf-8");
966
+ }
967
+
968
+ //#endregion
969
+ //#region src/generate/write-model.ts
970
+ function writeModel(model, output) {
971
+ const templateSource = loadHandlebarTemplate("export-model");
972
+ writeFile(output, handlebars.default.compile(templateSource)(model));
973
+ }
974
+
975
+ //#endregion
976
+ //#region src/generate/write-models.ts
977
+ /**
978
+ * 写入Model文件
979
+ * @param client
980
+ * @param options
981
+ */
982
+ function writeModels(client, options) {
983
+ if (!options.exportModels || !client.models) return;
984
+ const output = node_path.join(options.output, "models");
985
+ if (node_fs.existsSync(output)) rimraf.default.sync(output);
986
+ node_fs.mkdirSync(output, { recursive: true });
987
+ client.models.forEach((model) => {
988
+ const filename = `${model.name}.ts`;
989
+ writeModel(model, node_path.join(output, filename));
990
+ updateProgress(options.name || "default", "model");
991
+ });
992
+ }
993
+
994
+ //#endregion
995
+ //#region src/generate/write-service.ts
996
+ /**
997
+ * 写入Service文件
998
+ * @param service
999
+ * @param output
1000
+ */
1001
+ function writeService(service, output) {
1002
+ const templateSource = loadHandlebarTemplate("export-service");
1003
+ writeFile(output, handlebars.default.compile(templateSource)(service));
1004
+ }
1005
+
1006
+ //#endregion
1007
+ //#region src/generate/write-services.ts
1008
+ function writeServices(client, options) {
1009
+ if (!client.services) return;
1010
+ const output = node_path.join(options.output, "services");
1011
+ if (node_fs.existsSync(output)) rimraf.default.sync(output);
1012
+ node_fs.mkdirSync(output, { recursive: true });
1013
+ client.services.forEach((service) => {
1014
+ const filename = `${service.name}Service.ts`;
1015
+ service.application = Generate.options.appendService === false ? "" : options.application;
1016
+ writeService(service, node_path.join(output, filename));
1017
+ updateProgress(options.name || "default", "service");
1018
+ });
1019
+ }
1020
+
1021
+ //#endregion
1022
+ //#region src/generate/index.ts
1023
+ var Generate = class Generate {
1024
+ static options;
1025
+ /**
1026
+ * 生成入口
1027
+ * @param options
1028
+ * @returns Promise<void>
1029
+ */
1030
+ static async startup(options) {
1031
+ Generate.options = options;
1032
+ registerHandlebarTemplates();
1033
+ let applicationOptions = generateServiceOptions(options);
1034
+ applicationOptions = updateOptionsFromLocalConfig(applicationOptions, options.output);
1035
+ const applications = [];
1036
+ for (const applicationOption of applicationOptions) {
1037
+ const client = await Generate.generateApplicationClient(applicationOption);
1038
+ applications.push({
1039
+ client,
1040
+ options: applicationOption
1041
+ });
1042
+ }
1043
+ applications.forEach((application) => {
1044
+ createProgress(application.options.name || "default");
1045
+ Generate.writeClient(application.client, application.options);
1046
+ });
1047
+ }
1048
+ static async getApiDocument(input) {
1049
+ try {
1050
+ return await getOpenApiDocument(input);
1051
+ } catch {
1052
+ console.error(`请求文件[${input}]失败,请稍后重试.`);
1053
+ node_process.default.exit(0);
1054
+ }
1055
+ }
1056
+ /**
1057
+ * 生成应用
1058
+ */
1059
+ static async generateApplicationClient(options) {
1060
+ const document = await Generate.getApiDocument(options.input);
1061
+ const version = getOpenAPIVersion(document);
1062
+ return Generate.generateClient(document, version);
1063
+ }
1064
+ /**
1065
+ * 生成对象信息
1066
+ * @param document
1067
+ * @param version
1068
+ * @returns GenerateClient
1069
+ */
1070
+ static generateClient(document, version) {
1071
+ switch (version) {
1072
+ case OpenAPIVersion.V2: return parseV2(document);
1073
+ case OpenAPIVersion.V3: return parseV3(document);
1074
+ }
1075
+ }
1076
+ /**
1077
+ * 写入Client对象
1078
+ * @param client
1079
+ * @param options
1080
+ */
1081
+ static writeClient(client, options) {
1082
+ startProgress(options.name || "default", {
1083
+ models: client.models.length,
1084
+ services: client.services.length
1085
+ });
1086
+ writeModels(client, options);
1087
+ writeServices(client, options);
1088
+ }
1089
+ };
1090
+
1091
+ //#endregion
1092
+ //#region src/define-config.ts
1093
+ function defineConfig(config) {
1094
+ return config;
1095
+ }
1096
+
1097
+ //#endregion
1098
+ //#region src/index.ts
1099
+ const generate = Generate.startup;
1100
+ const download = Download.startup;
1101
+
1102
+ //#endregion
1103
+ exports.defineConfig = defineConfig;
1104
+ exports.download = download;
1105
+ exports.generate = generate;