| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 'use strict'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Documentation.js service | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @description: A set of functions similar to controller's actions to avoid code duplication. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | const fs = require('fs'); | 
					
						
							|  |  |  | const path = require('path'); | 
					
						
							|  |  |  | const _ = require('lodash'); | 
					
						
							|  |  |  | const moment = require('moment'); | 
					
						
							|  |  |  | const pathToRegexp = require('path-to-regexp'); | 
					
						
							|  |  |  | const defaultComponents = require('./utils/components.json'); | 
					
						
							|  |  |  | const form = require('./utils/forms.json'); | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  | const defaultSettings = require('../config/settings.json'); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | const parametersOptions = require('./utils/parametersOptions.json'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  | // keys to pick from the extended config
 | 
					
						
							|  |  |  | const defaultSettingsKeys = Object.keys(defaultSettings); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | module.exports = { | 
					
						
							|  |  |  |   areObjectsEquals: (obj1, obj2) => { | 
					
						
							|  |  |  |     return JSON.stringify(obj1) === JSON.stringify(obj2); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   checkIfAPIDocNeedsUpdate: function(apiName) { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |     const prevDocumentation = this.createDocObject(this.retrieveDocumentation(apiName)); | 
					
						
							|  |  |  |     const currentDocumentation = this.createDocObject(this.createDocumentationFile(apiName, false)); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return !this.areObjectsEquals(prevDocumentation, currentDocumentation); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Check if the documentation folder with its related version of an API exists | 
					
						
							|  |  |  |    * @param {String} apiName | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   checkIfDocumentationFolderExists: function(apiName) { | 
					
						
							|  |  |  |     try { | 
					
						
							|  |  |  |       fs.accessSync(this.getDocumentationPath(apiName)); | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |     } catch (err) { | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   checkIfPluginDocumentationFolderExists: function(pluginName) { | 
					
						
							|  |  |  |     try { | 
					
						
							|  |  |  |       fs.accessSync(this.getPluginDocumentationPath(pluginName)); | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |     } catch (err) { | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   checkIfPluginDocNeedsUpdate: function(pluginName) { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |     const prevDocumentation = this.createDocObject(this.retrieveDocumentation(pluginName, true)); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |     const currentDocumentation = this.createDocObject( | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |       this.createPluginDocumentationFile(pluginName, false) | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return !this.areObjectsEquals(prevDocumentation, currentDocumentation); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   checkIfApiDefaultDocumentationFileExist: function(apiName, docName) { | 
					
						
							|  |  |  |     try { | 
					
						
							|  |  |  |       fs.accessSync(this.getAPIOverrideDocumentationPath(apiName, docName)); | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |     } catch (err) { | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   checkIfPluginDefaultDocumentFileExists: function(pluginName, docName) { | 
					
						
							|  |  |  |     try { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |       fs.accessSync(this.getPluginOverrideDocumentationPath(pluginName, docName)); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       return true; | 
					
						
							|  |  |  |     } catch (err) { | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Check if the documentation folder exists in the documentation plugin | 
					
						
							|  |  |  |    * @returns {Boolean} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   checkIfMergedDocumentationFolderExists: function() { | 
					
						
							|  |  |  |     try { | 
					
						
							|  |  |  |       fs.accessSync(this.getMergedDocumentationPath()); | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |     } catch (err) { | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Recursively create missing directories | 
					
						
							| 
									
										
										
										
											2018-12-13 16:29:57 +01:00
										 |  |  |    * @param {String} targetDir | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |    * | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   createDocumentationDirectory: function(targetDir) { | 
					
						
							|  |  |  |     const sep = path.sep; | 
					
						
							|  |  |  |     const initDir = path.isAbsolute(targetDir) ? sep : ''; | 
					
						
							|  |  |  |     const baseDir = '.'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return targetDir.split(sep).reduce((parentDir, childDir) => { | 
					
						
							|  |  |  |       const curDir = path.resolve(baseDir, parentDir, childDir); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       try { | 
					
						
							|  |  |  |         fs.mkdirSync(curDir); | 
					
						
							|  |  |  |       } catch (err) { | 
					
						
							|  |  |  |         if (err.code === 'EEXIST') { | 
					
						
							|  |  |  |           // curDir already exists!
 | 
					
						
							|  |  |  |           return curDir; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // To avoid `EISDIR` error on Mac and `EACCES`-->`ENOENT` and `EPERM` on Windows.
 | 
					
						
							|  |  |  |         if (err.code === 'ENOENT') { | 
					
						
							|  |  |  |           // Throw the original parentDir error on curDir `ENOENT` failure.
 | 
					
						
							|  |  |  |           throw new Error( | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |             `Impossible to create the documentation folder in '${parentDir}', please check the permissions.` | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |           ); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const caughtErr = ['EACCES', 'EPERM', 'EISDIR'].indexOf(err.code) > -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!caughtErr || (caughtErr && targetDir === curDir)) { | 
					
						
							|  |  |  |           throw err; // Throw if it's just the last created dir.
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return curDir; | 
					
						
							|  |  |  |     }, initDir); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Create the apiName.json and unclassified.json files inside an api's documentation/version folder | 
					
						
							|  |  |  |    * @param {String} apiName | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   createDocumentationFile: function(apiName, writeFile = true) { | 
					
						
							|  |  |  |     // Retrieve all the routes from an API
 | 
					
						
							|  |  |  |     const apiRoutes = this.getApiRoutes(apiName); | 
					
						
							|  |  |  |     const apiDocumentation = this.generateApiDocumentation(apiName, apiRoutes); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return Object.keys(apiDocumentation).reduce((acc, docName) => { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |       const targetFile = path.resolve(this.getDocumentationPath(apiName), `${docName}.json`); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       // Create the components object in each documentation file when we can create it
 | 
					
						
							|  |  |  |       const components = | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |         strapi.models[docName] !== undefined ? this.generateResponseComponent(docName) : {}; | 
					
						
							|  |  |  |       const tags = docName.split('-').length > 1 ? [] : this.generateTags(apiName, docName); | 
					
						
							|  |  |  |       const documentation = Object.assign(apiDocumentation[docName], components, { tags }); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |       try { | 
					
						
							|  |  |  |         if (writeFile) { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |           return fs.writeFileSync(targetFile, JSON.stringify(documentation, null, 2), 'utf8'); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |         } else { | 
					
						
							|  |  |  |           return acc.concat(documentation); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } catch (err) { | 
					
						
							|  |  |  |         return acc; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }, []); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   createPluginDocumentationFile: function(pluginName, writeFile = true) { | 
					
						
							|  |  |  |     const pluginRoutes = this.getPluginRoutesWithDescription(pluginName); | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |     const pluginDocumentation = this.generatePluginDocumentation(pluginName, pluginRoutes); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return Object.keys(pluginDocumentation).reduce((acc, docName) => { | 
					
						
							|  |  |  |       const targetFile = path.resolve( | 
					
						
							|  |  |  |         this.getPluginDocumentationPath(pluginName), | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |         `${docName}.json` | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       ); | 
					
						
							|  |  |  |       const components = | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |         _.get(strapi, this.getModelForPlugin(docName, pluginName)) !== undefined && | 
					
						
							|  |  |  |         pluginName !== 'upload' | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |           ? this.generateResponseComponent(docName, pluginName, true) | 
					
						
							|  |  |  |           : {}; | 
					
						
							|  |  |  |       const [plugin, name] = this.getModelAndNameForPlugin(docName, pluginName); | 
					
						
							|  |  |  |       const tags = | 
					
						
							|  |  |  |         docName !== 'unclassified' | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |           ? this.generateTags(plugin, docName, _.upperFirst(this.formatTag(plugin, name)), true) | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |           : []; | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |       const documentation = Object.assign(pluginDocumentation[docName], components, { tags }); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |       try { | 
					
						
							|  |  |  |         if (writeFile) { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |           return fs.writeFileSync(targetFile, JSON.stringify(documentation, null, 2), 'utf8'); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |         } else { | 
					
						
							|  |  |  |           return acc.concat(documentation); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } catch (err) { | 
					
						
							|  |  |  |         // Silent
 | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }, []); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   createDocObject: array => { | 
					
						
							|  |  |  |     return array.reduce((acc, curr) => _.merge(acc, curr), {}); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |   deleteDocumentation: async function(version = this.getDocumentationVersion()) { | 
					
						
							|  |  |  |     const recursiveDeleteFiles = async (folderPath, removeCompleteFolder = true) => { | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       // Check if folderExist
 | 
					
						
							|  |  |  |       try { | 
					
						
							|  |  |  |         const arrayOfPromises = []; | 
					
						
							|  |  |  |         fs.accessSync(folderPath); | 
					
						
							|  |  |  |         const items = fs.readdirSync(folderPath).filter(x => x[0] !== '.'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         items.forEach(item => { | 
					
						
							|  |  |  |           const itemPath = path.join(folderPath, item); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           // Check if directory
 | 
					
						
							|  |  |  |           if (fs.lstatSync(itemPath).isDirectory()) { | 
					
						
							|  |  |  |             if (removeCompleteFolder) { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |               return arrayOfPromises.push(recursiveDeleteFiles(itemPath), removeCompleteFolder); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |             } else if (!itemPath.includes('overrides')) { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |               return arrayOfPromises.push(recursiveDeleteFiles(itemPath), removeCompleteFolder); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |           } else { | 
					
						
							|  |  |  |             // Delete all files
 | 
					
						
							|  |  |  |             try { | 
					
						
							|  |  |  |               fs.unlinkSync(itemPath); | 
					
						
							|  |  |  |             } catch (err) { | 
					
						
							|  |  |  |               console.log('Cannot delete file', err); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         await Promise.all(arrayOfPromises); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         try { | 
					
						
							|  |  |  |           if (removeCompleteFolder) { | 
					
						
							|  |  |  |             fs.rmdirSync(folderPath); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } catch (err) { | 
					
						
							|  |  |  |           // console.log(err);
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } catch (err) { | 
					
						
							|  |  |  |         // console.log('The folder does not exist');
 | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const arrayOfPromises = []; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Delete api's documentation
 | 
					
						
							|  |  |  |     const apis = this.getApis(); | 
					
						
							|  |  |  |     const plugins = this.getPluginsWithDocumentationNeeded(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     apis.forEach(api => { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |       const apiPath = path.join(strapi.config.appPath, 'api', api, 'documentation', version); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       arrayOfPromises.push(recursiveDeleteFiles(apiPath)); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     plugins.forEach(plugin => { | 
					
						
							|  |  |  |       const pluginPath = path.join( | 
					
						
							|  |  |  |         strapi.config.appPath, | 
					
						
							| 
									
										
										
										
											2019-04-15 18:57:58 +02:00
										 |  |  |         'extensions', | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |         plugin, | 
					
						
							|  |  |  |         'documentation', | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |         version | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (version !== '1.0.0') { | 
					
						
							|  |  |  |         arrayOfPromises.push(recursiveDeleteFiles(pluginPath)); | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         arrayOfPromises.push(recursiveDeleteFiles(pluginPath, false)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const fullDocPath = path.join( | 
					
						
							|  |  |  |       strapi.config.appPath, | 
					
						
							| 
									
										
										
										
											2019-04-15 18:57:58 +02:00
										 |  |  |       'extensions', | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       'documentation', | 
					
						
							|  |  |  |       'documentation', | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |       version | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |     ); | 
					
						
							|  |  |  |     arrayOfPromises.push(recursiveDeleteFiles(fullDocPath)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return await Promise.all(arrayOfPromises); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Wrap endpoints variables in curly braces | 
					
						
							|  |  |  |    * @param {String} endPoint | 
					
						
							|  |  |  |    * @returns {String} (/products/{id}) | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   formatApiEndPoint: endPoint => { | 
					
						
							|  |  |  |     return pathToRegexp | 
					
						
							|  |  |  |       .parse(endPoint) | 
					
						
							|  |  |  |       .map(token => { | 
					
						
							|  |  |  |         if (_.isObject(token)) { | 
					
						
							|  |  |  |           return token.prefix + '{' + token.name + '}'; // eslint-disable-line prefer-template
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return token; | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |       .join(''); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Format a plugin model for example users-permissions, user => Users-Permissions - User | 
					
						
							|  |  |  |    * @param {Sting} plugin | 
					
						
							|  |  |  |    * @param {String} name | 
					
						
							|  |  |  |    * @param {Boolean} withoutSpace | 
					
						
							|  |  |  |    * @return {String} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   formatTag: (plugin, name, withoutSpace = false) => { | 
					
						
							|  |  |  |     const formattedPluginName = plugin | 
					
						
							|  |  |  |       .split('-') | 
					
						
							|  |  |  |       .map(i => _.upperFirst(i)) | 
					
						
							| 
									
										
										
										
											2020-03-20 14:03:51 +01:00
										 |  |  |       .join(''); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |     const formattedName = _.upperFirst(name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (withoutSpace) { | 
					
						
							|  |  |  |       return `${formattedPluginName}${formattedName}`; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return `${formattedPluginName} - ${formattedName}`; | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   generateAssociationSchema: function(attributes, getter) { | 
					
						
							|  |  |  |     return Object.keys(attributes).reduce( | 
					
						
							|  |  |  |       (acc, curr) => { | 
					
						
							|  |  |  |         const attribute = attributes[curr]; | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |         const isField = !_.has(attribute, 'model') && !_.has(attribute, 'collection'); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (attribute.required) { | 
					
						
							|  |  |  |           acc.required.push(curr); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (isField) { | 
					
						
							|  |  |  |           acc.properties[curr] = { type: this.getType(attribute.type) }; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |           const newGetter = getter.slice(); | 
					
						
							|  |  |  |           newGetter.splice(newGetter.length - 1, 1, 'associations'); | 
					
						
							|  |  |  |           const relationNature = _.get(strapi, newGetter).filter( | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |             association => association.alias === curr | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |           )[0].nature; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           switch (relationNature) { | 
					
						
							|  |  |  |             case 'manyToMany': | 
					
						
							|  |  |  |             case 'oneToMany': | 
					
						
							|  |  |  |             case 'manyToManyMorph': | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |               acc.properties[curr] = { | 
					
						
							|  |  |  |                 type: 'array', | 
					
						
							|  |  |  |                 items: { type: 'string' }, | 
					
						
							|  |  |  |               }; | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |               break; | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |               acc.properties[curr] = { type: 'string' }; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return acc; | 
					
						
							|  |  |  |       }, | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |       { required: ['id'], properties: { id: { type: 'string' } } } | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |     ); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							| 
									
										
										
										
											2019-07-05 03:05:36 +02:00
										 |  |  |    * Creates the paths object with all the needed information | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |    * The object has the following structure { apiName: { paths: {} }, knownTag1: { paths: {} }, unclassified: { paths: {} } } | 
					
						
							|  |  |  |    * Each key will create a documentation.json file | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * @param {String} apiName | 
					
						
							|  |  |  |    * @param {Array} routes | 
					
						
							|  |  |  |    * @returns {Object} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   generateApiDocumentation: function(apiName, routes) { | 
					
						
							|  |  |  |     return routes.reduce((acc, current) => { | 
					
						
							|  |  |  |       const [controllerName, controllerMethod] = current.handler.split('.'); | 
					
						
							|  |  |  |       // Retrieve the tag key in the config object
 | 
					
						
							|  |  |  |       const routeTagConfig = _.get(current, ['config', 'tag']); | 
					
						
							|  |  |  |       // Add curly braces between dynamic params
 | 
					
						
							|  |  |  |       const endPoint = this.formatApiEndPoint(current.path); | 
					
						
							| 
									
										
										
										
											2019-03-05 20:55:46 +02:00
										 |  |  |       let verb; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (Array.isArray(current.method)) { | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |         verb = current.method.map(method => method.toLowerCase()); | 
					
						
							| 
									
										
										
										
											2019-03-05 20:55:46 +02:00
										 |  |  |       } else { | 
					
						
							|  |  |  |         verb = current.method.toLowerCase(); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       // The key corresponds to firsts keys of the returned object
 | 
					
						
							|  |  |  |       let key; | 
					
						
							|  |  |  |       let tags; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |       if (controllerName.toLowerCase() === apiName && !_.isObject(routeTagConfig)) { | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |         key = apiName; | 
					
						
							|  |  |  |       } else if (routeTagConfig !== undefined) { | 
					
						
							|  |  |  |         if (_.isObject(routeTagConfig)) { | 
					
						
							|  |  |  |           const { name, plugin } = routeTagConfig; | 
					
						
							| 
									
										
										
										
											2018-12-13 16:29:57 +01:00
										 |  |  |           const referencePlugin = !_.isEmpty(plugin); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           key = referencePlugin ? `${plugin}-${name}` : name.toLowerCase(); | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |           tags = referencePlugin ? this.formatTag(plugin, name) : _.upperFirst(name); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |         } else { | 
					
						
							|  |  |  |           key = routeTagConfig.toLowerCase(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         key = 'unclassified'; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       const verbObject = { | 
					
						
							|  |  |  |         deprecated: false, | 
					
						
							|  |  |  |         description: this.generateVerbDescription( | 
					
						
							|  |  |  |           verb, | 
					
						
							|  |  |  |           current.handler, | 
					
						
							|  |  |  |           key, | 
					
						
							|  |  |  |           endPoint.split('/')[1], | 
					
						
							| 
									
										
										
										
											2019-12-20 14:47:31 +01:00
										 |  |  |           _.get(current, 'config.description') | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |         ), | 
					
						
							|  |  |  |         responses: this.generateResponses(verb, current, key), | 
					
						
							|  |  |  |         summary: '', | 
					
						
							|  |  |  |         tags: _.isEmpty(tags) ? [_.upperFirst(key)] : [_.upperFirst(tags)], | 
					
						
							|  |  |  |       }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-05 22:03:52 +02:00
										 |  |  |       // Swagger is not support key with ',' symbol, for array of methods need generate documentation for each method
 | 
					
						
							|  |  |  |       if (Array.isArray(verb)) { | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |         verb.forEach(method => { | 
					
						
							| 
									
										
										
										
											2019-03-05 22:03:52 +02:00
										 |  |  |           _.set(acc, [key, 'paths', endPoint, method], verbObject); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         _.set(acc, [key, 'paths', endPoint, verb], verbObject); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-05 22:03:52 +02:00
										 |  |  |       if (verb.includes('post') || verb.includes('put')) { | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |         let requestBody; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (controllerMethod === 'create' || controllerMethod === 'update') { | 
					
						
							|  |  |  |           requestBody = { | 
					
						
							|  |  |  |             description: '', | 
					
						
							|  |  |  |             required: true, | 
					
						
							|  |  |  |             content: { | 
					
						
							|  |  |  |               'application/json': { | 
					
						
							|  |  |  |                 schema: { | 
					
						
							|  |  |  |                   $ref: `#/components/schemas/New${_.upperFirst(key)}`, | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |               }, | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |           }; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |           requestBody = { | 
					
						
							|  |  |  |             description: '', | 
					
						
							|  |  |  |             required: true, | 
					
						
							|  |  |  |             content: { | 
					
						
							|  |  |  |               'application/json': { | 
					
						
							|  |  |  |                 schema: { | 
					
						
							|  |  |  |                   properties: { | 
					
						
							|  |  |  |                     foo: { | 
					
						
							|  |  |  |                       type: 'string', | 
					
						
							|  |  |  |                     }, | 
					
						
							|  |  |  |                   }, | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |               }, | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |           }; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-05 22:03:52 +02:00
										 |  |  |         if (Array.isArray(verb)) { | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |           verb.forEach(method => { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |             _.set(acc, [key, 'paths', endPoint, method, 'requestBody'], requestBody); | 
					
						
							| 
									
										
										
										
											2019-03-05 22:03:52 +02:00
										 |  |  |           }); | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |           _.set(acc, [key, 'paths', endPoint, verb, 'requestBody'], requestBody); | 
					
						
							| 
									
										
										
										
											2019-03-05 22:03:52 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Refer to https://swagger.io/specification/#pathItemObject
 | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |       const parameters = this.generateVerbParameters(verb, controllerMethod, current.path); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-05 22:03:52 +02:00
										 |  |  |       if (!verb.includes('post')) { | 
					
						
							|  |  |  |         if (Array.isArray(verb)) { | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |           verb.forEach(method => { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |             _.set(acc, [key, 'paths', endPoint, method, 'parameters'], parameters); | 
					
						
							| 
									
										
										
										
											2019-03-05 22:03:52 +02:00
										 |  |  |           }); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |           _.set(acc, [key, 'paths', endPoint, verb, 'parameters'], parameters); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return acc; | 
					
						
							|  |  |  |     }, {}); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   generateFullDoc: function(version = this.getDocumentationVersion()) { | 
					
						
							|  |  |  |     const apisDoc = this.retrieveDocumentationFiles(false, version); | 
					
						
							|  |  |  |     const pluginsDoc = this.retrieveDocumentationFiles(true, version); | 
					
						
							|  |  |  |     const appDoc = [...apisDoc, ...pluginsDoc]; | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |     const defaultSettings = _.cloneDeep( | 
					
						
							|  |  |  |       _.pick(strapi.plugins.documentation.config, defaultSettingsKeys) | 
					
						
							|  |  |  |     ); | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |     _.set(defaultSettings, ['info', 'x-generation-date'], moment().format('L LTS')); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |     _.set(defaultSettings, ['info', 'version'], version); | 
					
						
							|  |  |  |     const tags = appDoc.reduce((acc, current) => { | 
					
						
							|  |  |  |       const tags = current.tags.filter(el => { | 
					
						
							|  |  |  |         return _.findIndex(acc, ['name', el.name || '']) === -1; | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return acc.concat(tags); | 
					
						
							|  |  |  |     }, []); | 
					
						
							|  |  |  |     const fullDoc = _.merge( | 
					
						
							|  |  |  |       appDoc.reduce((acc, current) => { | 
					
						
							|  |  |  |         return _.merge(acc, current); | 
					
						
							|  |  |  |       }, defaultSettings), | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |       defaultComponents | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       // { tags },
 | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fullDoc.tags = tags; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return fullDoc; | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Generate the main component that has refs to sub components | 
					
						
							|  |  |  |    * @param {Object} attributes | 
					
						
							|  |  |  |    * @param {Array} associations | 
					
						
							|  |  |  |    * @returns {Object} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   generateMainComponent: function(attributes, associations) { | 
					
						
							|  |  |  |     return Object.keys(attributes).reduce( | 
					
						
							|  |  |  |       (acc, current) => { | 
					
						
							|  |  |  |         const attribute = attributes[current]; | 
					
						
							|  |  |  |         // Refer to https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md#dataTypes
 | 
					
						
							|  |  |  |         const type = this.getType(attribute.type); | 
					
						
							|  |  |  |         const { | 
					
						
							|  |  |  |           description, | 
					
						
							|  |  |  |           default: defaultValue, | 
					
						
							|  |  |  |           minimum, | 
					
						
							|  |  |  |           maxmimun, | 
					
						
							|  |  |  |           maxLength, | 
					
						
							|  |  |  |           minLength, | 
					
						
							|  |  |  |           enum: enumeration, | 
					
						
							|  |  |  |         } = attribute; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (attribute.required === true) { | 
					
						
							|  |  |  |           acc.required.push(current); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (attribute.model || attribute.collection) { | 
					
						
							|  |  |  |           const currentAssociation = associations.filter( | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |             association => association.alias === current | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |           )[0]; | 
					
						
							|  |  |  |           const relationNature = currentAssociation.nature; | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |           const name = currentAssociation.model || currentAssociation.collection; | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |           const getter = | 
					
						
							|  |  |  |             currentAssociation.plugin !== undefined | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |               ? ['plugins', currentAssociation.plugin, 'models', name, 'attributes'] | 
					
						
							| 
									
										
										
										
											2019-02-28 21:41:46 +02:00
										 |  |  |               : ['models', name.toLowerCase(), 'attributes']; | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |           const associationAttributes = _.get(strapi, getter); | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |           const associationSchema = this.generateAssociationSchema(associationAttributes, getter); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |           switch (relationNature) { | 
					
						
							|  |  |  |             case 'manyToMany': | 
					
						
							|  |  |  |             case 'oneToMany': | 
					
						
							|  |  |  |             case 'manyToManyMorph': | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |               acc.properties[current] = { | 
					
						
							|  |  |  |                 type: 'array', | 
					
						
							|  |  |  |                 items: associationSchema, | 
					
						
							|  |  |  |               }; | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |               break; | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |               acc.properties[current] = associationSchema; | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-10-22 18:01:03 +02:00
										 |  |  |         } else if (type === 'component') { | 
					
						
							|  |  |  |           const { repeatable, component, min, max } = attribute; | 
					
						
							| 
									
										
										
										
											2019-09-10 16:52:51 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |           const cmp = this.generateMainComponent( | 
					
						
							| 
									
										
										
										
											2019-10-22 18:01:03 +02:00
										 |  |  |             strapi.components[component].attributes, | 
					
						
							|  |  |  |             strapi.components[component].associations | 
					
						
							| 
									
										
										
										
											2019-09-10 16:52:51 +02:00
										 |  |  |           ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if (repeatable) { | 
					
						
							|  |  |  |             acc.properties[current] = { | 
					
						
							|  |  |  |               type: 'array', | 
					
						
							|  |  |  |               items: { | 
					
						
							|  |  |  |                 type: 'object', | 
					
						
							|  |  |  |                 ...cmp, | 
					
						
							|  |  |  |               }, | 
					
						
							|  |  |  |               minItems: min, | 
					
						
							|  |  |  |               maxItems: max, | 
					
						
							|  |  |  |             }; | 
					
						
							|  |  |  |           } else { | 
					
						
							|  |  |  |             acc.properties[current] = { | 
					
						
							|  |  |  |               type: 'object', | 
					
						
							|  |  |  |               ...cmp, | 
					
						
							|  |  |  |               description, | 
					
						
							|  |  |  |             }; | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-12-02 11:59:31 +01:00
										 |  |  |         } else if (type === 'dynamiczone') { | 
					
						
							|  |  |  |           const { components, min, max } = attribute; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           const cmps = components.map(component => { | 
					
						
							|  |  |  |             const schema = this.generateMainComponent( | 
					
						
							|  |  |  |               strapi.components[component].attributes, | 
					
						
							|  |  |  |               strapi.components[component].associations | 
					
						
							|  |  |  |             ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return _.merge( | 
					
						
							|  |  |  |               { | 
					
						
							|  |  |  |                 properties: { | 
					
						
							|  |  |  |                   __component: { | 
					
						
							|  |  |  |                     type: 'string', | 
					
						
							|  |  |  |                     enum: components, | 
					
						
							|  |  |  |                   }, | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |               }, | 
					
						
							|  |  |  |               schema | 
					
						
							|  |  |  |             ); | 
					
						
							|  |  |  |           }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           acc.properties[current] = { | 
					
						
							|  |  |  |             type: 'array', | 
					
						
							|  |  |  |             items: { | 
					
						
							|  |  |  |               oneOf: cmps, | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             minItems: min, | 
					
						
							|  |  |  |             maxItems: max, | 
					
						
							|  |  |  |           }; | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |         } else { | 
					
						
							|  |  |  |           acc.properties[current] = { | 
					
						
							|  |  |  |             type, | 
					
						
							|  |  |  |             description, | 
					
						
							|  |  |  |             default: defaultValue, | 
					
						
							|  |  |  |             minimum, | 
					
						
							|  |  |  |             maxmimun, | 
					
						
							|  |  |  |             maxLength, | 
					
						
							|  |  |  |             minLength, | 
					
						
							|  |  |  |             enum: enumeration, | 
					
						
							|  |  |  |           }; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return acc; | 
					
						
							|  |  |  |       }, | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |       { required: ['id'], properties: { id: { type: 'string' } } } | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |     ); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   generatePluginDocumentation: function(pluginName, routes) { | 
					
						
							|  |  |  |     return routes.reduce((acc, current) => { | 
					
						
							|  |  |  |       const { | 
					
						
							|  |  |  |         config: { description, prefix }, | 
					
						
							|  |  |  |       } = current; | 
					
						
							|  |  |  |       const endPoint = | 
					
						
							|  |  |  |         prefix === undefined | 
					
						
							|  |  |  |           ? this.formatApiEndPoint(`/${pluginName}${current.path}`) | 
					
						
							|  |  |  |           : this.formatApiEndPoint(`${prefix}${current.path}`); | 
					
						
							| 
									
										
										
										
											2019-03-05 22:03:52 +02:00
										 |  |  |       let verb; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (Array.isArray(current.method)) { | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |         verb = current.method.map(method => method.toLowerCase()); | 
					
						
							| 
									
										
										
										
											2019-03-05 22:03:52 +02:00
										 |  |  |       } else { | 
					
						
							|  |  |  |         verb = current.method.toLowerCase(); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-15 18:57:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       const actionType = _.get(current, ['config', 'tag', 'actionType'], ''); | 
					
						
							|  |  |  |       let key; | 
					
						
							|  |  |  |       let tags; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (_.isObject(current.config.tag)) { | 
					
						
							|  |  |  |         const { name, plugin } = current.config.tag; | 
					
						
							|  |  |  |         key = plugin ? `${plugin}-${name}` : name; | 
					
						
							|  |  |  |         tags = plugin ? [this.formatTag(plugin, name)] : [name]; | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         const tag = current.config.tag; | 
					
						
							|  |  |  |         key = !_.isEmpty(tag) ? tag : 'unclassified'; | 
					
						
							|  |  |  |         tags = !_.isEmpty(tag) ? [tag] : ['Unclassified']; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |       const hasDefaultDocumentation = this.checkIfPluginDefaultDocumentFileExists(pluginName, key); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       const defaultDocumentation = hasDefaultDocumentation | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |         ? this.getPluginDefaultVerbDocumentation(pluginName, key, endPoint, verb) | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |         : null; | 
					
						
							|  |  |  |       const verbObject = { | 
					
						
							|  |  |  |         deprecated: false, | 
					
						
							|  |  |  |         description, | 
					
						
							|  |  |  |         responses: this.generatePluginVerbResponses(current), | 
					
						
							|  |  |  |         summary: '', | 
					
						
							|  |  |  |         tags, | 
					
						
							|  |  |  |       }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       _.set(acc, [key, 'paths', endPoint, verb], verbObject); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       const parameters = this.generateVerbParameters( | 
					
						
							|  |  |  |         verb, | 
					
						
							|  |  |  |         actionType, | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |         `/${pluginName}${current.path}` | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (_.isEmpty(defaultDocumentation)) { | 
					
						
							| 
									
										
										
										
											2019-03-05 22:03:52 +02:00
										 |  |  |         if (!verb.includes('post')) { | 
					
						
							|  |  |  |           if (Array.isArray(verb)) { | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |             verb.forEach(method => { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |               _.set(acc, [key, 'paths', endPoint, method, 'parameters'], parameters); | 
					
						
							| 
									
										
										
										
											2019-03-05 22:03:52 +02:00
										 |  |  |             }); | 
					
						
							|  |  |  |           } else { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |             _.set(acc, [key, 'paths', endPoint, verb, 'parameters'], parameters); | 
					
						
							| 
									
										
										
										
											2019-03-05 22:03:52 +02:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-05 22:03:52 +02:00
										 |  |  |         if (verb.includes('post') || verb.includes('put')) { | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |           let requestBody; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if (actionType === 'create' || actionType === 'update') { | 
					
						
							|  |  |  |             const { name, plugin } = _.isObject(current.config.tag) | 
					
						
							|  |  |  |               ? current.config.tag | 
					
						
							|  |  |  |               : { tag: current.config.tag }; | 
					
						
							|  |  |  |             const $ref = plugin | 
					
						
							|  |  |  |               ? `#/components/schemas/New${this.formatTag(plugin, name, true)}` | 
					
						
							|  |  |  |               : `#/components/schemas/New${_.upperFirst(name)}`; | 
					
						
							|  |  |  |             requestBody = { | 
					
						
							|  |  |  |               description: '', | 
					
						
							|  |  |  |               required: true, | 
					
						
							|  |  |  |               content: { | 
					
						
							|  |  |  |                 'application/json': { | 
					
						
							|  |  |  |                   schema: { | 
					
						
							|  |  |  |                     $ref, | 
					
						
							|  |  |  |                   }, | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |               }, | 
					
						
							|  |  |  |             }; | 
					
						
							|  |  |  |           } else { | 
					
						
							|  |  |  |             requestBody = { | 
					
						
							|  |  |  |               description: '', | 
					
						
							|  |  |  |               required: true, | 
					
						
							|  |  |  |               content: { | 
					
						
							|  |  |  |                 'application/json': { | 
					
						
							|  |  |  |                   schema: { | 
					
						
							|  |  |  |                     properties: { | 
					
						
							|  |  |  |                       foo: { | 
					
						
							|  |  |  |                         type: 'string', | 
					
						
							|  |  |  |                       }, | 
					
						
							|  |  |  |                     }, | 
					
						
							|  |  |  |                   }, | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |               }, | 
					
						
							|  |  |  |             }; | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-04-15 18:57:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-05 22:03:52 +02:00
										 |  |  |           if (Array.isArray(verb)) { | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |             verb.forEach(method => { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |               _.set(acc, [key, 'paths', endPoint, method, 'requestBody'], requestBody); | 
					
						
							| 
									
										
										
										
											2019-03-05 22:03:52 +02:00
										 |  |  |             }); | 
					
						
							|  |  |  |           } else { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |             _.set(acc, [key, 'paths', endPoint, verb, 'requestBody'], requestBody); | 
					
						
							| 
									
										
										
										
											2019-03-05 22:03:52 +02:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return acc; | 
					
						
							|  |  |  |     }, {}); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   generatePluginResponseSchema: function(tag) { | 
					
						
							|  |  |  |     const { actionType, name, plugin } = _.isObject(tag) ? tag : { tag }; | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |     const getter = plugin ? ['plugins', plugin, 'models', name.toLowerCase()] : ['models', name]; | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |     const isModelRelated = | 
					
						
							|  |  |  |       _.get(strapi, getter) !== undefined && | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |       ['find', 'findOne', 'create', 'search', 'update', 'destroy', 'count'].includes(actionType); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |     const $ref = plugin | 
					
						
							|  |  |  |       ? `#/components/schemas/${this.formatTag(plugin, name, true)}` | 
					
						
							|  |  |  |       : `#/components/schemas/${_.upperFirst(name)}`; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (isModelRelated) { | 
					
						
							|  |  |  |       switch (actionType) { | 
					
						
							|  |  |  |         case 'find': | 
					
						
							|  |  |  |           return { | 
					
						
							|  |  |  |             type: 'array', | 
					
						
							|  |  |  |             items: { | 
					
						
							|  |  |  |               $ref, | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |           }; | 
					
						
							|  |  |  |         case 'count': | 
					
						
							|  |  |  |           return { | 
					
						
							|  |  |  |             properties: { | 
					
						
							|  |  |  |               count: { | 
					
						
							|  |  |  |                 type: 'integer', | 
					
						
							|  |  |  |               }, | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |           }; | 
					
						
							|  |  |  |         case 'findOne': | 
					
						
							|  |  |  |         case 'update': | 
					
						
							|  |  |  |         case 'create': | 
					
						
							|  |  |  |           return { | 
					
						
							|  |  |  |             $ref, | 
					
						
							|  |  |  |           }; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |           return { | 
					
						
							|  |  |  |             properties: { | 
					
						
							|  |  |  |               foo: { | 
					
						
							|  |  |  |                 type: 'string', | 
					
						
							|  |  |  |               }, | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |           }; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return { | 
					
						
							|  |  |  |       properties: { | 
					
						
							|  |  |  |         foo: { | 
					
						
							|  |  |  |           type: 'string', | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   generatePluginVerbResponses: function(routeObject) { | 
					
						
							|  |  |  |     const { | 
					
						
							|  |  |  |       config: { tag }, | 
					
						
							|  |  |  |     } = routeObject; | 
					
						
							|  |  |  |     const actionType = _.get(tag, 'actionType'); | 
					
						
							|  |  |  |     let schema; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!tag || !actionType) { | 
					
						
							|  |  |  |       schema = { | 
					
						
							|  |  |  |         properties: { | 
					
						
							|  |  |  |           foo: { | 
					
						
							|  |  |  |             type: 'string', | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       }; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       schema = this.generatePluginResponseSchema(tag); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-20 14:03:51 +01:00
										 |  |  |     const response = { | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       200: { | 
					
						
							|  |  |  |         description: 'response', | 
					
						
							|  |  |  |         content: { | 
					
						
							|  |  |  |           'application/json': { | 
					
						
							|  |  |  |             schema, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       403: { | 
					
						
							|  |  |  |         description: 'Forbidden', | 
					
						
							|  |  |  |         content: { | 
					
						
							|  |  |  |           'application/json': { | 
					
						
							|  |  |  |             schema: { | 
					
						
							|  |  |  |               $ref: '#/components/schemas/Error', | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       404: { | 
					
						
							|  |  |  |         description: 'Not found', | 
					
						
							|  |  |  |         content: { | 
					
						
							|  |  |  |           'application/json': { | 
					
						
							|  |  |  |             schema: { | 
					
						
							|  |  |  |               $ref: '#/components/schemas/Error', | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2020-03-20 14:03:51 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const { generateDefaultResponse } = strapi.plugins.documentation.config['x-strapi-config']; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (generateDefaultResponse) { | 
					
						
							|  |  |  |       response.default = { | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |         description: 'unexpected error', | 
					
						
							|  |  |  |         content: { | 
					
						
							|  |  |  |           'application/json': { | 
					
						
							|  |  |  |             schema: { | 
					
						
							|  |  |  |               $ref: '#/components/schemas/Error', | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |       }; | 
					
						
							| 
									
										
										
										
											2020-03-20 14:03:51 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return response; | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Create the response object https://swagger.io/specification/#responsesObject
 | 
					
						
							|  |  |  |    * @param {String} verb | 
					
						
							|  |  |  |    * @param {Object} routeObject | 
					
						
							|  |  |  |    * @param {String} tag | 
					
						
							|  |  |  |    * @returns {Object} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   generateResponses: function(verb, routeObject, tag) { | 
					
						
							|  |  |  |     const endPoint = routeObject.path.split('/')[1]; | 
					
						
							|  |  |  |     const description = this.generateResponseDescription(verb, tag, endPoint); | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |     const schema = this.generateResponseSchema(verb, routeObject, tag, endPoint); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-20 14:03:51 +01:00
										 |  |  |     const response = { | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       200: { | 
					
						
							|  |  |  |         description, | 
					
						
							|  |  |  |         content: { | 
					
						
							|  |  |  |           'application/json': { | 
					
						
							|  |  |  |             schema, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       403: { | 
					
						
							|  |  |  |         description: 'Forbidden', | 
					
						
							|  |  |  |         content: { | 
					
						
							|  |  |  |           'application/json': { | 
					
						
							|  |  |  |             schema: { | 
					
						
							|  |  |  |               $ref: '#/components/schemas/Error', | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       404: { | 
					
						
							|  |  |  |         description: 'Not found', | 
					
						
							|  |  |  |         content: { | 
					
						
							|  |  |  |           'application/json': { | 
					
						
							|  |  |  |             schema: { | 
					
						
							|  |  |  |               $ref: '#/components/schemas/Error', | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2020-03-20 14:03:51 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const { generateDefaultResponse } = strapi.plugins.documentation.config['x-strapi-config']; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (generateDefaultResponse) { | 
					
						
							|  |  |  |       response.default = { | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |         description: 'unexpected error', | 
					
						
							|  |  |  |         content: { | 
					
						
							|  |  |  |           'application/json': { | 
					
						
							|  |  |  |             schema: { | 
					
						
							|  |  |  |               $ref: '#/components/schemas/Error', | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |       }; | 
					
						
							| 
									
										
										
										
											2020-03-20 14:03:51 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return response; | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Retrieve all privates attributes from a model | 
					
						
							|  |  |  |    * @param {Object} attributes | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   getPrivateAttributes: function(attributes) { | 
					
						
							|  |  |  |     const privateAttributes = Object.keys(attributes).reduce((acc, current) => { | 
					
						
							|  |  |  |       if (attributes[current].private === true) { | 
					
						
							|  |  |  |         acc.push(current); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       return acc; | 
					
						
							|  |  |  |     }, []); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return privateAttributes; | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Create a component object with the model's attributes and relations | 
					
						
							|  |  |  |    * Refer to https://swagger.io/docs/specification/components/
 | 
					
						
							|  |  |  |    * @param {String} tag | 
					
						
							|  |  |  |    * @returns {Object} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   generateResponseComponent: function(tag, pluginName = '', isPlugin = false) { | 
					
						
							|  |  |  |     // The component's name have to be capitalised
 | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |     const [plugin, name] = isPlugin ? this.getModelAndNameForPlugin(tag, pluginName) : [null, null]; | 
					
						
							|  |  |  |     const upperFirstTag = isPlugin ? this.formatTag(plugin, name, true) : _.upperFirst(tag); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |     const attributesGetter = isPlugin | 
					
						
							|  |  |  |       ? [...this.getModelForPlugin(tag, plugin), 'attributes'] | 
					
						
							|  |  |  |       : ['models', tag, 'attributes']; | 
					
						
							|  |  |  |     const associationGetter = isPlugin | 
					
						
							|  |  |  |       ? [...this.getModelForPlugin(tag, plugin), 'associations'] | 
					
						
							|  |  |  |       : ['models', tag, 'associations']; | 
					
						
							|  |  |  |     const attributesObject = _.get(strapi, attributesGetter); | 
					
						
							|  |  |  |     const privateAttributes = this.getPrivateAttributes(attributesObject); | 
					
						
							|  |  |  |     const modelAssociations = _.get(strapi, associationGetter); | 
					
						
							|  |  |  |     const { attributes } = this.getModelAttributes(attributesObject); | 
					
						
							|  |  |  |     const associationsWithUpload = modelAssociations | 
					
						
							|  |  |  |       .filter(association => { | 
					
						
							|  |  |  |         return association.plugin === 'upload'; | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |       .map(obj => obj.alias); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // We always create two nested components from the main one
 | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |     const mainComponent = this.generateMainComponent(attributes, modelAssociations, upperFirstTag); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Get Component that doesn't display the privates attributes since a mask is applied
 | 
					
						
							|  |  |  |     // Please refer https://github.com/strapi/strapi/blob/585800b7b98093f596759b296a43f89c491d4f4f/packages/strapi/lib/middlewares/mask/index.js#L92-L100
 | 
					
						
							|  |  |  |     const getComponent = Object.keys(mainComponent.properties).reduce( | 
					
						
							|  |  |  |       (acc, current) => { | 
					
						
							|  |  |  |         if (privateAttributes.indexOf(current) === -1) { | 
					
						
							|  |  |  |           acc.properties[current] = mainComponent.properties[current]; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return acc; | 
					
						
							|  |  |  |       }, | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |       { required: mainComponent.required, properties: {} } | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Special component only for POST || PUT verbs since the upload is made with a different route
 | 
					
						
							|  |  |  |     const postComponent = Object.keys(mainComponent).reduce((acc, current) => { | 
					
						
							|  |  |  |       if (current === 'required') { | 
					
						
							|  |  |  |         const required = mainComponent.required.slice().filter(attr => { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |           return associationsWithUpload.indexOf(attr) === -1 && attr !== 'id' && attr !== '_id'; | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (required.length > 0) { | 
					
						
							|  |  |  |           acc.required = required; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (current === 'properties') { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |         const properties = Object.keys(mainComponent.properties).reduce((acc, current) => { | 
					
						
							|  |  |  |           if ( | 
					
						
							|  |  |  |             associationsWithUpload.indexOf(current) === -1 && | 
					
						
							|  |  |  |             current !== 'id' && | 
					
						
							|  |  |  |             current !== '_id' | 
					
						
							|  |  |  |           ) { | 
					
						
							|  |  |  |             // The post request shouldn't include nested relations of type 2
 | 
					
						
							|  |  |  |             // For instance if a product has many tags
 | 
					
						
							|  |  |  |             // we expect to find an array of tags objects containing other relations in the get response
 | 
					
						
							|  |  |  |             // and since we use to getComponent to generate this one we need to
 | 
					
						
							|  |  |  |             // remove this object since we only send an array of tag ids.
 | 
					
						
							|  |  |  |             if (_.find(modelAssociations, ['alias', current])) { | 
					
						
							|  |  |  |               const isArrayProperty = | 
					
						
							|  |  |  |                 _.get(mainComponent, ['properties', current, 'type']) !== undefined; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |               if (isArrayProperty) { | 
					
						
							|  |  |  |                 acc[current] = { type: 'array', items: { type: 'string' } }; | 
					
						
							| 
									
										
										
										
											2018-12-07 00:14:59 +01:00
										 |  |  |               } else { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |                 acc[current] = { type: 'string' }; | 
					
						
							| 
									
										
										
										
											2018-12-07 00:14:59 +01:00
										 |  |  |               } | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |             } else { | 
					
						
							|  |  |  |               // If the field is not an association we take the one from the component
 | 
					
						
							|  |  |  |               acc[current] = mainComponent.properties[current]; | 
					
						
							| 
									
										
										
										
											2018-12-07 00:14:59 +01:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |           return acc; | 
					
						
							|  |  |  |         }, {}); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         acc.properties = properties; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return acc; | 
					
						
							|  |  |  |     }, {}); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return { | 
					
						
							|  |  |  |       components: { | 
					
						
							|  |  |  |         schemas: { | 
					
						
							|  |  |  |           [upperFirstTag]: getComponent, | 
					
						
							|  |  |  |           [`New${upperFirstTag}`]: postComponent, | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Generate a better description for a response when we can guess what's the user is going to retrieve | 
					
						
							|  |  |  |    * @param {String} verb | 
					
						
							|  |  |  |    * @param {String} tag | 
					
						
							|  |  |  |    * @param {String} endPoint | 
					
						
							|  |  |  |    * @returns {String} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   generateResponseDescription: function(verb, tag, endPoint) { | 
					
						
							|  |  |  |     const isModelRelated = strapi.models[tag] !== undefined && tag === endPoint; | 
					
						
							| 
									
										
										
										
											2019-04-15 18:57:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-05 20:55:46 +02:00
										 |  |  |     if (Array.isArray(verb)) { | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |       verb = verb.map(method => method.toLocaleLowerCase()); | 
					
						
							| 
									
										
										
										
											2019-03-05 20:55:46 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |     if (verb.includes('get') || verb.includes('put') || verb.includes('post')) { | 
					
						
							| 
									
										
										
										
											2019-03-05 21:12:35 +02:00
										 |  |  |       return isModelRelated ? `Retrieve ${tag} document(s)` : 'response'; | 
					
						
							|  |  |  |     } else if (verb.includes('delete')) { | 
					
						
							|  |  |  |       return isModelRelated | 
					
						
							|  |  |  |         ? `deletes a single ${tag} based on the ID supplied` | 
					
						
							|  |  |  |         : 'deletes a single record based on the ID supplied'; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       return 'response'; | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * For each response generate its schema | 
					
						
							|  |  |  |    * Its schema is either a component when we know what the routes returns otherwise, it returns a dummy schema | 
					
						
							|  |  |  |    * that the user will modify later | 
					
						
							|  |  |  |    * @param {String} verb | 
					
						
							| 
									
										
										
										
											2018-12-13 16:29:57 +01:00
										 |  |  |    * @param {Object} route | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |    * @param {String} tag | 
					
						
							|  |  |  |    * @param {String} endPoint | 
					
						
							|  |  |  |    * @returns {Object} | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2018-12-13 16:29:57 +01:00
										 |  |  |   generateResponseSchema: function(verb, routeObject, tag) { | 
					
						
							|  |  |  |     const { handler } = routeObject; | 
					
						
							|  |  |  |     let [controller, handlerMethod] = handler.split('.'); | 
					
						
							|  |  |  |     let upperFirstTag = _.upperFirst(tag); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (verb === 'delete') { | 
					
						
							|  |  |  |       return { | 
					
						
							|  |  |  |         type: 'integer', | 
					
						
							|  |  |  |         format: 'int64', | 
					
						
							|  |  |  |       }; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-12-13 16:29:57 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // A tag key might be added to a route to tell if a custom endPoint in an api/<model>/config/routes.json
 | 
					
						
							|  |  |  |     // Retrieves data from another model it is a faster way to generate the response
 | 
					
						
							|  |  |  |     const routeReferenceTag = _.get(routeObject, ['config', 'tag']); | 
					
						
							|  |  |  |     let isModelRelated = false; | 
					
						
							|  |  |  |     const shouldCheckIfACustomEndPointReferencesAnotherModel = | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |       _.isObject(routeReferenceTag) && !_.isEmpty(_.get(routeReferenceTag, 'name')); | 
					
						
							| 
									
										
										
										
											2018-12-13 16:29:57 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (shouldCheckIfACustomEndPointReferencesAnotherModel) { | 
					
						
							|  |  |  |       const { actionType, name, plugin } = routeReferenceTag; | 
					
						
							|  |  |  |       // A model could be in either a plugin or the api folder
 | 
					
						
							|  |  |  |       // The path is different depending on the case
 | 
					
						
							|  |  |  |       const getter = !_.isEmpty(plugin) | 
					
						
							|  |  |  |         ? ['plugins', plugin, 'models', name.toLowerCase()] | 
					
						
							|  |  |  |         : ['models', name.toLowerCase()]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // An actionType key might be added to the tag object to guide the algorithm is generating an automatic response
 | 
					
						
							|  |  |  |       const isKnownAction = [ | 
					
						
							|  |  |  |         'find', | 
					
						
							|  |  |  |         'findOne', | 
					
						
							|  |  |  |         'create', | 
					
						
							|  |  |  |         'search', | 
					
						
							|  |  |  |         'update', | 
					
						
							|  |  |  |         'destroy', | 
					
						
							|  |  |  |         'count', | 
					
						
							|  |  |  |       ].includes(actionType); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Check if a route points to a model
 | 
					
						
							|  |  |  |       isModelRelated = _.get(strapi, getter) !== undefined && isKnownAction; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (isModelRelated && isKnownAction) { | 
					
						
							|  |  |  |         // We need to change the handlerMethod name if it is know to generate the good schema
 | 
					
						
							|  |  |  |         handlerMethod = actionType; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // This is to retrieve the correct component if a custom endpoints references a plugin model
 | 
					
						
							|  |  |  |         if (!_.isEmpty(plugin)) { | 
					
						
							|  |  |  |           upperFirstTag = this.formatTag(plugin, name, true); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       // Normal way there's no tag object
 | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |       isModelRelated = strapi.models[tag] !== undefined && tag === _.lowerCase(controller); | 
					
						
							| 
									
										
										
										
											2018-12-13 16:29:57 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // We create a component when we are sure that we can 'guess' what's needed to be sent
 | 
					
						
							|  |  |  |     // https://swagger.io/specification/#referenceObject
 | 
					
						
							|  |  |  |     if (isModelRelated) { | 
					
						
							|  |  |  |       switch (handlerMethod) { | 
					
						
							|  |  |  |         case 'find': | 
					
						
							|  |  |  |           return { | 
					
						
							|  |  |  |             type: 'array', | 
					
						
							|  |  |  |             items: { | 
					
						
							|  |  |  |               $ref: `#/components/schemas/${upperFirstTag}`, | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |           }; | 
					
						
							|  |  |  |         case 'count': | 
					
						
							|  |  |  |           return { | 
					
						
							|  |  |  |             properties: { | 
					
						
							|  |  |  |               count: { | 
					
						
							|  |  |  |                 type: 'integer', | 
					
						
							|  |  |  |               }, | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |           }; | 
					
						
							|  |  |  |         case 'findOne': | 
					
						
							|  |  |  |         case 'update': | 
					
						
							|  |  |  |         case 'create': | 
					
						
							|  |  |  |           return { | 
					
						
							|  |  |  |             $ref: `#/components/schemas/${upperFirstTag}`, | 
					
						
							|  |  |  |           }; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |           return { | 
					
						
							|  |  |  |             properties: { | 
					
						
							|  |  |  |               foo: { | 
					
						
							|  |  |  |                 type: 'string', | 
					
						
							|  |  |  |               }, | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |           }; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return { | 
					
						
							|  |  |  |       properties: { | 
					
						
							|  |  |  |         foo: { | 
					
						
							|  |  |  |           type: 'string', | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   generateTags: function(name, docName, tag = '', isPlugin = false) { | 
					
						
							|  |  |  |     return [ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         name: isPlugin ? tag : _.upperFirst(docName), | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |     ]; | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Add a default description when it's implied | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * @param {String} verb | 
					
						
							|  |  |  |    * @param {String} handler | 
					
						
							|  |  |  |    * @param {String} tag | 
					
						
							|  |  |  |    * @param {String} endPoint | 
					
						
							|  |  |  |    * @returns {String} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   generateVerbDescription: (verb, handler, tag, endPoint, description) => { | 
					
						
							|  |  |  |     const isModelRelated = strapi.models[tag] !== undefined && tag === endPoint; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (description) { | 
					
						
							|  |  |  |       return description; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-15 18:57:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-05 20:55:46 +02:00
										 |  |  |     if (Array.isArray(verb)) { | 
					
						
							|  |  |  |       const [, controllerMethod] = handler.split('.'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |       if ((verb.includes('get') && verb.includes('post')) || controllerMethod === 'findOrCreate') { | 
					
						
							| 
									
										
										
										
											2019-03-05 20:55:46 +02:00
										 |  |  |         return `Find or create ${tag} record`; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if ( | 
					
						
							|  |  |  |         (verb.includes('put') && verb.includes('post')) || | 
					
						
							|  |  |  |         controllerMethod === 'createOrUpdate' | 
					
						
							|  |  |  |       ) { | 
					
						
							|  |  |  |         return `Create or update ${tag} record`; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return ''; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     switch (verb) { | 
					
						
							|  |  |  |       case 'get': { | 
					
						
							|  |  |  |         const [, controllerMethod] = handler.split('.'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (isModelRelated) { | 
					
						
							|  |  |  |           switch (controllerMethod) { | 
					
						
							|  |  |  |             case 'count': | 
					
						
							|  |  |  |               return `Retrieve the numver of ${tag} documents`; | 
					
						
							|  |  |  |             case 'findOne': | 
					
						
							|  |  |  |               return `Find one ${tag} record`; | 
					
						
							|  |  |  |             case 'find': | 
					
						
							|  |  |  |               return `Find all the ${tag}'s records`; | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |               return ''; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return ''; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       case 'delete': | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |         return isModelRelated ? `Delete a single ${tag} record` : 'Delete a record'; | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       case 'post': | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |         return isModelRelated ? `Create a new ${tag} record` : 'Create a new record'; | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       case 'put': | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |         return isModelRelated ? `Update a single ${tag} record` : 'Update a record'; | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       case 'patch': | 
					
						
							|  |  |  |         return ''; | 
					
						
							|  |  |  |       case 'head': | 
					
						
							|  |  |  |         return ''; | 
					
						
							|  |  |  |       default: | 
					
						
							|  |  |  |         return ''; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Generate the verb parameters object | 
					
						
							|  |  |  |    * Refer to https://swagger.io/specification/#pathItemObject
 | 
					
						
							|  |  |  |    * @param {Sting} verb | 
					
						
							|  |  |  |    * @param {String} controllerMethod | 
					
						
							|  |  |  |    * @param {String} endPoint | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   generateVerbParameters: function(verb, controllerMethod, endPoint) { | 
					
						
							|  |  |  |     const params = pathToRegexp | 
					
						
							|  |  |  |       .parse(endPoint) | 
					
						
							|  |  |  |       .filter(token => _.isObject(token)) | 
					
						
							|  |  |  |       .reduce((acc, current) => { | 
					
						
							|  |  |  |         const param = { | 
					
						
							|  |  |  |           name: current.name, | 
					
						
							|  |  |  |           in: 'path', | 
					
						
							|  |  |  |           description: '', | 
					
						
							|  |  |  |           deprecated: false, | 
					
						
							|  |  |  |           required: true, | 
					
						
							|  |  |  |           schema: { type: 'string' }, | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return acc.concat(param); | 
					
						
							|  |  |  |       }, []); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (verb === 'get' && controllerMethod === 'find') { | 
					
						
							|  |  |  |       // parametersOptions corresponds to this section
 | 
					
						
							|  |  |  |       // of the documentation https://strapi.io/documentation/guides/filters.html
 | 
					
						
							|  |  |  |       return [...params, ...parametersOptions]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return params; | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Retrieve the apis in /api | 
					
						
							|  |  |  |    * @returns {Array} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   getApis: () => { | 
					
						
							|  |  |  |     return Object.keys(strapi.api || {}); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   getAPIOverrideComponentsDocumentation: function(apiName, docName) { | 
					
						
							|  |  |  |     try { | 
					
						
							|  |  |  |       const documentation = JSON.parse( | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |         fs.readFileSync(this.getAPIOverrideDocumentationPath(apiName, docName), 'utf8') | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return _.get(documentation, 'components', null); | 
					
						
							|  |  |  |     } catch (err) { | 
					
						
							|  |  |  |       return null; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   getAPIDefaultTagsDocumentation: function(name, docName) { | 
					
						
							|  |  |  |     try { | 
					
						
							|  |  |  |       const documentation = JSON.parse( | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |         fs.readFileSync(this.getAPIOverrideDocumentationPath(name, docName), 'utf8') | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return _.get(documentation, 'tags', null); | 
					
						
							|  |  |  |     } catch (err) { | 
					
						
							|  |  |  |       return null; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   getAPIDefaultVerbDocumentation: function(apiName, docName, routePath, verb) { | 
					
						
							|  |  |  |     try { | 
					
						
							|  |  |  |       const documentation = JSON.parse( | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |         fs.readFileSync(this.getAPIOverrideDocumentationPath(apiName, docName), 'utf8') | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return _.get(documentation, ['paths', routePath, verb], null); | 
					
						
							|  |  |  |     } catch (err) { | 
					
						
							|  |  |  |       return null; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   getAPIOverrideDocumentationPath: function(apiName, docName) { | 
					
						
							|  |  |  |     return path.join( | 
					
						
							|  |  |  |       strapi.config.appPath, | 
					
						
							|  |  |  |       'api', | 
					
						
							|  |  |  |       apiName, | 
					
						
							|  |  |  |       'documentation', | 
					
						
							|  |  |  |       'overrides', | 
					
						
							|  |  |  |       this.getDocumentationVersion(), | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |       `${docName}.json` | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |     ); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Given an api retrieve its endpoints | 
					
						
							|  |  |  |    * @param {String} | 
					
						
							|  |  |  |    * @returns {Array} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   getApiRoutes: apiName => { | 
					
						
							|  |  |  |     return _.get(strapi, ['api', apiName, 'config', 'routes'], []); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   getDocumentationOverridesPath: function(apiName) { | 
					
						
							|  |  |  |     return path.join( | 
					
						
							|  |  |  |       strapi.config.appPath, | 
					
						
							|  |  |  |       'api', | 
					
						
							|  |  |  |       apiName, | 
					
						
							|  |  |  |       'documentation', | 
					
						
							|  |  |  |       this.getDocumentationVersion(), | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |       'overrides' | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |     ); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Given an api from /api retrieve its version directory | 
					
						
							|  |  |  |    * @param {String} apiName | 
					
						
							|  |  |  |    * @returns {Path} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   getDocumentationPath: function(apiName) { | 
					
						
							|  |  |  |     return path.join( | 
					
						
							|  |  |  |       strapi.config.appPath, | 
					
						
							|  |  |  |       'api', | 
					
						
							|  |  |  |       apiName, | 
					
						
							|  |  |  |       'documentation', | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |       this.getDocumentationVersion() | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |     ); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   getFullDocumentationPath: () => { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |     return path.join(strapi.config.appPath, 'extensions', 'documentation', 'documentation'); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Retrieve the plugin's configuration version | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   getDocumentationVersion: () => { | 
					
						
							|  |  |  |     const version = strapi.plugins['documentation'].config.info.version; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return version; | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Retrieve the documentation plugin documentation directory | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |   getMergedDocumentationPath: function(version = this.getDocumentationVersion()) { | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |     return path.join( | 
					
						
							|  |  |  |       strapi.config.appPath, | 
					
						
							|  |  |  |       'extensions', | 
					
						
							|  |  |  |       'documentation', | 
					
						
							|  |  |  |       'documentation', | 
					
						
							|  |  |  |       version | 
					
						
							|  |  |  |     ); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Retrieve the model's attributes | 
					
						
							|  |  |  |    * @param {Objet} modelAttributes | 
					
						
							|  |  |  |    * @returns {Object} { associations: [{ name: 'foo', getter: [], tag: 'foos' }], attributes } | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   getModelAttributes: function(modelAttributes) { | 
					
						
							|  |  |  |     const associations = []; | 
					
						
							|  |  |  |     const attributes = Object.keys(modelAttributes) | 
					
						
							|  |  |  |       .map(attr => { | 
					
						
							|  |  |  |         const attribute = modelAttributes[attr]; | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |         const isField = !_.has(attribute, 'model') && !_.has(attribute, 'collection'); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (!isField) { | 
					
						
							|  |  |  |           const name = attribute.model || attribute.collection; | 
					
						
							|  |  |  |           const getter = | 
					
						
							|  |  |  |             attribute.plugin !== undefined | 
					
						
							|  |  |  |               ? ['plugins', attribute.plugin, 'models', name, 'attributes'] | 
					
						
							|  |  |  |               : ['models', name, 'attributes']; | 
					
						
							|  |  |  |           associations.push({ name, getter, tag: attr }); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return attr; | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |       .reduce((acc, current) => { | 
					
						
							|  |  |  |         acc[current] = modelAttributes[current]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return acc; | 
					
						
							|  |  |  |       }, {}); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return { associations, attributes }; | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Refer to https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md#dataTypes
 | 
					
						
							|  |  |  |    * @param {String} type | 
					
						
							|  |  |  |    * @returns {String} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   getType: type => { | 
					
						
							|  |  |  |     switch (type) { | 
					
						
							|  |  |  |       case 'string': | 
					
						
							|  |  |  |       case 'byte': | 
					
						
							|  |  |  |       case 'binary': | 
					
						
							|  |  |  |       case 'password': | 
					
						
							|  |  |  |       case 'email': | 
					
						
							|  |  |  |       case 'text': | 
					
						
							|  |  |  |       case 'enumeration': | 
					
						
							|  |  |  |       case 'date': | 
					
						
							| 
									
										
										
										
											2019-09-10 16:27:38 +02:00
										 |  |  |       case 'richtext': | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |         return 'string'; | 
					
						
							|  |  |  |       case 'float': | 
					
						
							|  |  |  |       case 'decimal': | 
					
						
							|  |  |  |       case 'double': | 
					
						
							|  |  |  |         return 'number'; | 
					
						
							|  |  |  |       case 'integer': | 
					
						
							| 
									
										
										
										
											2019-01-21 05:11:48 -07:00
										 |  |  |       case 'biginteger': | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       case 'long': | 
					
						
							|  |  |  |         return 'integer'; | 
					
						
							| 
									
										
										
										
											2019-09-10 16:52:51 +02:00
										 |  |  |       case 'json': | 
					
						
							|  |  |  |         return 'object'; | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       default: | 
					
						
							|  |  |  |         return type; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |   getPluginDefaultVerbDocumentation: function(pluginName, docName, routePath, verb) { | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |     try { | 
					
						
							|  |  |  |       const documentation = JSON.parse( | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |         fs.readFileSync(this.getPluginOverrideDocumentationPath(pluginName, docName), 'utf8') | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return _.get(documentation, ['paths', routePath, verb], null); | 
					
						
							|  |  |  |     } catch (err) { | 
					
						
							|  |  |  |       return null; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   getPluginDefaultTagsDocumentation: function(pluginName, docName) { | 
					
						
							|  |  |  |     try { | 
					
						
							|  |  |  |       const documentation = JSON.parse( | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |         fs.readFileSync(this.getPluginOverrideDocumentationPath(pluginName, docName), 'utf8') | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return _.get(documentation, ['tags'], null); | 
					
						
							|  |  |  |     } catch (err) { | 
					
						
							|  |  |  |       return null; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   getPluginOverrideComponents: function(pluginName, docName) { | 
					
						
							|  |  |  |     try { | 
					
						
							|  |  |  |       const documentation = JSON.parse( | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |         fs.readFileSync(this.getPluginOverrideDocumentationPath(pluginName, docName), 'utf8') | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return _.get(documentation, 'components', null); | 
					
						
							|  |  |  |     } catch (err) { | 
					
						
							|  |  |  |       return null; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   getPluginOverrideDocumentationPath: function(pluginName, docName) { | 
					
						
							|  |  |  |     const defaultPath = path.join( | 
					
						
							|  |  |  |       strapi.config.appPath, | 
					
						
							| 
									
										
										
										
											2019-04-15 18:57:58 +02:00
										 |  |  |       'extensions', | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       pluginName, | 
					
						
							|  |  |  |       'documentation', | 
					
						
							|  |  |  |       this.getDocumentationVersion(), | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |       'overrides' | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (docName) { | 
					
						
							|  |  |  |       return path.resolve(defaultPath, `${docName.json}`); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       return defaultPath; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Given a plugin retrieve its documentation version | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   getPluginDocumentationPath: function(pluginName) { | 
					
						
							|  |  |  |     return path.join( | 
					
						
							|  |  |  |       strapi.config.appPath, | 
					
						
							| 
									
										
										
										
											2019-04-15 18:57:58 +02:00
										 |  |  |       'extensions', | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |       pluginName, | 
					
						
							|  |  |  |       'documentation', | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |       this.getDocumentationVersion() | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |     ); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Retrieve all plugins that have a description inside one of its route | 
					
						
							|  |  |  |    * @return {Arrray} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   getPluginsWithDocumentationNeeded: function() { | 
					
						
							|  |  |  |     return Object.keys(strapi.plugins).reduce((acc, current) => { | 
					
						
							|  |  |  |       const isDocumentationNeeded = this.isPluginDocumentationNeeded(current); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (isDocumentationNeeded) { | 
					
						
							|  |  |  |         return acc.concat(current); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return acc; | 
					
						
							|  |  |  |     }, []); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Retrieve all the routes that have a description from a plugin | 
					
						
							|  |  |  |    * @param {String} pluginName | 
					
						
							|  |  |  |    * @returns {Array} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   getPluginRoutesWithDescription: function(pluginName) { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |     return _.get(strapi, ['plugins', pluginName, 'config', 'routes'], []).filter( | 
					
						
							|  |  |  |       route => _.get(route, ['config', 'description']) !== undefined | 
					
						
							|  |  |  |     ); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Given a string and a pluginName retrieve the model and the pluginName | 
					
						
							|  |  |  |    * @param {String} string | 
					
						
							|  |  |  |    * @param {Sting} pluginName | 
					
						
							|  |  |  |    * @returns {Array} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   getModelAndNameForPlugin: (string, pluginName) => { | 
					
						
							|  |  |  |     return _.replace(string, `${pluginName}-`, `${pluginName}.`).split('.'); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Retrieve the path needed to get a model from a plugin | 
					
						
							|  |  |  |    * @param (String) string | 
					
						
							|  |  |  |    * @param {String} plugin | 
					
						
							|  |  |  |    * @returns {Array} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   getModelForPlugin: function(string, pluginName) { | 
					
						
							|  |  |  |     const [plugin, model] = this.getModelAndNameForPlugin(string, pluginName); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return ['plugins', plugin, 'models', _.lowerCase(model)]; | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Check whether or not a plugin needs documentation | 
					
						
							|  |  |  |    * @param {String} pluginName | 
					
						
							|  |  |  |    * @returns {Boolean} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   isPluginDocumentationNeeded: function(pluginName) { | 
					
						
							| 
									
										
										
										
											2020-03-20 14:03:51 +01:00
										 |  |  |     const { pluginsForWhichToGenerateDoc } = strapi.plugins.documentation.config['x-strapi-config']; | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |     if ( | 
					
						
							|  |  |  |       Array.isArray(pluginsForWhichToGenerateDoc) && | 
					
						
							|  |  |  |       !pluginsForWhichToGenerateDoc.includes(pluginName) | 
					
						
							|  |  |  |     ) { | 
					
						
							|  |  |  |       return false; | 
					
						
							| 
									
										
										
										
											2020-03-20 14:03:51 +01:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |       return this.getPluginRoutesWithDescription(pluginName).length > 0; | 
					
						
							| 
									
										
										
										
											2020-03-20 14:03:51 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Merge two components by replacing the default ones by the overides and keeping the others | 
					
						
							|  |  |  |    * @param {Object} initObj | 
					
						
							|  |  |  |    * @param {Object} srcObj | 
					
						
							|  |  |  |    * @returns {Object} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   mergeComponents: (initObj, srcObj) => { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |     const cleanedObj = Object.keys(_.get(initObj, 'schemas', {})).reduce((acc, current) => { | 
					
						
							|  |  |  |       const targetObj = _.has(_.get(srcObj, ['schemas'], {}), current) ? srcObj : initObj; | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |       _.set(acc, ['schemas', current], _.get(targetObj, ['schemas', current], {})); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return acc; | 
					
						
							|  |  |  |     }, {}); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return _.merge(cleanedObj, srcObj); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   mergePaths: function(initObj, srcObj) { | 
					
						
							|  |  |  |     return Object.keys(initObj.paths).reduce((acc, current) => { | 
					
						
							| 
									
										
										
										
											2019-08-23 14:13:19 +02:00
										 |  |  |       if (_.has(_.get(srcObj, ['paths'], {}), current)) { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |         const verbs = Object.keys(initObj.paths[current]).reduce((acc1, curr) => { | 
					
						
							|  |  |  |           const verb = this.mergeVerbObject( | 
					
						
							|  |  |  |             initObj.paths[current][curr], | 
					
						
							|  |  |  |             _.get(srcObj, ['paths', current, curr], {}) | 
					
						
							|  |  |  |           ); | 
					
						
							|  |  |  |           _.set(acc1, [curr], verb); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |           return acc1; | 
					
						
							|  |  |  |         }, {}); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |         _.set(acc, ['paths', current], verbs); | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         _.set(acc, ['paths', current], _.get(initObj, ['paths', current], {})); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return acc; | 
					
						
							|  |  |  |     }, {}); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   mergeTags: (initObj, srcObj) => { | 
					
						
							|  |  |  |     return _.get(srcObj, 'tags', _.get(initObj, 'tags', [])); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Merge two verb objects with a customizer | 
					
						
							|  |  |  |    * @param {Object} initObj | 
					
						
							|  |  |  |    * @param {Object} srcObj | 
					
						
							|  |  |  |    * @returns {Object} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   mergeVerbObject: function(initObj, srcObj) { | 
					
						
							|  |  |  |     return _.mergeWith(initObj, srcObj, (objValue, srcValue) => { | 
					
						
							|  |  |  |       if (_.isPlainObject(objValue)) { | 
					
						
							|  |  |  |         return Object.assign(objValue, srcValue); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return srcValue; | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   retrieveDocumentation: function(name, isPlugin = false) { | 
					
						
							|  |  |  |     const documentationPath = isPlugin | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |       ? [strapi.config.appPath, 'extensions', name, 'documentation', this.getDocumentationVersion()] | 
					
						
							|  |  |  |       : [strapi.config.appPath, 'api', name, 'documentation', this.getDocumentationVersion()]; | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     try { | 
					
						
							|  |  |  |       const documentationFiles = fs | 
					
						
							|  |  |  |         .readdirSync(path.resolve(documentationPath.join('/'))) | 
					
						
							|  |  |  |         .filter(el => el.includes('.json')); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return documentationFiles.reduce((acc, current) => { | 
					
						
							|  |  |  |         try { | 
					
						
							|  |  |  |           const doc = JSON.parse( | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |             fs.readFileSync(path.resolve([...documentationPath, current].join('/')), 'utf8') | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |           ); | 
					
						
							|  |  |  |           acc.push(doc); | 
					
						
							|  |  |  |         } catch (err) { | 
					
						
							|  |  |  |           // console.log(path.resolve([...documentationPath, current].join('/')), err);
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return acc; | 
					
						
							|  |  |  |       }, []); | 
					
						
							|  |  |  |     } catch (err) { | 
					
						
							|  |  |  |       return []; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Retrieve all documentation files from either the APIs or the plugins | 
					
						
							|  |  |  |    * @param {Boolean} isPlugin | 
					
						
							|  |  |  |    * @returns {Array} | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |   retrieveDocumentationFiles: function(isPlugin = false, version = this.getDocumentationVersion()) { | 
					
						
							|  |  |  |     const array = isPlugin ? this.getPluginsWithDocumentationNeeded() : this.getApis(); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return array.reduce((acc, current) => { | 
					
						
							|  |  |  |       const documentationPath = isPlugin | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |         ? [strapi.config.appPath, 'extensions', current, 'documentation', version] | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |         : [strapi.config.appPath, 'api', current, 'documentation', version]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       try { | 
					
						
							|  |  |  |         const documentationFiles = fs | 
					
						
							|  |  |  |           .readdirSync(path.resolve(documentationPath.join('/'))) | 
					
						
							|  |  |  |           .filter(el => el.includes('.json')); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         documentationFiles.forEach(el => { | 
					
						
							|  |  |  |           try { | 
					
						
							|  |  |  |             let documentation = JSON.parse( | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |               fs.readFileSync(path.resolve([...documentationPath, el].join('/')), 'utf8') | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |             ); | 
					
						
							| 
									
										
										
										
											2018-12-07 00:14:59 +01:00
										 |  |  |             /* eslint-disable indent */ | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |             const overrideDocumentationPath = isPlugin | 
					
						
							|  |  |  |               ? path.resolve( | 
					
						
							| 
									
										
										
										
											2018-12-07 00:14:59 +01:00
										 |  |  |                   strapi.config.appPath, | 
					
						
							| 
									
										
										
										
											2020-01-09 17:19:48 +01:00
										 |  |  |                   'extensions', | 
					
						
							| 
									
										
										
										
											2018-12-07 00:14:59 +01:00
										 |  |  |                   current, | 
					
						
							|  |  |  |                   'documentation', | 
					
						
							|  |  |  |                   version, | 
					
						
							|  |  |  |                   'overrides', | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |                   el | 
					
						
							| 
									
										
										
										
											2018-12-07 00:14:59 +01:00
										 |  |  |                 ) | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |               : path.resolve( | 
					
						
							| 
									
										
										
										
											2018-12-07 00:14:59 +01:00
										 |  |  |                   strapi.config.appPath, | 
					
						
							|  |  |  |                   'api', | 
					
						
							|  |  |  |                   current, | 
					
						
							|  |  |  |                   'documentation', | 
					
						
							|  |  |  |                   version, | 
					
						
							|  |  |  |                   'overrides', | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |                   el | 
					
						
							| 
									
										
										
										
											2018-12-07 00:14:59 +01:00
										 |  |  |                 ); | 
					
						
							|  |  |  |             /* eslint-enable indent */ | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |             let overrideDocumentation; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             try { | 
					
						
							|  |  |  |               overrideDocumentation = JSON.parse( | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |                 fs.readFileSync(overrideDocumentationPath, 'utf8') | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |               ); | 
					
						
							|  |  |  |             } catch (err) { | 
					
						
							|  |  |  |               overrideDocumentation = null; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!_.isEmpty(overrideDocumentation)) { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |               documentation.paths = this.mergePaths(documentation, overrideDocumentation).paths; | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |               documentation.tags = _.cloneDeep( | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |                 this.mergeTags(documentation, overrideDocumentation) | 
					
						
							|  |  |  |               ); | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |               const documentationComponents = _.get(documentation, 'components', {}); | 
					
						
							|  |  |  |               const overrideComponents = _.get(overrideDocumentation, 'components', {}); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |               const mergedComponents = this.mergeComponents( | 
					
						
							|  |  |  |                 documentationComponents, | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |                 overrideComponents | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |               ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |               if (!_.isEmpty(mergedComponents)) { | 
					
						
							|  |  |  |                 documentation.components = mergedComponents; | 
					
						
							|  |  |  |               } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             acc.push(documentation); | 
					
						
							|  |  |  |           } catch (err) { | 
					
						
							| 
									
										
										
										
											2019-04-15 18:57:58 +02:00
										 |  |  |             strapi.log.error(err); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |             console.log( | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |               `Unable to access the documentation for ${[...documentationPath, el].join('/')}` | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |             ); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |       } catch (err) { | 
					
						
							| 
									
										
										
										
											2019-04-15 18:57:58 +02:00
										 |  |  |         strapi.log.error(err); | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |         console.log( | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |           `Unable to retrieve documentation for the ${isPlugin ? 'plugin' : 'api'} ${current}` | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |         ); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return acc; | 
					
						
							|  |  |  |     }, []); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   retrieveDocumentationVersions: function() { | 
					
						
							|  |  |  |     return fs | 
					
						
							|  |  |  |       .readdirSync(this.getFullDocumentationPath()) | 
					
						
							|  |  |  |       .map(version => { | 
					
						
							|  |  |  |         try { | 
					
						
							|  |  |  |           const doc = JSON.parse( | 
					
						
							|  |  |  |             fs.readFileSync( | 
					
						
							| 
									
										
										
										
											2020-03-23 14:33:17 +01:00
										 |  |  |               path.resolve(this.getFullDocumentationPath(), version, 'full_documentation.json') | 
					
						
							| 
									
										
										
										
											2019-06-14 23:07:04 +02:00
										 |  |  |             ) | 
					
						
							| 
									
										
										
										
											2018-12-06 18:03:56 +01:00
										 |  |  |           ); | 
					
						
							|  |  |  |           const generatedDate = _.get(doc, ['info', 'x-generation-date'], null); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           return { version, generatedDate, url: '' }; | 
					
						
							|  |  |  |         } catch (err) { | 
					
						
							|  |  |  |           return null; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |       .filter(x => x); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   retrieveFrontForm: async function() { | 
					
						
							|  |  |  |     const config = await strapi | 
					
						
							|  |  |  |       .store({ | 
					
						
							|  |  |  |         environment: '', | 
					
						
							|  |  |  |         type: 'plugin', | 
					
						
							|  |  |  |         name: 'documentation', | 
					
						
							|  |  |  |         key: 'config', | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |       .get(); | 
					
						
							|  |  |  |     const forms = JSON.parse(JSON.stringify(form)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _.set(forms, [0, 0, 'value'], config.restrictedAccess); | 
					
						
							|  |  |  |     _.set(forms, [0, 1, 'value'], config.password || ''); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return forms; | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | }; |