| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  | 'use strict'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const os = require('os'); | 
					
						
							|  |  |  | const path = require('path'); | 
					
						
							|  |  |  | const fse = require('fs-extra'); | 
					
						
							| 
									
										
										
										
											2021-10-27 16:03:35 +02:00
										 |  |  | const _ = require('lodash/fp'); | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  | const chalk = require('chalk'); | 
					
						
							| 
									
										
										
										
											2021-09-27 19:16:08 +02:00
										 |  |  | const { getTemplatePackageInfo, downloadNpmTemplate } = require('./fetch-npm-template'); | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Specify all the files and directories a template can have
 | 
					
						
							| 
									
										
										
										
											2021-09-24 15:09:28 +02:00
										 |  |  | const allowFile = Symbol(); | 
					
						
							|  |  |  | const allowChildren = Symbol(); | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  | const allowedTemplateContents = { | 
					
						
							| 
									
										
										
										
											2021-09-24 15:09:28 +02:00
										 |  |  |   'README.md': allowFile, | 
					
						
							|  |  |  |   '.env.example': allowFile, | 
					
						
							|  |  |  |   'package.json': allowFile, | 
					
						
							| 
									
										
										
										
											2021-10-25 12:30:53 +02:00
										 |  |  |   src: allowChildren, | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  |   data: allowChildren, | 
					
						
							| 
									
										
										
										
											2021-09-24 15:09:28 +02:00
										 |  |  |   database: allowChildren, | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  |   public: allowChildren, | 
					
						
							|  |  |  |   scripts: allowChildren, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							| 
									
										
										
										
											2021-09-27 19:16:08 +02:00
										 |  |  |  * Merge template with new project being created | 
					
						
							| 
									
										
										
										
											2020-12-04 11:22:20 +01:00
										 |  |  |  * @param {string} scope  project creation params | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  |  * @param {string} rootPath  project path | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-12-04 11:22:20 +01:00
										 |  |  | module.exports = async function mergeTemplate(scope, rootPath) { | 
					
						
							| 
									
										
										
										
											2021-09-30 11:42:24 +02:00
										 |  |  |   let templatePath; | 
					
						
							|  |  |  |   let templateParentPath; | 
					
						
							|  |  |  |   let templatePackageInfo = {}; | 
					
						
							| 
									
										
										
										
											2021-10-28 18:50:41 +02:00
										 |  |  |   const isLocalTemplate = ['./', '../', '/'].some(filePrefix => | 
					
						
							|  |  |  |     scope.template.startsWith(filePrefix) | 
					
						
							|  |  |  |   ); | 
					
						
							| 
									
										
										
										
											2021-09-30 11:42:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (isLocalTemplate) { | 
					
						
							|  |  |  |     // Template is a local directory
 | 
					
						
							|  |  |  |     console.log('Installing local template.'); | 
					
						
							| 
									
										
										
										
											2021-10-26 20:01:26 +02:00
										 |  |  |     templatePath = path.resolve(rootPath, '..', scope.template); | 
					
						
							| 
									
										
										
										
											2021-09-30 11:42:24 +02:00
										 |  |  |   } else { | 
					
						
							|  |  |  |     // Template should be an npm package. Fetch template info
 | 
					
						
							|  |  |  |     templatePackageInfo = await getTemplatePackageInfo(scope.template); | 
					
						
							|  |  |  |     console.log(`Installing ${chalk.yellow(templatePackageInfo.name)} template.`); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Download template repository to a temporary directory
 | 
					
						
							|  |  |  |     templateParentPath = await fse.mkdtemp(path.join(os.tmpdir(), 'strapi-')); | 
					
						
							| 
									
										
										
										
											2021-10-27 16:37:30 +02:00
										 |  |  |     templatePath = await downloadNpmTemplate(templatePackageInfo, templateParentPath); | 
					
						
							| 
									
										
										
										
											2021-09-30 11:42:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Make sure the downloaded template matches the required format
 | 
					
						
							| 
									
										
										
										
											2021-10-26 20:01:26 +02:00
										 |  |  |   const templateConfig = await checkTemplateRootStructure(templatePath, scope); | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  |   await checkTemplateContentsStructure(path.resolve(templatePath, 'template')); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Merge contents of the template in the project
 | 
					
						
							| 
									
										
										
										
											2021-09-27 19:16:08 +02:00
										 |  |  |   await mergePackageJSON({ rootPath, templateConfig, templatePackageInfo }); | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  |   await mergeFilesAndDirectories(rootPath, templatePath); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-30 11:42:24 +02:00
										 |  |  |   // Delete the template directory if it was downloaded
 | 
					
						
							|  |  |  |   if (!isLocalTemplate) { | 
					
						
							|  |  |  |     await fse.remove(templateParentPath); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-27 19:16:08 +02:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Make sure the template has the required top-level structure | 
					
						
							|  |  |  |  * @param {string} templatePath - Path of the locally downloaded template | 
					
						
							| 
									
										
										
										
											2021-10-26 20:01:26 +02:00
										 |  |  |  * @returns {Object} - The template config object | 
					
						
							| 
									
										
										
										
											2021-09-27 19:16:08 +02:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2021-10-26 20:01:26 +02:00
										 |  |  | async function checkTemplateRootStructure(templatePath) { | 
					
						
							|  |  |  |   // Make sure the root of the repo has a template.json file
 | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  |   const templateJsonPath = path.join(templatePath, 'template.json'); | 
					
						
							| 
									
										
										
										
											2021-10-27 16:03:35 +02:00
										 |  |  |   const templateJsonExists = await fse.exists(templateJsonPath); | 
					
						
							|  |  |  |   if (!templateJsonExists) { | 
					
						
							| 
									
										
										
										
											2021-10-26 20:01:26 +02:00
										 |  |  |     throw new Error(`A template must have a ${chalk.green('template.json')} root file`); | 
					
						
							| 
									
										
										
										
											2020-12-04 11:22:20 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-10-27 16:03:35 +02:00
										 |  |  |   const templateJsonStat = await fse.stat(templateJsonPath); | 
					
						
							|  |  |  |   if (!templateJsonStat.isFile()) { | 
					
						
							| 
									
										
										
										
											2021-10-26 20:01:26 +02:00
										 |  |  |     throw new Error(`A template's ${chalk.green('template.json')} must be a file`); | 
					
						
							| 
									
										
										
										
											2020-12-04 11:22:20 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 20:01:26 +02:00
										 |  |  |   const templateConfig = require(templateJsonPath); | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-04 11:22:20 +01:00
										 |  |  |   // Make sure the root of the repo has a template folder
 | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  |   const templateDirPath = path.join(templatePath, 'template'); | 
					
						
							|  |  |  |   try { | 
					
						
							|  |  |  |     const stat = await fse.stat(templateDirPath); | 
					
						
							|  |  |  |     if (!stat.isDirectory()) { | 
					
						
							|  |  |  |       throw Error(`A template must have a root ${chalk.green('template/')} directory`); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } catch (error) { | 
					
						
							|  |  |  |     if (error.code === 'ENOENT') { | 
					
						
							|  |  |  |       throw Error(`A template must have a root ${chalk.green('template/')} directory`); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     throw error; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-12-04 11:22:20 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 20:01:26 +02:00
										 |  |  |   return templateConfig; | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-27 19:16:08 +02:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Traverse template tree to make sure each file and folder is allowed | 
					
						
							|  |  |  |  * @param {string} templateContentsPath | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  | async function checkTemplateContentsStructure(templateContentsPath) { | 
					
						
							|  |  |  |   // Recursively check if each item in a directory is allowed
 | 
					
						
							| 
									
										
										
										
											2021-10-27 16:03:35 +02:00
										 |  |  |   const checkPathContents = async (pathToCheck, parents) => { | 
					
						
							|  |  |  |     const contents = await fse.readdir(pathToCheck); | 
					
						
							|  |  |  |     for (const item of contents) { | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  |       const nextParents = [...parents, item]; | 
					
						
							| 
									
										
										
										
											2021-10-27 16:03:35 +02:00
										 |  |  |       const matchingTreeValue = _.get(nextParents, allowedTemplateContents); | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |       // Treat files and directories separately
 | 
					
						
							|  |  |  |       const itemPath = path.resolve(pathToCheck, item); | 
					
						
							| 
									
										
										
										
											2021-10-27 16:03:35 +02:00
										 |  |  |       const isDirectory = (await fse.stat(itemPath)).isDirectory(); | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |       if (matchingTreeValue === undefined) { | 
					
						
							|  |  |  |         // Unknown paths are forbidden
 | 
					
						
							|  |  |  |         throw Error( | 
					
						
							|  |  |  |           `Illegal template structure, unknown path ${chalk.green(nextParents.join('/'))}` | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-24 15:09:28 +02:00
										 |  |  |       if (matchingTreeValue === allowFile) { | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  |         if (!isDirectory) { | 
					
						
							|  |  |  |           // All good, the file is allowed
 | 
					
						
							|  |  |  |           return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         throw Error( | 
					
						
							|  |  |  |           `Illegal template structure, expected a file and got a directory at ${chalk.green( | 
					
						
							|  |  |  |             nextParents.join('/') | 
					
						
							|  |  |  |           )}`
 | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (isDirectory) { | 
					
						
							|  |  |  |         if (matchingTreeValue === allowChildren) { | 
					
						
							|  |  |  |           // All children are allowed
 | 
					
						
							|  |  |  |           return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // Check if the contents of the directory are allowed
 | 
					
						
							| 
									
										
										
										
											2021-10-27 16:03:35 +02:00
										 |  |  |         await checkPathContents(itemPath, nextParents); | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  |       } else { | 
					
						
							|  |  |  |         throw Error( | 
					
						
							|  |  |  |           `Illegal template structure, unknow file ${chalk.green(nextParents.join('/'))}` | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2021-10-27 16:03:35 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-27 16:03:35 +02:00
										 |  |  |   await checkPathContents(templateContentsPath, []); | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-27 19:16:08 +02:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Merge the template's template.json into the Strapi project's package.json | 
					
						
							|  |  |  |  * @param {Object} config | 
					
						
							|  |  |  |  * @param {string} config.rootPath | 
					
						
							|  |  |  |  * @param {string} config.templateConfig | 
					
						
							|  |  |  |  * @param {Object} config.templatePackageInfo - Info about the template's package on npm | 
					
						
							|  |  |  |  * @param {Object} config.templatePackageInfo.name - The name of the template's package on npm | 
					
						
							|  |  |  |  * @param {Object} config.templatePackageInfo.version - The name of the template's package on npm | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2021-09-30 11:42:24 +02:00
										 |  |  | async function mergePackageJSON({ rootPath, templateConfig, templatePackageInfo }) { | 
					
						
							| 
									
										
										
										
											2020-12-04 11:22:20 +01:00
										 |  |  |   // Import the package.json as an object
 | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  |   const packageJSON = require(path.resolve(rootPath, 'package.json')); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-04 11:22:20 +01:00
										 |  |  |   if (!templateConfig.package) { | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  |     // Nothing to overwrite
 | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Make sure the template.json doesn't overwrite the UUID
 | 
					
						
							| 
									
										
										
										
											2020-12-04 11:22:20 +01:00
										 |  |  |   if (templateConfig.package.strapi && templateConfig.package.strapi.uuid) { | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  |     throw Error('A template cannot overwrite the Strapi UUID'); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Use lodash to deeply merge them
 | 
					
						
							| 
									
										
										
										
											2021-10-27 16:03:35 +02:00
										 |  |  |   const mergedConfig = _.merge(templateConfig.package, packageJSON); | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-27 13:23:35 +02:00
										 |  |  |   // Add template info to package.json
 | 
					
						
							| 
									
										
										
										
											2021-09-30 11:42:24 +02:00
										 |  |  |   if (templatePackageInfo.name) { | 
					
						
							| 
									
										
										
										
											2021-10-27 16:03:35 +02:00
										 |  |  |     _.set('strapi.template', templatePackageInfo.name, mergedConfig); | 
					
						
							| 
									
										
										
										
											2021-09-30 11:42:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-10-02 15:20:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Save the merged config as the new package.json
 | 
					
						
							|  |  |  |   const packageJSONPath = path.join(rootPath, 'package.json'); | 
					
						
							|  |  |  |   await fse.writeJSON(packageJSONPath, mergedConfig, { spaces: 2 }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Merge all allowed files and directories
 | 
					
						
							|  |  |  | async function mergeFilesAndDirectories(rootPath, templatePath) { | 
					
						
							|  |  |  |   const templateDir = path.join(templatePath, 'template'); | 
					
						
							|  |  |  |   await fse.copy(templateDir, rootPath, { overwrite: true, recursive: true }); | 
					
						
							|  |  |  | } |