2021-06-08 15:10:56 +00:00
import fs from 'fs' ;
import path from 'path' ;
2022-04-29 15:45:43 +00:00
import { bgCyan , black , bold , cyan , gray , green , red , yellow } from 'kleur/colors' ;
2021-06-08 15:10:56 +00:00
import prompts from 'prompts' ;
import degit from 'degit' ;
import yargs from 'yargs-parser' ;
2022-03-21 17:33:31 +00:00
import ora from 'ora' ;
2021-06-08 16:56:37 +00:00
import { TEMPLATES } from './templates.js' ;
2022-01-21 00:00:22 +00:00
import { logger , defaultLogLevel } from './logger.js' ;
2022-04-28 00:58:18 +00:00
import { execa , execaCommand } from 'execa' ;
2021-11-17 18:30:12 +00:00
// NOTE: In the v7.x version of npm, the default behavior of `npm init` was changed
2021-11-17 18:32:36 +00:00
// to no longer require `--` to pass args and instead pass `--` directly to us. This
// broke our arg parser, since `--` is a special kind of flag. Filtering for `--` here
2021-11-17 18:30:12 +00:00
// fixes the issue so that create-astro now works on all npm version.
2021-11-17 18:32:36 +00:00
const cleanArgv = process . argv . filter ( ( arg ) = > arg !== '--' ) ;
2022-03-18 22:35:45 +00:00
const args = yargs ( cleanArgv ) ;
2021-06-08 15:10:56 +00:00
prompts . override ( args ) ;
export function mkdirp ( dir : string ) {
2021-12-22 21:11:05 +00:00
try {
fs . mkdirSync ( dir , { recursive : true } ) ;
} catch ( e : any ) {
if ( e . code === 'EEXIST' ) return ;
throw e ;
}
2021-06-08 15:10:56 +00:00
}
2022-04-21 20:36:48 +00:00
function isEmpty ( dirPath : string ) {
return ! fs . existsSync ( dirPath ) || fs . readdirSync ( dirPath ) . length === 0 ;
}
2022-04-02 20:15:41 +00:00
const { version } = JSON . parse (
fs . readFileSync ( new URL ( '../package.json' , import . meta . url ) , 'utf-8' )
) ;
2021-06-08 15:10:56 +00:00
2022-04-28 00:58:18 +00:00
const FILES_TO_REMOVE = [ '.stackblitzrc' , 'sandbox.config.json' , 'CHANGELOG.md' ] ; // some files are only needed for online editors when using astro.new. Remove for create-astro installs.
2021-06-10 16:30:48 +00:00
2021-06-08 15:10:56 +00:00
export async function main() {
2022-04-26 15:24:24 +00:00
const pkgManager = pkgManagerFromUserAgent ( process . env . npm_config_user_agent ) ;
2022-01-21 00:00:22 +00:00
logger . debug ( 'Verbose logging turned on' ) ;
2021-12-22 21:11:05 +00:00
console . log ( ` \ n ${ bold ( 'Welcome to Astro!' ) } ${ gray ( ` (create-astro v ${ version } ) ` ) } ` ) ;
2022-03-21 17:33:31 +00:00
let spinner = ora ( { color : 'green' , text : 'Prepare for liftoff.' } ) ;
spinner . succeed ( ) ;
2021-12-22 21:11:05 +00:00
2022-04-21 20:36:48 +00:00
let cwd = args [ '_' ] [ 2 ] as string ;
if ( cwd && isEmpty ( cwd ) ) {
let acknowledgeProjectDir = ora ( {
color : 'green' ,
text : ` Using ${ bold ( cwd ) } as project directory. ` ,
} ) ;
acknowledgeProjectDir . succeed ( ) ;
}
if ( ! cwd || ! isEmpty ( cwd ) ) {
2022-04-27 00:38:31 +00:00
const notEmptyMsg = ( dirPath : string ) = > ` " ${ bold ( dirPath ) } " is not empty! ` ;
2022-04-21 20:36:48 +00:00
if ( ! isEmpty ( cwd ) ) {
let rejectProjectDir = ora ( { color : 'red' , text : notEmptyMsg ( cwd ) } ) ;
rejectProjectDir . fail ( ) ;
2021-12-22 21:11:05 +00:00
}
2022-04-21 20:36:48 +00:00
const dirResponse = await prompts ( {
type : 'text' ,
name : 'directory' ,
message : 'Where would you like to create your app?' ,
initial : './my-astro-site' ,
validate ( value ) {
if ( ! isEmpty ( value ) ) {
return notEmptyMsg ( value ) ;
}
return true ;
} ,
} ) ;
cwd = dirResponse . directory ;
}
if ( ! cwd ) {
process . exit ( 1 ) ;
2021-12-22 21:11:05 +00:00
}
2022-03-21 17:33:31 +00:00
const options = await prompts ( [
2021-12-22 21:11:05 +00:00
{
type : 'select' ,
name : 'template' ,
message : 'Which app template would you like to use?' ,
choices : TEMPLATES ,
} ,
] ) ;
if ( ! options . template ) {
process . exit ( 1 ) ;
}
const hash = args . commit ? ` # ${ args . commit } ` : '' ;
2022-04-28 00:58:18 +00:00
const templateTarget = ` withastro/astro/examples/ ${ options . template } #latest ` ;
2021-12-22 21:11:05 +00:00
const emitter = degit ( ` ${ templateTarget } ${ hash } ` , {
cache : false ,
force : true ,
2022-01-21 00:00:22 +00:00
verbose : defaultLogLevel === 'debug' ? true : false ,
} ) ;
logger . debug ( 'Initialized degit with following config:' , ` ${ templateTarget } ${ hash } ` , {
cache : false ,
force : true ,
verbose : defaultLogLevel === 'debug' ? true : false ,
2021-12-22 21:11:05 +00:00
} ) ;
2022-03-21 17:33:31 +00:00
spinner = ora ( { color : 'green' , text : 'Copying project files...' } ) . start ( ) ;
2021-12-22 21:11:05 +00:00
// Copy
2022-04-26 15:24:24 +00:00
if ( ! args . dryrun ) {
try {
emitter . on ( 'info' , ( info ) = > {
logger . debug ( info . message ) ;
} ) ;
await emitter . clone ( cwd ) ;
} catch ( err : any ) {
// degit is compiled, so the stacktrace is pretty noisy. Only report the stacktrace when using verbose mode.
logger . debug ( err ) ;
console . error ( red ( err . message ) ) ;
// Warning for issue #655
if ( err . message === 'zlib: unexpected end of file' ) {
console . log (
yellow (
"This seems to be a cache related problem. Remove the folder '~/.degit/github/withastro' to fix this error."
)
) ;
console . log (
yellow (
'For more information check out this issue: https://github.com/withastro/astro/issues/655'
)
) ;
}
2021-12-22 21:11:05 +00:00
2022-04-26 15:24:24 +00:00
// Helpful message when encountering the "could not find commit hash for ..." error
if ( err . code === 'MISSING_REF' ) {
console . log (
yellow (
"This seems to be an issue with degit. Please check if you have 'git' installed on your system, and install it if you don't have (https://git-scm.com)."
)
) ;
console . log (
yellow (
"If you do have 'git' installed, please run this command with the --verbose flag and file a new issue with the command output here: https://github.com/withastro/astro/issues"
)
) ;
}
spinner . fail ( ) ;
process . exit ( 1 ) ;
2021-12-22 21:11:05 +00:00
}
2022-04-26 15:24:24 +00:00
// Post-process in parallel
2022-04-28 00:58:18 +00:00
await Promise . all (
FILES_TO_REMOVE . map ( async ( file ) = > {
2022-04-26 15:24:24 +00:00
const fileLoc = path . resolve ( path . join ( cwd , file ) ) ;
2022-04-28 00:58:18 +00:00
if ( fs . existsSync ( fileLoc ) ) {
return fs . promises . rm ( fileLoc , { } ) ;
2021-12-22 21:11:05 +00:00
}
2022-04-28 00:58:18 +00:00
} )
) ;
2021-12-22 21:11:05 +00:00
}
2022-03-21 17:33:31 +00:00
spinner . succeed ( ) ;
2021-12-22 21:11:05 +00:00
console . log ( bold ( green ( '✔' ) + ' Done!' ) ) ;
2022-04-26 15:24:24 +00:00
const installResponse = await prompts ( {
type : 'confirm' ,
name : 'install' ,
message : ` Would you like us to run " ${ pkgManager } install?" ` ,
initial : true ,
} ) ;
if ( ! installResponse ) {
process . exit ( 0 ) ;
}
2022-04-29 15:45:43 +00:00
if ( installResponse . install && ! args . dryrun ) {
2022-04-26 15:24:24 +00:00
const installExec = execa ( pkgManager , [ 'install' ] , { cwd } ) ;
const installingPackagesMsg = ` Installing packages ${ emojiWithFallback ( ' 📦' , '...' ) } ` ;
spinner = ora ( { color : 'green' , text : installingPackagesMsg } ) . start ( ) ;
2022-04-29 15:45:43 +00:00
await new Promise < void > ( ( resolve , reject ) = > {
installExec . stdout ? . on ( 'data' , function ( data ) {
spinner . text = ` ${ installingPackagesMsg } \ n ${ bold ( ` [ ${ pkgManager } ] ` ) } ${ data } ` ;
2022-04-26 15:24:24 +00:00
} ) ;
2022-04-29 15:45:43 +00:00
installExec . on ( 'error' , ( error ) = > reject ( error ) ) ;
installExec . on ( 'close' , ( ) = > resolve ( ) ) ;
} ) ;
2022-04-26 15:24:24 +00:00
spinner . succeed ( ) ;
}
2022-04-28 00:58:18 +00:00
const astroAddCommand = installResponse . install
? 'astro add --yes'
: ` ${ pkgManagerExecCommand ( pkgManager ) } astro@latest add --yes ` ;
const astroAddResponse = await prompts ( {
type : 'confirm' ,
name : 'astroAdd' ,
message : ` Run " ${ astroAddCommand } ?" This lets you optionally add component frameworks (ex. React), CSS frameworks (ex. Tailwind), and more. ` ,
initial : true ,
} ) ;
if ( ! astroAddResponse ) {
process . exit ( 0 ) ;
}
if ( ! astroAddResponse . astroAdd ) {
ora ( ) . info (
` No problem. You can always run " ${ pkgManagerExecCommand ( pkgManager ) } astro add" later! `
) ;
}
if ( astroAddResponse . astroAdd && ! args . dryrun ) {
await execaCommand (
astroAddCommand ,
astroAddCommand === 'astro add --yes'
? { cwd , stdio : 'inherit' , localDir : cwd , preferLocal : true }
: { cwd , stdio : 'inherit' }
) ;
}
2022-04-29 15:45:43 +00:00
const gitResponse = await prompts ( {
type : 'confirm' ,
name : 'git' ,
message : 'Initialize a git repository?' ,
initial : true ,
} ) ;
if ( ! gitResponse ) {
process . exit ( 0 ) ;
}
if ( gitResponse . git && ! args . dryrun ) {
await execaCommand ( 'git init' , { cwd } ) ;
}
console . log ( ` \ n ${ bgCyan ( black ( ' Next steps ' ) ) } \ n ` ) ;
2021-12-22 21:11:05 +00:00
const relative = path . relative ( process . cwd ( ) , cwd ) ;
2022-04-29 15:45:43 +00:00
const startCommand = [ ] ;
2021-12-22 21:11:05 +00:00
if ( relative !== '' ) {
2022-04-29 15:45:43 +00:00
startCommand . push ( bold ( cyan ( ` cd ${ relative } ` ) ) ) ;
2021-12-22 21:11:05 +00:00
}
2022-04-26 15:24:24 +00:00
if ( ! installResponse . install ) {
2022-04-29 15:45:43 +00:00
startCommand . push ( bold ( cyan ( ` ${ pkgManager } install ` ) ) ) ;
2022-04-26 15:24:24 +00:00
}
2022-04-29 15:45:43 +00:00
startCommand . push ( bold ( cyan ( pkgManager === 'npm' ? 'npm run dev' : ` ${ pkgManager } dev ` ) ) ) ;
console . log ( startCommand . join ( ' && ' ) ) ;
2021-12-22 21:11:05 +00:00
console . log ( ` \ nTo close the dev server, hit ${ bold ( cyan ( 'Ctrl-C' ) ) } ` ) ;
2022-04-29 15:45:43 +00:00
console . log ( ` Stuck? Visit us at ${ cyan ( 'https://astro.build/chat' ) } \ n ` ) ;
2021-06-08 15:12:07 +00:00
}
2022-04-26 15:24:24 +00:00
function emojiWithFallback ( char : string , fallback : string ) {
return process . platform !== 'win32' ? char : fallback ;
}
function pkgManagerFromUserAgent ( userAgent? : string ) {
if ( ! userAgent ) return 'npm' ;
const pkgSpec = userAgent . split ( ' ' ) [ 0 ] ;
const pkgSpecArr = pkgSpec . split ( '/' ) ;
return pkgSpecArr [ 0 ] ;
}
2022-04-28 00:58:18 +00:00
function pkgManagerExecCommand ( pkgManager : string ) {
if ( pkgManager === 'pnpm' ) {
return 'pnpx' ;
} else {
// note: yarn does not have an "npx" equivalent
return 'npx' ;
}
}