let _ = require("lodash") let { pathToRegexp } = require("path-to-regexp") /******************************************************** * Based on: https://github.com/fsbahman/apidoc-swagger * ********************************************************/ let swagger = { swagger: "2.0", info: {}, paths: {}, definitions: {}, } function toSwagger(apidocJson, projectJson) { swagger.info = addInfo(projectJson) swagger.paths = extractPaths(apidocJson) return swagger } let tagsRegex = /(<([^>]+)>)/gi // Removes <p> </p> tags from text function removeTags(text) { return text ? text.replace(tagsRegex, "") : text } function addInfo(projectJson) { let info = {} info["title"] = projectJson.title || projectJson.name info["version"] = projectJson.version info["description"] = projectJson.description return info } /** * Extracts paths provided in json format * post, patch, put request parameters are extracted in body * get and delete are extracted to path parameters * @param apidocJson * @returns {{}} */ function extractPaths(apidocJson) { let apiPaths = groupByUrl(apidocJson) let paths = {} for (let i = 0; i < apiPaths.length; i++) { let verbs = apiPaths[i].verbs let url = verbs[0].url let pattern = pathToRegexp(url, null) let matches = pattern.exec(url) // Surrounds URL parameters with curly brackets -> :email with {email} let pathKeys = [] for (let j = 1; j < matches.length; j++) { let key = matches[j].slice(1) url = url.replace(matches[j], "{" + key + "}") pathKeys.push(key) } for (let j = 0; j < verbs.length; j++) { let verb = verbs[j] let type = verb.type let obj = (paths[url] = paths[url] || {}) if (type === "post" || type === "patch" || type === "put") { _.extend( obj, createPostPushPutOutput(verb, swagger.definitions, pathKeys) ) } else { _.extend(obj, createGetDeleteOutput(verb, swagger.definitions)) } } } return paths } function createPostPushPutOutput(verbs, definitions, pathKeys) { let pathItemObject = {} let verbDefinitionResult = createVerbDefinitions(verbs, definitions) let params = [] let pathParams = createPathParameters(verbs, pathKeys) pathParams = _.filter(pathParams, function (param) { let hasKey = pathKeys.indexOf(param.name) !== -1 return !(param.in === "path" && !hasKey) }) params = params.concat(pathParams) let required = verbs.parameter && verbs.parameter.fields && verbs.parameter.fields.Parameter && verbs.parameter.fields.Parameter.length > 0 params.push({ in: "body", name: "body", description: removeTags(verbs.description), required: required, schema: { $ref: "#/definitions/" + verbDefinitionResult.topLevelParametersRef, }, }) pathItemObject[verbs.type] = { tags: [verbs.group], summary: removeTags(verbs.description), consumes: ["application/json"], produces: ["application/json"], parameters: params, } if (verbDefinitionResult.topLevelSuccessRef) { pathItemObject[verbs.type].responses = { 200: { description: "successful operation", schema: { type: verbDefinitionResult.topLevelSuccessRefType, items: { $ref: "#/definitions/" + verbDefinitionResult.topLevelSuccessRef, }, }, }, } } return pathItemObject } function createVerbDefinitions(verbs, definitions) { let result = { topLevelParametersRef: null, topLevelSuccessRef: null, topLevelSuccessRefType: null, } let defaultObjectName = verbs.name let fieldArrayResult = {} if (verbs && verbs.parameter && verbs.parameter.fields) { fieldArrayResult = createFieldArrayDefinitions( verbs.parameter.fields.Parameter, definitions, verbs.name, defaultObjectName ) result.topLevelParametersRef = fieldArrayResult.topLevelRef } if (verbs && verbs.success && verbs.success.fields) { fieldArrayResult = createFieldArrayDefinitions( verbs.success.fields["Success 200"], definitions, verbs.name, defaultObjectName ) result.topLevelSuccessRef = fieldArrayResult.topLevelRef result.topLevelSuccessRefType = fieldArrayResult.topLevelRefType } return result } function createFieldArrayDefinitions( fieldArray, definitions, topLevelRef, defaultObjectName ) { let result = { topLevelRef: topLevelRef, topLevelRefType: null, } if (!fieldArray) { return result } for (let i = 0; i < fieldArray.length; i++) { let parameter = fieldArray[i] let nestedName = createNestedName(parameter.field) let objectName = nestedName.objectName if (!objectName) { objectName = defaultObjectName } let type = parameter.type if (i === 0) { result.topLevelRefType = type if (parameter.type === "Object") { objectName = nestedName.propertyName nestedName.propertyName = null } else if (parameter.type === "Array") { objectName = nestedName.propertyName nestedName.propertyName = null result.topLevelRefType = "array" } result.topLevelRef = objectName } definitions[objectName] = definitions[objectName] || { properties: {}, required: [], } if (nestedName.propertyName) { let prop = { type: (parameter.type || "").toLowerCase(), description: removeTags(parameter.description), } if (parameter.type === "Object") { prop.$ref = "#/definitions/" + parameter.field } let typeIndex = type.indexOf("[]") if (typeIndex !== -1 && typeIndex === type.length - 2) { prop.type = "array" prop.items = { type: type.slice(0, type.length - 2), } } definitions[objectName]["properties"][nestedName.propertyName] = prop if (!parameter.optional) { let arr = definitions[objectName]["required"] if (arr.indexOf(nestedName.propertyName) === -1) { arr.push(nestedName.propertyName) } } } } return result } function createNestedName(field) { let propertyName = field let objectName let propertyNames = field.split(".") if (propertyNames && propertyNames.length > 1) { propertyName = propertyNames[propertyNames.length - 1] propertyNames.pop() objectName = propertyNames.join(".") } return { propertyName: propertyName, objectName: objectName, } } /** * Generate get, delete method output * @param verbs * @param definitions * @returns {{}} */ function createGetDeleteOutput(verbs, definitions) { let pathItemObject = {} verbs.type = verbs.type === "del" ? "delete" : verbs.type let verbDefinitionResult = createVerbDefinitions(verbs, definitions) pathItemObject[verbs.type] = { tags: [verbs.group], summary: removeTags(verbs.description), consumes: ["application/json"], produces: ["application/json"], parameters: createPathParameters(verbs), } if (verbDefinitionResult.topLevelSuccessRef) { pathItemObject[verbs.type].responses = { 200: { description: "successful operation", schema: { type: verbDefinitionResult.topLevelSuccessRefType, items: { $ref: "#/definitions/" + verbDefinitionResult.topLevelSuccessRef, }, }, }, } } return pathItemObject } /** * Iterate through all method parameters and create array of parameter objects which are stored as path parameters * @param verbs * @returns {Array} */ function createPathParameters(verbs) { let pathItemObject = [] if (verbs.parameter && verbs.parameter.fields.Parameter) { for (let i = 0; i < verbs.parameter.fields.Parameter.length; i++) { let param = verbs.parameter.fields.Parameter[i] let field = param.field let type = param.type pathItemObject.push({ name: field, in: type === "file" ? "formData" : "path", required: !param.optional, type: param.type.toLowerCase(), description: removeTags(param.description), }) } } return pathItemObject } function groupByUrl(apidocJson) { return _.chain(apidocJson) .groupBy("url") .toPairs() .map(function (element) { return _.zipObject(["url", "verbs"], element) }) .value() } module.exports = toSwagger