#!/usr/bin/env node // Native const { resolve } = require('path') // Packages const Progress = require('progress') const fs = require('fs-extra') const bytes = require('bytes') const chalk = require('chalk') const minimist = require('minimist') const ms = require('ms') const flatten = require('arr-flatten') const dotenv = require('dotenv') const { eraseLines } = require('ansi-escapes') const { write: copy } = require('clipboardy') const inquirer = require('inquirer') // Ours const login = require('../lib/login') const cfg = require('../lib/cfg') const { version } = require('../lib/pkg') const Logger = require('../lib/build-logger') const Now = require('../lib') const toHumanPath = require('../lib/utils/to-human-path') const { handleError, error } = require('../lib/error') const { fromGit, isRepoPath, gitPathParts } = require('../lib/git') const readMetaData = require('../lib/read-metadata') const checkPath = require('../lib/utils/check-path') const { reAlias, assignAlias } = require('../lib/re-alias') const exit = require('../lib/utils/exit') const logo = require('../lib/utils/output/logo') const cmd = require('../lib/utils/output/cmd') const info = require('../lib/utils/output/info') const wait = require('../lib/utils/output/wait') const NowPlans = require('../lib/plans') const promptBool = require('../lib/utils/input/prompt-bool') const promptOptions = require('../lib/utils/input/prompt-options') const note = require('../lib/utils/output/note') const argv = minimist(process.argv.slice(2), { string: ['config', 'token', 'name', 'alias', 'session-affinity'], boolean: [ 'help', 'version', 'debug', 'force', 'links', 'login', 'no-clipboard', 'forward-npm', 'docker', 'npm', 'static' ], alias: { env: 'e', dotenv: 'E', help: 'h', config: 'c', debug: 'd', version: 'v', force: 'f', token: 't', forceSync: 'F', links: 'l', login: 'L', public: 'p', 'no-clipboard': 'C', 'forward-npm': 'N', 'session-affinity': 'S', name: 'n', alias: 'a' } }) const help = () => { console.log(` ${chalk.bold(`${logo} now`)} [options] ${chalk.dim('Commands:')} ${chalk.dim('Cloud')} deploy [path] Performs a deployment ${chalk.bold( '(default)' )} ls | list [app] List deployments rm | remove [id] Remove a deployment ln | alias [id] [url] Configures aliases for deployments domains [name] Manages your domain names certs [cmd] Manages your SSL certificates secrets [name] Manages your secret environment variables dns [name] Manages your DNS records logs [url] Displays the logs for a deployment scale [args] Scales the instance count of a deployment help [cmd] Displays complete help for [cmd] ${chalk.dim('Administrative')} billing | cc [cmd] Manages your credit cards and billing methods upgrade | downgrade [plan] Upgrades or downgrades your plan teams [team] Manages your teams switch Switches between teams and your account login Login into your account or creates a new one logout Logout from your account ${chalk.dim('Options:')} -h, --help Output usage information -v, --version Output the version number -n, --name Set the name of the deployment -c ${chalk.underline('FILE')}, --config=${chalk.underline( 'FILE' )} Config file -d, --debug Debug mode [off] -f, --force Force a new deployment even if nothing has changed -t ${chalk.underline('TOKEN')}, --token=${chalk.underline( 'TOKEN' )} Login token -L, --login Configure login -l, --links Copy symlinks without resolving their target -p, --public Deployment is public (${chalk.dim( '`/_src`' )} is exposed) [on for oss, off for premium] -e, --env Include an env var (e.g.: ${chalk.dim( '`-e KEY=value`' )}). Can appear many times. -E ${chalk.underline('FILE')}, --dotenv=${chalk.underline( 'FILE' )} Include env vars from .env file. Defaults to '.env' -C, --no-clipboard Do not attempt to copy URL to clipboard -N, --forward-npm Forward login information to install private npm modules --session-affinity Session affinity, \`ip\` (default) or \`random\` to control session affinity. ${chalk.dim( 'Enforcable Types (when both package.json and Dockerfile exist):' )} --npm Node.js application --docker Docker container --static Static file hosting ${chalk.dim('Examples:')} ${chalk.gray('–')} Deploys the current directory ${chalk.cyan('$ now')} ${chalk.gray('–')} Deploys a custom path ${chalk.dim('`/usr/src/project`')} ${chalk.cyan('$ now /usr/src/project')} ${chalk.gray('–')} Deploys a GitHub repository ${chalk.cyan('$ now user/repo#ref')} ${chalk.gray('–')} Deploys a GitHub, GitLab or Bitbucket repo using its URL ${chalk.cyan('$ now https://gitlab.com/user/repo')} ${chalk.gray('–')} Deploys with ENV vars ${chalk.cyan( '$ now -e NODE_ENV=production -e MYSQL_PASSWORD=@mysql-password' )} ${chalk.gray('–')} Displays comprehensive help for the subcommand ${chalk.dim( '`list`' )} ${chalk.cyan('$ now help list')} `) } let path = argv._[0] if (path) { // If path is relative: resolve // if path is absolute: clear up strange `/` etc path = resolve(process.cwd(), path) } else { path = process.cwd() } // If the current deployment is a repo const gitRepo = {} // Options let forceNew = argv.force let deploymentName = argv.name let sessionAffinity = argv['session-affinity'] const debug = argv.debug const clipboard = !argv['no-clipboard'] const forwardNpm = argv['forward-npm'] const forceSync = argv.forceSync const shouldLogin = argv.login const followSymlinks = !argv.links const wantsPublic = argv.public const apiUrl = argv.url || 'https://api.zeit.co' const isTTY = process.stdout.isTTY const quiet = !isTTY const autoAliases = typeof argv.alias === 'undefined' ? false : flatten([argv.alias]) if (argv.config) { cfg.setConfigFile(argv.config) } if (Array.isArray(autoAliases)) { console.log( `${chalk.red('Deprecated!')} The option ${chalk.grey( '--alias' )} will be removed soon.` ) console.log('Read more about the new way here: http://bit.ly/2l2v5Fg\n') } const stopDeployment = msg => { handleError(msg) process.exit(1) } const envFields = async list => { const questions = [] for (const field of list) { questions.push({ name: field, message: field }) } // eslint-disable-next-line import/no-unassigned-import require('../lib/utils/input/patch-inquirer') info('Please enter the values for the following environment variables:') const answers = await inquirer.prompt(questions) for (const answer in answers) { if (!{}.hasOwnProperty.call(answers, answer)) { continue } const content = answers[answer] if (content === '') { stopDeployment(`Enter a value for ${answer}`) } } return answers } // Create a new deployment if user changed the name or made `_src` public. // This works fine because it doesn't force a new sync, // it just forces a new deployment. if (deploymentName || wantsPublic) { forceNew = true } let alwaysForwardNpm async function main() { let config = await cfg.read({ token: argv.token }) alwaysForwardNpm = config.forwardNpm if (argv.h || argv.help) { help() return exit(0) } else if (argv.v || argv.version) { console.log(version) return exit(0) } let token = argv.token || config.token if (!token || shouldLogin) { try { token = await login(apiUrl) config = await cfg.read() } catch (err) { return stopDeployment(`Authentication error – ${err.message}`) } console.log( `> Logged in successfully. Token saved to ${chalk.bold('~/.now.json')}.` ) console.log( `> Run ${cmd('now')} to deploy the current directory, or ${cmd( 'now --help' )} for usage info.\n` ) return exit(0) } // If we got to here then `token` should be set try { await sync({ token, config }) } catch (err) { return stopDeployment(err) } } async function sync({ token, config: { currentTeam, user } }) { const start = Date.now() const rawPath = argv._[0] const planPromise = new NowPlans({ apiUrl, token, debug, currentTeam }).getCurrent() try { await fs.stat(path) } catch (err) { let repo let isValidRepo = false try { isValidRepo = isRepoPath(rawPath) } catch (err) { if (err.code === 'INVALID_URL') { stopDeployment(err) } else { throw err } } if (isValidRepo) { const gitParts = gitPathParts(rawPath) Object.assign(gitRepo, gitParts) const searchMessage = setTimeout(() => { console.log(`> Didn't find directory. Searching on ${gitRepo.type}...`) }, 500) try { repo = await fromGit(rawPath, debug) } catch (err) {} clearTimeout(searchMessage) } if (repo) { // Tell now which directory to deploy path = repo.path // Set global variable for deleting tmp dir later // once the deployment has finished Object.assign(gitRepo, repo) } else if (isValidRepo) { const gitRef = gitRepo.ref ? `with "${chalk.bold(gitRepo.ref)}" ` : '' stopDeployment( `There's no repository named "${chalk.bold( gitRepo.main )}" ${gitRef}on ${gitRepo.type}` ) } else { stopDeployment(`Could not read directory ${chalk.bold(path)}`) } } // Make sure that directory is deployable try { await checkPath(path) } catch (err) { error(err) return } if (!quiet) { if (gitRepo.main) { const gitRef = gitRepo.ref ? ` at "${chalk.bold(gitRepo.ref)}" ` : '' console.log( `> Deploying ${gitRepo.type} repository "${chalk.bold( gitRepo.main )}" ${gitRef} under ${chalk.bold( (currentTeam && currentTeam.slug) || user.username || user.email )}` ) } else { console.log( `> Deploying ${chalk.bold(toHumanPath(path))} under ${chalk.bold( (currentTeam && currentTeam.slug) || user.username || user.email )}` ) } } let deploymentType // CLI deployment type explicit overrides if (argv.docker) { if (debug) { console.log(`> [debug] Forcing \`deploymentType\` = \`docker\``) } deploymentType = 'docker' } else if (argv.npm) { if (debug) { console.log(`> [debug] Forcing \`deploymentType\` = \`npm\``) } deploymentType = 'npm' } else if (argv.static) { if (debug) { console.log(`> [debug] Forcing \`deploymentType\` = \`static\``) } deploymentType = 'static' } let meta ;({ meta, deploymentName, deploymentType, sessionAffinity } = await readMeta( path, deploymentName, deploymentType, sessionAffinity )) const nowConfig = meta.nowConfig const now = new Now({ apiUrl, token, debug, currentTeam }) let dotenvConfig let dotenvOption if (argv.dotenv) { dotenvOption = argv.dotenv } else if (nowConfig && nowConfig.dotenv) { dotenvOption = nowConfig.dotenv } if (dotenvOption) { const dotenvFileName = typeof dotenvOption === 'string' ? dotenvOption : '.env' if (!fs.existsSync(dotenvFileName)) { error(`--dotenv flag is set but ${dotenvFileName} file is missing`) return process.exit(1) } const dotenvFile = await fs.readFile(dotenvFileName) dotenvConfig = dotenv.parse(dotenvFile) } let pkgEnv = nowConfig && nowConfig.env const argEnv = [].concat(argv.env || []) if (pkgEnv && Array.isArray(nowConfig.env)) { const defined = argEnv.join() const askFor = nowConfig.env.filter(item => !defined.includes(`${item}=`)) pkgEnv = await envFields(askFor) } // Merge `now.env` from package.json with `-e` arguments const envs = [ ...Object.keys(dotenvConfig || {}).map(k => `${k}=${dotenvConfig[k]}`), ...Object.keys(pkgEnv || {}).map(k => `${k}=${pkgEnv[k]}`), ...argEnv ] let secrets const findSecret = async uidOrName => { if (!secrets) { secrets = await now.listSecrets() } return secrets.filter(secret => { return secret.name === uidOrName || secret.uid === uidOrName }) } const env_ = await Promise.all( envs.map(async kv => { if (typeof kv !== 'string') { error('Env key and value missing') return process.exit(1) } const [key, ...rest] = kv.split('=') let val if (rest.length > 0) { val = rest.join('=') } if (/[^A-z0-9_]/i.test(key)) { error( `Invalid ${chalk.dim('-e')} key ${chalk.bold( `"${chalk.bold(key)}"` )}. Only letters, digits and underscores are allowed.` ) return process.exit(1) } if (!key) { error(`Invalid env option ${chalk.bold(`"${kv}"`)}`) return process.exit(1) } if (val === undefined) { if (key in process.env) { console.log( `> Reading ${chalk.bold( `"${chalk.bold(key)}"` )} from your env (as no value was specified)` ) // Escape value if it begins with @ val = process.env[key].replace(/^@/, '\\@') } else { error( `No value specified for env ${chalk.bold( `"${chalk.bold(key)}"` )} and it was not found in your env.` ) return process.exit(1) } } if (val[0] === '@') { const uidOrName = val.substr(1) const secrets = await findSecret(uidOrName) if (secrets.length === 0) { if (uidOrName === '') { error( `Empty reference provided for env key ${chalk.bold( `"${chalk.bold(key)}"` )}` ) } else { error( `No secret found by uid or name ${chalk.bold(`"${uidOrName}"`)}` ) } return process.exit(1) } else if (secrets.length > 1) { error( `Ambiguous secret ${chalk.bold( `"${uidOrName}"` )} (matches ${chalk.bold(secrets.length)} secrets)` ) return process.exit(1) } val = { uid: secrets[0].uid } } return [key, typeof val === 'string' ? val.replace(/^\\@/, '@') : val] }) ) const env = {} env_.filter(v => Boolean(v)).forEach(([key, val]) => { if (key in env) { note(`Overriding duplicate env key ${chalk.bold(`"${key}"`)}`) } env[key] = val }) try { await now.create( path, Object.assign( { env, followSymlinks, forceNew, forceSync, forwardNpm: alwaysForwardNpm || forwardNpm, quiet, wantsPublic, sessionAffinity }, meta ) ) } catch (err) { if (debug) { console.log(`> [debug] error: ${err}\n${err.stack}`) } return stopDeployment(err) } const { url } = now const elapsed = ms(new Date() - start) if (isTTY) { if (clipboard) { try { await copy(url) console.log( `${chalk.cyan('> Ready!')} ${chalk.bold( url )} (copied to clipboard) [${elapsed}]` ) } catch (err) { console.log(`${chalk.cyan('> Ready!')} ${chalk.bold(url)} [${elapsed}]`) } } else { console.log(`> ${url} [${elapsed}]`) } } else { process.stdout.write(url) } const startU = new Date() const complete = ({ syncCount }) => { if (!quiet) { const elapsedU = ms(new Date() - startU) console.log( `> Synced ${syncCount} (${bytes(now.syncAmount)}) [${elapsedU}] ` ) console.log('> Initializing…') } // Close http2 agent now.close() // Show build logs if (!quiet) { if (deploymentType === 'static') { console.log(`${chalk.cyan('> Deployment complete!')}`) } else { printLogs(now.host, token, currentTeam, user) } } } const plan = await planPromise if (plan.id === 'oss' && !wantsPublic) { if (isTTY) { info( `${chalk.bold( (currentTeam && `${currentTeam.slug} is`) || `You (${user.username || user.email}) are` )} on the OSS plan. Your code and logs will be made ${chalk.bold( 'public' )}.` ) const proceed = await promptBool( 'Are you sure you want to proceed with the deployment?', { trailing: eraseLines(1) } ) if (proceed) { note(`You can use ${cmd('now --public')} to skip this prompt`) } else { const stopSpinner = wait('Canceling deployment') await now.remove(now.id, { hard: true }) stopSpinner() info('Deployment aborted. No files were synced.') info(`You can upgrade by running ${cmd('now upgrade')}.`) return exit() } } else if (!wantsPublic) { const msg = '\nYou are on the OSS plan. Your code and logs will be made public.' + ' If you agree with that, please run again with --public.' return stopDeployment(msg) } } if (now.syncAmount) { if (debug && now.syncFileCount !== now.fileCount) { console.log( `> [debug] total files ${now.fileCount}, ${now.syncFileCount} changed. ` ) } const size = bytes(now.syncAmount) const syncCount = `${now.syncFileCount} file${now.syncFileCount > 1 ? 's' : ''}` const bar = new Progress( `> Upload [:bar] :percent :etas (${size}) [${syncCount}]`, { width: 20, complete: '=', incomplete: '', total: now.syncAmount, clear: true } ) now.upload() now.on('upload', ({ names, data }) => { const amount = data.length if (debug) { console.log( `> [debug] Uploaded: ${names.join(' ')} (${bytes(data.length)})` ) } bar.tick(amount) }) now.on('complete', () => complete({ syncCount })) now.on('error', err => { error('Upload failed') return stopDeployment(err) }) } else { if (!quiet) { console.log(`> Initializing…`) } // Close http2 agent now.close() // Show build logs if (!quiet) { if (deploymentType === 'static') { console.log(`${chalk.cyan('> Deployment complete!')}`) } else { printLogs(now.host, token, currentTeam, user) } } } } async function readMeta(path, deploymentName, deploymentType, sessionAffinity) { try { const meta = await readMetaData(path, { deploymentType, deploymentName, quiet: true, sessionAffinity }) if (!deploymentType) { deploymentType = meta.type if (debug) { console.log( `> [debug] Detected \`deploymentType\` = \`${deploymentType}\`` ) } } if (!deploymentName) { deploymentName = meta.name if (debug) { console.log( `> [debug] Detected \`deploymentName\` = "${deploymentName}"` ) } } return { meta, deploymentName, deploymentType, sessionAffinity } } catch (err) { if (isTTY && err.code === 'MULTIPLE_MANIFESTS') { if (debug) { console.log('> [debug] Multiple manifests found, disambiguating') } console.log( `> Two manifests found. Press [${chalk.bold( 'n' )}] to deploy or re-run with --flag` ) deploymentType = await promptOptions([ ['npm', `${chalk.bold('package.json')}\t${chalk.gray(' --npm')} `], ['docker', `${chalk.bold('Dockerfile')}\t${chalk.gray('--docker')} `] ]) if (debug) { console.log( `> [debug] Selected \`deploymentType\` = "${deploymentType}"` ) } return readMeta(path, deploymentName, deploymentType) } throw err } } function printLogs(host, token, currentTeam, user) { // Log build const logger = new Logger(host, token, { debug, quiet }) logger.on('error', async err => { if (!quiet) { if (err && err.type === 'BUILD_ERROR') { error( `The build step of your project failed. To retry, run ${cmd( 'now --force' )}.` ) } else { error('Deployment failed') } } if (gitRepo && gitRepo.cleanup) { // Delete temporary directory that contains repository gitRepo.cleanup() if (debug) { console.log(`> [debug] Removed temporary repo directory`) } } process.exit(1) }) logger.on('close', async () => { if (Array.isArray(autoAliases)) { const aliasList = autoAliases.filter(item => item !== '') if (aliasList.length > 0) { const assignments = [] for (const alias of aliasList) { assignments.push( assignAlias(alias, token, host, apiUrl, debug, currentTeam, user) ) } await Promise.all(assignments) } else { await reAlias( token, host, null, help, exit, apiUrl, debug, currentTeam, user ) } } if (!quiet) { console.log(`${chalk.cyan('> Deployment complete!')}`) } if (gitRepo && gitRepo.cleanup) { // Delete temporary directory that contains repository gitRepo.cleanup() if (debug) { console.log(`> [debug] Removed temporary repo directory`) } } process.exit(0) }) } main()