"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); var fs = require("fs"); var path = require("path"); var Config_1 = require("./Config"); var OpenAPI_1 = require("./OpenAPI"); var CodePrinter_1 = require("./CodePrinter"); function codegenIHandler(funcs, config, cp) { var schemasName = config.schemasName, utilsTSPath = config.utilsTSPath, clientOnly = config.clientOnly; // import cp.writeln("import * as Schemas from './".concat(schemasName, "'")); cp.writeln('import {FullDate, StrictTypeParser as STP, APIPromise} ' + "from '".concat(utilsTSPath, "'")); if (!clientOnly) { cp.writeln('import {RouterContext as CTX} from \'@koa/router\''); } cp.writeln('import {AxiosResponse} from \'axios\''); // api req, res types cp.writeln("export type TAPI = {", 1); for (var _i = 0, _a = Object.entries(funcs); _i < _a.length; _i++) { var _b = _a[_i], funcName = _b[0], func = _b[1]; var reqTypes = func.reqTypes, resTypes = func.resTypes, method = func.method; cp.writeln("".concat(funcName, ": {"), 1); // req // req.path, ... cp.writeln("req: {", 1); for (var _c = 0, ELParameterIn_1 = OpenAPI_1.ELParameterIn; _c < ELParameterIn_1.length; _c++) { var _in = ELParameterIn_1[_c]; var paras = reqTypes[_in]; if (paras == null) continue; cp.writeln("".concat(_in, ": {"), 1); for (var _d = 0, _e = Object.entries(paras); _d < _e.length; _d++) { var _f = _e[_d], propName = _f[0], schemaType = _f[1]; cp.writeln(schemaType.forProp(propName) + ';'); } cp.writeln('};', -1); } // body var body = reqTypes.body; if (body != null) { // PATCH's req body: Partial var typeName = body.typeName; if (method === 'patch') typeName = "Partial<".concat(typeName, ">"); cp.writeln("body".concat(body.required ? '' : '?', ": ").concat(typeName, ";")); } cp.writeln('}', -1); // req END // res cp.writeln("res: {", 1); for (var _g = 0, _h = Object.entries(resTypes); _g < _h.length; _g++) { var _j = _h[_g], status_1 = _j[0], schema = _j[1]; cp.writeln(schema.required ? "".concat(schema.forProp(status_1), ";") : "".concat(status_1, ": void;")); } cp.writeln('}', -1); // res END // operation END cp.writeln('}', -1); } // TAPI END cp.writeln('}', -1); // export IServerAPI if (!clientOnly) { cp.writeln(''); cp.writeln('type ValueOf = T[keyof T];'); cp.writeln('type RServerAPI = ValueOf<', 1); cp.writeln('{[K in keyof T]: T[K] extends void ? [K, any?] : [K, T[K]]}>;', -1, false); cp.writeln('export type IServerAPI = {[K in keyof TAPI]:', 1); cp.writeln("(req: TAPI[K]['req'], state: IState, ctx: CTX) =>", 1); cp.writeln("Promise>}", -2, false); } // return return cp.end(); } function codegenRouter(funcs, config, cp) { var schemasName = config.schemasName, IHandlerName = config.IHandlerName, ServerAPITSPath = config.ServerAPITSPath, utilsTSPath = config.utilsTSPath; // import cp.writeln("import * as Schemas from './".concat(schemasName, "'")); cp.writeln("import {IServerAPI} from './".concat(IHandlerName, "'")); cp.writeln("import Router from '@koa/router'"); cp.writeln("import {FullDate, StrictTypeParser as STP} from '".concat(utilsTSPath, "'")); cp.writeln("import bodyParser from 'koa-body'"); // api cp.writeln("\nimport api from '".concat(ServerAPITSPath, "'")); cp.writeln("type IState = typeof api extends IServerAPI ? T : any;"); // router cp.writeln("type CTX = Router.RouterContext;"); cp.writeln("\nconst router = new Router();"); // STP cp.writeln("router.use((ctx, next) => next().catch(err => {\n if (err instanceof STP.BadValueError) return ctx.throw(400, err.toString());\n throw err;\n}))"); // function var gcGetParams = { path: function (attr) { return "ctx.params['".concat(attr, "']"); }, query: function (attr) { return "ctx.query['".concat(attr, "']"); }, header: function (attr) { return "ctx.headers['".concat(attr, "']"); }, cookie: function (attr) { return "ctx.cookies.get('".concat(attr, "')"); }, }; // route for (var _i = 0, _a = Object.entries(funcs); _i < _a.length; _i++) { var _b = _a[_i], funcName = _b[0], func = _b[1]; var method = func.method, url = func.url, reqTypes = func.reqTypes; var isPartial = method === 'patch'; // TODO escape var sURL = url.replace(/{(.*?)}/g, ':$1'); // {a} -> :a var mid = ''; if (reqTypes.body) { var maxSize = reqTypes.body.maxSize; // TODO doc var config_1 = maxSize == null ? '' : "{jsonLimit: '".concat(maxSize, "'}"); mid = "bodyParser(".concat(config_1, "), "); } cp.writeln("router.".concat(method, "('").concat(sURL, "', ").concat(mid, "async ctx => {"), 1); // req if (Object.keys(reqTypes).length === 0) { cp.writeln('const req = {};'); } else { cp.writeln('const req = {', 1); // paras for (var _c = 0, ELParameterIn_2 = OpenAPI_1.ELParameterIn; _c < ELParameterIn_2.length; _c++) { var _in = ELParameterIn_2[_c]; var paras = reqTypes[_in]; if (paras == null) continue; cp.writeln("".concat(_in, ": {"), 1); for (var _d = 0, _e = Object.entries(paras); _d < _e.length; _d++) { var _f = _e[_d], name_1 = _f[0], schema = _f[1]; var pn = gcGetParams[_in](name_1); var label = "req.".concat(_in); cp.writeln("".concat(name_1, ": ").concat(schema.stp(pn, label), ",")); } cp.writeln('},', -1); } // body var body = reqTypes.body; if (body != null) { cp.writeln("body: ".concat(body.stp('ctx.request.body', 'req.body', isPartial))); } // req END cp.writeln('};', -1); } // call cp.writeln("const r = await api.".concat(funcName, "(req, ctx.state, ctx);")); cp.writeln("ctx.status = r[0];"); cp.writeln("ctx.body = r[1] ?? '';"); // ctx END cp.writeln('});', -1); } cp.writeln('\nexport default router;'); return cp.end(); } function codegenClientAPI(funcs, config, cp) { var IHandlerName = config.IHandlerName, schemasName = config.schemasName, utilsTSPath = config.utilsTSPath, validateStatus = config.validateStatus; // import cp.writeln("import {TAPI} from './".concat(IHandlerName, "'")); cp.writeln("import * as Schemas from './".concat(schemasName, "'")); cp.writeln("import {APIPromise, StrictTypeParser as STP, " + "qStringify} from '".concat(utilsTSPath, "'")); cp.writeln("import axios from 'axios'"); cp.writeln(''); // type cp.writeln("type TSTP = {[K in keyof T]: (data: any) =>" + "T[K] extends void ? any : T[K]};"); cp.writeln("export type ExID = {[K in keyof T]: " + "'id' extends keyof T[K] ? {id: T[K]['id']} : T[K]};"); cp.writeln(); // axios cp.writeln('const $http = axios.create({', 1); cp.writeln('validateStatus: () => true,'); cp.writeln("paramsSerializer: params => qStringify(params, {arrayFormat: 'repeat'}),"); cp.writeln('});', -1); // function cp.writeln('\nfunction urlReplacer(url: string, ' + 'rules: {[_: string]: any}): string {', 1); cp.writeln('for(const [attr, value] of Object.entries(rules))', 1); cp.writeln("url = url.replace('{'+attr+'}', value)"); cp.writeln('return url;', -1); cp.writeln('};', -1); // implementation // export default cp.writeln('\nexport default {', 1); // set $baseURL cp.writeln('set $baseURL(url: string) {', 1); cp.writeln('$http.interceptors.request.use(async config => {', 1); cp.writeln('config.baseURL = url;'); cp.writeln('return config;'); cp.writeln('}, err => Promise.reject(err));', -1); cp.writeln('},', -1); var _loop_1 = function (funcName, func) { var gcReq = function (_in) { return "TAPI['".concat(funcName, "']['req']['").concat(_in, "']"); }; var method = func.method, url = func.url, reqTypes = func.reqTypes, resTypes = func.resTypes; var query = reqTypes.query, header = reqTypes.header, path_1 = reqTypes.path, body = reqTypes.body; // TODO cookie // name cp.writeln("".concat(funcName, ": ("), 1); // paras for (var _c = 0, ELParameterIn_3 = OpenAPI_1.ELParameterIn; _c < ELParameterIn_3.length; _c++) { var _in = ELParameterIn_3[_c]; var paras = reqTypes[_in]; if (paras == null) continue; var _required = false; for (var _d = 0, _e = Object.values(paras); _d < _e.length; _d++) { var required = _e[_d].required; if (required) { _required = true; break; } } cp.writeln("".concat(_in, ": ").concat(gcReq(_in)).concat(_required ? '' : '={}', ",")); } // body if (body != null) { cp.writeln("body".concat(body.required ? '' : '?', ": ExID<").concat(gcReq('body'), ">,")); } // return value cp.tab(-1); cp.writeln(") => APIPromise.init($http({", 1); // req cp.writeln("method: '".concat(method, "',")); var sURL = "'".concat(url, "'"); cp.writeln("url: ".concat(path_1 ? "urlReplacer(".concat(sURL, ", path)") : sURL, ",")); if (query) cp.writeln('params: query,'); if (header) cp.writeln('header: header,'); if (body != null) cp.writeln('data: body,'); cp.writeln('}), {', -1); cp.tab(1); // stp for (var _f = 0, _g = Object.entries(resTypes); _f < _g.length; _f++) { var _h = _g[_f], status_2 = _h[0], schema = _h[1]; var label = "ClientAPI[".concat(funcName, "][").concat(status_2, "]"); cp.writeln("".concat(status_2, ": x => ").concat(schema.stp('x', label), ",")); } cp.writeln("} as TSTP,"), -1); cp.tab(1); // kRsv cp.writeln("[".concat(Object.keys(resTypes).filter(validateStatus).join(', '), "]),"), -1); }; // functions for (var _i = 0, _a = Object.entries(funcs); _i < _a.length; _i++) { var _b = _a[_i], funcName = _b[0], func = _b[1]; _loop_1(funcName, func); } cp.writeln('}', -1); return cp.end(); } function codegenSchemas(schemas, config, cp) { var _a; var utilsTSPath = config.utilsTSPath; // import cp.writeln("import {FullDate, StrictTypeParser as STP} from '".concat(utilsTSPath, "'")); // schema for (var _i = 0, _b = Object.entries(schemas); _i < _b.length; _i++) { var _c = _b[_i], typeName = _c[0], rSchema = _c[1]; var schema = (0, OpenAPI_1.resolveRef)(rSchema, schemas, '#/components/schemas'); if (schema == null) continue; cp.writeln(); if ((0, OpenAPI_1.isObjectSchema)(schema)) { // interface cp.writeln("export interface ".concat(typeName, " {"), 1); var propTypes = []; var requireds = new Set((_a = schema.required) !== null && _a !== void 0 ? _a : []); for (var _d = 0, _e = Object.entries(schema.properties); _d < _e.length; _d++) { var _f = _e[_d], propName = _f[0], prop = _f[1]; var propType = new OpenAPI_1.SchemaType(prop, requireds.has(propName), true); propTypes.push([propName, propType]); cp.writeln(propType.forProp(propName) + ';'); } cp.writeln('}', -1); // interface END // const cp.writeln("export const ".concat(typeName, " = {"), 1); // .from cp.writeln("from: (o: {[_: string]: any}): ".concat(typeName, " => ({"), 1); for (var _g = 0, propTypes_1 = propTypes; _g < propTypes_1.length; _g++) { var _h = propTypes_1[_g], n = _h[0], t = _h[1]; cp.writeln("".concat(n, ": ").concat(t.stp("o.".concat(n), typeName + '.' + n, false, true), ",")); } cp.writeln('}),', -1); // Partial cp.writeln("Partial: (o: {[_: string]: any}): Partial<".concat(typeName, "> => {"), 1); cp.writeln("const r: Partial<".concat(typeName, "> = {};")); var locPartial = "Partial<".concat(typeName, ">"); for (var _j = 0, propTypes_2 = propTypes; _j < propTypes_2.length; _j++) { var _k = propTypes_2[_j], n = _k[0], t = _k[1]; cp.writeln("if (o.".concat(n, " !== void 0) r.").concat(n, " = ").concat(t.stp("o.".concat(n), locPartial + '.' + n, false, true), ";")); } cp.writeln('return r;'); cp.writeln('},', -1); // fields cp.writeln("fields: [", 1); cp.writeln(propTypes.map(function (e) { return "'".concat(e[0], "',"); }).join(' ')); cp.writeln("] as Array"), -1); // end of const cp.writeln('}', -1); } else { cp.writeln("export type ".concat(typeName, " = ") + OpenAPI_1.SchemaType.typeNameOf(schema, true)); } } // return return cp.end(); } function codegen(openAPI, configUser) { var _a; var config = Object.assign({}, Config_1.configDefault, configUser); // prepare fs.mkdirSync(config.outputDir, { recursive: true }); var apiFuncs = (0, OpenAPI_1.apiFunctionsOf)(openAPI); var gCP = function (fn) { return new CodePrinter_1.CodePrinter(fs.createWriteStream(path.join(config.outputDir, fn + '.ts')), config.indentString); }; var ps = []; // write files // handler ps.push(codegenIHandler(apiFuncs, config, gCP(config.IHandlerName))); // server if (!config.clientOnly) { ps.push(codegenRouter(apiFuncs, config, gCP(config.routerName))); } // client ps.push(codegenClientAPI(apiFuncs, config, gCP(config.ClientAPIName))); // schema var schemas = (_a = openAPI.components) === null || _a === void 0 ? void 0 : _a.schemas; if (schemas != null) { ps.push(codegenSchemas(schemas, config, gCP(config.schemasName))); } // return return Promise.all(ps); } exports.default = codegen;