wsdl-tsclient
Version:
Generate typescript soap client with typescript definitons from WSDL file.
348 lines • 19.1 kB
JavaScript
var __assign = (this && this.__assign) || function () {
__assign = Object.assign || function(t) {
for (var s, i = 1, n = arguments.length; i < n; i++) {
s = arguments[i];
for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p))
t[p] = s[p];
}
return t;
};
return __assign.apply(this, arguments);
};
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
var __generator = (this && this.__generator) || function (thisArg, body) {
var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;
return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
function verb(n) { return function (v) { return step([n, v]); }; }
function step(op) {
if (f) throw new TypeError("Generator is already executing.");
while (g && (g = 0, op[0] && (_ = 0)), _) try {
if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
if (y = 0, t) op = [op[0] & 2, t.value];
switch (op[0]) {
case 0: case 1: t = op; break;
case 4: _.label++; return { value: op[1], done: false };
case 5: _.label++; y = op[1]; op = [0]; continue;
case 7: op = _.ops.pop(); _.trys.pop(); continue;
default:
if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
if (t[2]) _.ops.pop();
_.trys.pop(); continue;
}
op = body.call(thisArg, _);
} catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
}
};
var __spreadArray = (this && this.__spreadArray) || function (to, from, pack) {
if (pack || arguments.length === 2) for (var i = 0, l = from.length, ar; i < l; i++) {
if (ar || !(i in from)) {
if (!ar) ar = Array.prototype.slice.call(from, 0, i);
ar[i] = from[i];
}
}
return to.concat(ar || Array.prototype.slice.call(from));
};
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.generate = generate;
var camelcase_1 = __importDefault(require("camelcase"));
var path_1 = __importDefault(require("path"));
var ts_morph_1 = require("ts-morph");
var _1 = require(".");
var logger_1 = require("./utils/logger");
var defaultOptions = {
emitDefinitionsOnly: false,
modelPropertyNaming: null,
esm: false,
};
/**
* To avoid duplicated imports
*/
function addSafeImport(imports, moduleSpecifier, namedImport) {
if (!imports.find(function (imp) { return imp.moduleSpecifier == moduleSpecifier; })) {
imports.push({
moduleSpecifier: moduleSpecifier,
namedImports: [{ name: namedImport }],
});
}
}
var incorrectPropNameChars = [" ", "-", "."];
/**
* This is temporally method to fix this issue https://github.com/dsherret/ts-morph/issues/1160
*/
function sanitizePropName(propName) {
if (incorrectPropNameChars.some(function (char) { return propName.includes(char); })) {
return "\"".concat(propName, "\"");
}
return propName;
}
function createProperty(name, type, doc, isArray, optional) {
if (optional === void 0) { optional = true; }
return {
kind: ts_morph_1.StructureKind.PropertySignature,
name: sanitizePropName(name),
docs: [doc],
hasQuestionToken: true,
type: isArray ? "Array<".concat(type, ">") : type,
};
}
function generateDefinitionFile(project, definition, defDir, stack, generated, options) {
var defName = definition.name;
var defFilePath = path_1.default.join(defDir, "".concat(defName, ".ts"));
var defFile = project.createSourceFile(defFilePath, "", {
overwrite: true,
});
generated.push(definition);
var definitionImports = [];
var definitionProperties = [];
for (var _i = 0, _a = definition.properties; _i < _a.length; _i++) {
var prop = _a[_i];
if (options.modelPropertyNaming) {
switch (options.modelPropertyNaming) {
case _1.ModelPropertyNaming.camelCase:
prop.name = (0, camelcase_1.default)(prop.name);
break;
case _1.ModelPropertyNaming.PascalCase:
prop.name = (0, camelcase_1.default)(prop.name, { pascalCase: true });
break;
}
}
if (prop.kind === "PRIMITIVE") {
// e.g. string
definitionProperties.push(createProperty(prop.name, prop.type, prop.description, prop.isArray));
}
else if (prop.kind === "REFERENCE") {
// e.g. Items
if (!generated.includes(prop.ref)) {
// Wasn't generated yet
generateDefinitionFile(project, prop.ref, defDir, __spreadArray(__spreadArray([], stack, true), [prop.ref.name], false), generated, options);
}
// If a property is of the same type as its parent type, don't add import
if (prop.ref.name !== definition.name) {
addSafeImport(definitionImports, "./".concat(prop.ref.name).concat(options.esm ? ".js" : ""), prop.ref.name);
}
definitionProperties.push(createProperty(prop.name, prop.ref.name, prop.sourceName, prop.isArray));
}
}
defFile.addImportDeclarations(definitionImports);
defFile.addStatements([
{
leadingTrivia: function (writer) { return writer.newLine(); },
isExported: true,
name: defName,
docs: [definition.docs.join("\n")],
kind: ts_morph_1.StructureKind.Interface,
properties: definitionProperties,
},
]);
logger_1.Logger.log("Writing Definition file: ".concat(path_1.default.resolve(path_1.default.join(defDir, defName)), ".ts"));
defFile.saveSync();
}
function generate(parsedWsdl, outDir, options) {
return __awaiter(this, void 0, void 0, function () {
var mergedOptions, project, portsDir, servicesDir, defDir, allMethods, allDefinitions, clientImports, clientServices, _i, _a, service, serviceFilePath, serviceFile, serviceImports, servicePorts, _b, _c, port, portFilePath, portFile, portImports, portFileMethods, _d, _e, method, clientFilePath, clientFile, createClientDeclaration, indexFilePath, indexFile;
return __generator(this, function (_f) {
mergedOptions = __assign(__assign({}, defaultOptions), options);
project = new ts_morph_1.Project();
portsDir = path_1.default.join(outDir, "ports");
servicesDir = path_1.default.join(outDir, "services");
defDir = path_1.default.join(outDir, "definitions");
allMethods = [];
allDefinitions = [];
clientImports = [];
clientServices = [];
for (_i = 0, _a = parsedWsdl.services; _i < _a.length; _i++) {
service = _a[_i];
serviceFilePath = path_1.default.join(servicesDir, "".concat(service.name, ".ts"));
serviceFile = project.createSourceFile(serviceFilePath, "", {
overwrite: true,
});
serviceImports = [];
servicePorts = [];
for (_b = 0, _c = parsedWsdl.ports; _b < _c.length; _b++) {
port = _c[_b];
portFilePath = path_1.default.join(portsDir, "".concat(port.name, ".ts"));
portFile = project.createSourceFile(portFilePath, "", {
overwrite: true,
});
portImports = [];
portFileMethods = [];
for (_d = 0, _e = port.methods; _d < _e.length; _d++) {
method = _e[_d];
// TODO: Deduplicate PortImports
if (method.paramDefinition !== null) {
if (!allDefinitions.includes(method.paramDefinition)) {
// Definition is not generated
generateDefinitionFile(project, method.paramDefinition, defDir, [method.paramDefinition.name], allDefinitions, mergedOptions);
addSafeImport(clientImports, "./definitions/".concat(method.paramDefinition.name).concat(mergedOptions.esm ? ".js" : ""), method.paramDefinition.name);
}
addSafeImport(portImports, "../definitions/".concat(method.paramDefinition.name).concat(mergedOptions.esm ? ".js" : ""), method.paramDefinition.name);
}
if (method.returnDefinition !== null) {
if (!allDefinitions.includes(method.returnDefinition)) {
// Definition is not generated
generateDefinitionFile(project, method.returnDefinition, defDir, [method.returnDefinition.name], allDefinitions, mergedOptions);
addSafeImport(clientImports, "./definitions/".concat(method.returnDefinition.name).concat(mergedOptions.esm ? ".js" : ""), method.returnDefinition.name);
}
addSafeImport(portImports, "../definitions/".concat(method.returnDefinition.name).concat(mergedOptions.esm ? ".js" : ""), method.returnDefinition.name);
}
// TODO: Deduplicate PortMethods
allMethods.push(method);
portFileMethods.push({
name: sanitizePropName(method.name),
parameters: [
{
name: (0, camelcase_1.default)(method.paramName),
type: method.paramDefinition ? method.paramDefinition.name : "{}",
},
{
name: "callback",
type: "(err: any, result: ".concat(method.returnDefinition ? method.returnDefinition.name : "unknown", ", rawResponse: any, soapHeader: any, rawRequest: any) => void"), // TODO: Use ts-morph to generate proper type
},
],
returnType: "void",
});
} // End of PortMethod
if (!mergedOptions.emitDefinitionsOnly) {
addSafeImport(serviceImports, "../ports/".concat(port.name).concat(mergedOptions.esm ? ".js" : ""), port.name);
servicePorts.push({
name: sanitizePropName(port.name),
isReadonly: true,
type: port.name,
});
portFile.addImportDeclarations(portImports);
portFile.addStatements([
{
leadingTrivia: function (writer) { return writer.newLine(); },
isExported: true,
kind: ts_morph_1.StructureKind.Interface,
name: port.name,
methods: portFileMethods,
},
]);
logger_1.Logger.log("Writing Port file: ".concat(path_1.default.resolve(path_1.default.join(portsDir, port.name)), ".ts"));
portFile.saveSync();
}
} // End of Port
if (!mergedOptions.emitDefinitionsOnly) {
addSafeImport(clientImports, "./services/".concat(service.name).concat(mergedOptions.esm ? ".js" : ""), service.name);
clientServices.push({ name: sanitizePropName(service.name), type: service.name });
serviceFile.addImportDeclarations(serviceImports);
serviceFile.addStatements([
{
leadingTrivia: function (writer) { return writer.newLine(); },
isExported: true,
kind: ts_morph_1.StructureKind.Interface,
name: service.name,
properties: servicePorts,
},
]);
logger_1.Logger.log("Writing Service file: ".concat(path_1.default.resolve(path_1.default.join(servicesDir, service.name)), ".ts"));
serviceFile.saveSync();
}
} // End of Service
if (!mergedOptions.emitDefinitionsOnly) {
clientFilePath = path_1.default.join(outDir, "client.ts");
clientFile = project.createSourceFile(clientFilePath, "", {
overwrite: true,
});
clientFile.addImportDeclaration({
moduleSpecifier: "soap",
namedImports: [
{ name: "Client", alias: "SoapClient" },
{ name: "createClientAsync", alias: "soapCreateClientAsync" },
{ name: "IExOptions", alias: "ISoapExOptions" },
],
});
clientFile.addImportDeclarations(clientImports);
clientFile.addStatements([
{
leadingTrivia: function (writer) { return writer.newLine(); },
isExported: true,
kind: ts_morph_1.StructureKind.Interface,
// docs: [`${parsedWsdl.name}Client`],
name: "".concat(parsedWsdl.name, "Client"),
properties: clientServices,
extends: ["SoapClient"],
methods: allMethods.map(function (method) { return ({
name: sanitizePropName("".concat(method.name, "Async")),
parameters: [
{
name: (0, camelcase_1.default)(method.paramName),
type: method.paramDefinition ? method.paramDefinition.name : "{}",
},
{
name: "options",
type: "ISoapExOptions",
hasQuestionToken: true,
},
],
returnType: "Promise<[result: ".concat(method.returnDefinition ? method.returnDefinition.name : "unknown", ", rawResponse: any, soapHeader: any, rawRequest: any]>"),
}); }),
},
]);
createClientDeclaration = clientFile.addFunction({
name: "createClientAsync",
docs: ["Create ".concat(parsedWsdl.name, "Client")],
isExported: true,
parameters: [
{
isRestParameter: true,
name: "args",
type: "Parameters<typeof soapCreateClientAsync>",
},
],
returnType: "Promise<".concat(parsedWsdl.name, "Client>"), // TODO: `any` keyword is very dangerous
});
createClientDeclaration.setBodyText("return soapCreateClientAsync(args[0], args[1], args[2]) as any;");
logger_1.Logger.log("Writing Client file: ".concat(path_1.default.resolve(path_1.default.join(outDir, "client")), ".ts"));
clientFile.saveSync();
}
indexFilePath = path_1.default.join(outDir, "index.ts");
indexFile = project.createSourceFile(indexFilePath, "", {
overwrite: true,
});
indexFile.addExportDeclarations(allDefinitions.map(function (def) { return ({
namedExports: [def.name],
moduleSpecifier: "./definitions/".concat(def.name).concat(mergedOptions.esm ? ".js" : ""),
}); }));
if (!mergedOptions.emitDefinitionsOnly) {
// TODO: Aggregate all exports during declarations generation
// https://ts-morph.com/details/exports
indexFile.addExportDeclarations([
{
namedExports: ["createClientAsync", "".concat(parsedWsdl.name, "Client")],
moduleSpecifier: "./client".concat(mergedOptions.esm ? ".js" : ""),
},
]);
indexFile.addExportDeclarations(parsedWsdl.services.map(function (service) { return ({
namedExports: [service.name],
moduleSpecifier: "./services/".concat(service.name).concat(mergedOptions.esm ? ".js" : ""),
}); }));
indexFile.addExportDeclarations(parsedWsdl.ports.map(function (port) { return ({
namedExports: [port.name],
moduleSpecifier: "./ports/".concat(port.name).concat(mergedOptions.esm ? ".js" : ""),
}); }));
}
logger_1.Logger.log("Writing Index file: ".concat(path_1.default.resolve(path_1.default.join(outDir, "index")), ".ts"));
indexFile.saveSync();
return [2 /*return*/];
});
});
}
//# sourceMappingURL=generator.js.map
;