| 
									
										
										
										
											2021-08-04 19:39:40 +02:00
										 |  |  | /* eslint-disable no-unreachable */ | 
					
						
							| 
									
										
										
										
											2019-06-20 16:38:15 +02:00
										 |  |  | 'use strict'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const { join } = require('path'); | 
					
						
							|  |  |  | const fse = require('fs-extra'); | 
					
						
							|  |  |  | const inquirer = require('inquirer'); | 
					
						
							|  |  |  | const execa = require('execa'); | 
					
						
							| 
									
										
										
										
											2021-06-29 16:27:35 +02:00
										 |  |  | const { merge } = require('lodash'); | 
					
						
							| 
									
										
										
										
											2019-06-20 16:38:15 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | const stopProcess = require('./utils/stop-process'); | 
					
						
							|  |  |  | const { trackUsage } = require('./utils/usage'); | 
					
						
							|  |  |  | const defaultConfigs = require('./utils/db-configs'); | 
					
						
							|  |  |  | const clientDependencies = require('./utils/db-client-dependencies'); | 
					
						
							|  |  |  | const dbQuestions = require('./utils/db-questions'); | 
					
						
							|  |  |  | const createProject = require('./create-project'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module.exports = async scope => { | 
					
						
							|  |  |  |   await trackUsage({ event: 'didChooseCustomDatabase', scope }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const configuration = await askDbInfosAndTest(scope).catch(error => { | 
					
						
							| 
									
										
										
										
											2021-04-29 11:11:46 +02:00
										 |  |  |     return trackUsage({ event: 'didNotConnectDatabase', scope, error }).then(() => { | 
					
						
							|  |  |  |       throw error; | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2019-06-20 16:38:15 +02:00
										 |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-08 16:13:36 +02:00
										 |  |  |   console.log(); | 
					
						
							| 
									
										
										
										
											2019-07-08 15:39:08 +02:00
										 |  |  |   console.log('Creating a project with custom database options.'); | 
					
						
							| 
									
										
										
										
											2019-06-20 16:38:15 +02:00
										 |  |  |   await trackUsage({ event: 'didConnectDatabase', scope }); | 
					
						
							|  |  |  |   return createProject(scope, configuration); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const MAX_RETRIES = 5; | 
					
						
							|  |  |  | async function askDbInfosAndTest(scope) { | 
					
						
							|  |  |  |   let retries = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   async function loop() { | 
					
						
							|  |  |  |     // else ask for the client name
 | 
					
						
							|  |  |  |     const { client, connection } = await askDatabaseInfos(scope); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const configuration = { | 
					
						
							|  |  |  |       client, | 
					
						
							|  |  |  |       connection, | 
					
						
							| 
									
										
										
										
											2019-07-03 11:20:14 +02:00
										 |  |  |       dependencies: clientDependencies({ scope, client }), | 
					
						
							| 
									
										
										
										
											2019-06-20 16:38:15 +02:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-14 17:39:52 +01:00
										 |  |  |     return testDatabaseConnection({ | 
					
						
							| 
									
										
										
										
											2019-06-20 16:38:15 +02:00
										 |  |  |       scope, | 
					
						
							|  |  |  |       configuration, | 
					
						
							|  |  |  |     }) | 
					
						
							| 
									
										
										
										
											2019-07-08 15:39:08 +02:00
										 |  |  |       .then(result => { | 
					
						
							| 
									
										
										
										
											2021-04-29 11:11:46 +02:00
										 |  |  |         if (result && result.shouldRetry === true && retries < MAX_RETRIES - 1) { | 
					
						
							| 
									
										
										
										
											2019-07-08 15:39:08 +02:00
										 |  |  |           console.log('Retrying...'); | 
					
						
							|  |  |  |           retries++; | 
					
						
							|  |  |  |           return loop(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       }) | 
					
						
							| 
									
										
										
										
											2019-06-20 16:38:15 +02:00
										 |  |  |       .then( | 
					
						
							|  |  |  |         () => fse.remove(scope.tmpPath), | 
					
						
							|  |  |  |         err => { | 
					
						
							|  |  |  |           return fse.remove(scope.tmpPath).then(() => { | 
					
						
							|  |  |  |             throw err; | 
					
						
							|  |  |  |           }); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       ) | 
					
						
							| 
									
										
										
										
											2019-11-14 17:39:52 +01:00
										 |  |  |       .then(() => configuration) | 
					
						
							| 
									
										
										
										
											2019-06-20 16:38:15 +02:00
										 |  |  |       .catch(err => { | 
					
						
							|  |  |  |         if (retries < MAX_RETRIES - 1) { | 
					
						
							| 
									
										
										
										
											2019-07-08 16:13:36 +02:00
										 |  |  |           console.log(); | 
					
						
							| 
									
										
										
										
											2019-06-20 16:38:15 +02:00
										 |  |  |           console.log(`⛔️ Connection test failed: ${err.message}`); | 
					
						
							| 
									
										
										
										
											2019-07-08 16:13:36 +02:00
										 |  |  |           console.log(); | 
					
						
							| 
									
										
										
										
											2019-06-20 16:38:15 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |           if (scope.debug) { | 
					
						
							|  |  |  |             console.log('Full error log:'); | 
					
						
							|  |  |  |             console.log(err); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           console.log('Retrying...'); | 
					
						
							|  |  |  |           retries++; | 
					
						
							|  |  |  |           return loop(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         console.log(err); | 
					
						
							|  |  |  |         stopProcess( | 
					
						
							|  |  |  |           `️⛔️ Could not connect to your database after ${MAX_RETRIES} tries. Try to check your database configuration an retry.` | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return loop(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function testDatabaseConnection({ scope, configuration }) { | 
					
						
							|  |  |  |   const { client } = configuration; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (client === 'sqlite') return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-29 16:27:35 +02:00
										 |  |  |   return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // TODO: test DB connection somehow
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-20 16:38:15 +02:00
										 |  |  |   await installDatabaseTestingDep({ | 
					
						
							|  |  |  |     scope, | 
					
						
							|  |  |  |     configuration, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const connectivityFile = join( | 
					
						
							|  |  |  |     scope.tmpPath, | 
					
						
							|  |  |  |     'node_modules', | 
					
						
							| 
									
										
										
										
											2021-04-29 13:51:12 +02:00
										 |  |  |     `@strapi/connector-${configuration.connection.connector}`, | 
					
						
							| 
									
										
										
										
											2019-06-20 16:38:15 +02:00
										 |  |  |     'lib', | 
					
						
							|  |  |  |     'utils', | 
					
						
							|  |  |  |     'connectivity.js' | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const tester = require(connectivityFile); | 
					
						
							|  |  |  |   return tester({ scope, connection: configuration.connection }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function askDatabaseInfos(scope) { | 
					
						
							|  |  |  |   const { client } = await inquirer.prompt([ | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       type: 'list', | 
					
						
							|  |  |  |       name: 'client', | 
					
						
							|  |  |  |       message: 'Choose your default database client', | 
					
						
							| 
									
										
										
										
											2021-06-29 16:27:35 +02:00
										 |  |  |       choices: ['sqlite', 'postgres', 'mysql'], | 
					
						
							| 
									
										
										
										
											2019-06-20 16:38:15 +02:00
										 |  |  |       default: 'sqlite', | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |   ]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-29 11:11:46 +02:00
										 |  |  |   const responses = await inquirer.prompt(dbQuestions[client].map(q => q({ scope, client }))); | 
					
						
							| 
									
										
										
										
											2019-06-20 16:38:15 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const connection = merge({}, defaultConfigs[client] || {}, { | 
					
						
							| 
									
										
										
										
											2021-06-29 16:27:35 +02:00
										 |  |  |     client, | 
					
						
							|  |  |  |     connection: responses, | 
					
						
							| 
									
										
										
										
											2019-06-20 16:38:15 +02:00
										 |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return { | 
					
						
							|  |  |  |     client, | 
					
						
							|  |  |  |     connection, | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function installDatabaseTestingDep({ scope, configuration }) { | 
					
						
							| 
									
										
										
										
											2019-12-23 09:58:01 +01:00
										 |  |  |   let packageManager = scope.useYarn ? 'yarnpkg' : 'npm'; | 
					
						
							| 
									
										
										
										
											2019-12-18 01:15:45 +01:00
										 |  |  |   let cmd = scope.useYarn | 
					
						
							|  |  |  |     ? ['--cwd', scope.tmpPath, 'add'] | 
					
						
							|  |  |  |     : ['install', '--prefix', scope.tmpPath]; | 
					
						
							| 
									
										
										
										
											2019-06-20 16:38:15 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Manually create the temp directory for yarn
 | 
					
						
							| 
									
										
										
										
											2019-07-02 17:38:06 +02:00
										 |  |  |   if (scope.useYarn) { | 
					
						
							| 
									
										
										
										
											2019-06-20 16:38:15 +02:00
										 |  |  |     await fse.ensureDir(scope.tmpPath); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-18 01:15:45 +01:00
										 |  |  |   const deps = Object.keys(configuration.dependencies).map(dep => { | 
					
						
							| 
									
										
										
										
											2019-06-20 16:38:15 +02:00
										 |  |  |     return `${dep}@${configuration.dependencies[dep]}`; | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-18 01:15:45 +01:00
										 |  |  |   await execa(packageManager, cmd.concat(deps)); | 
					
						
							| 
									
										
										
										
											2019-06-20 16:38:15 +02:00
										 |  |  | } |