You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

193 lines
4.7 KiB

// Native
const { basename, resolve: resolvePath } = require('path')
// Packages
const chalk = require('chalk')
const { readFile } = require('fs-promise')
const { parse: parseDockerfile } = require('docker-file-parser')
// Ours
const { devDependencies: { serve: serveVersion } } = require('../lib/pkg')
// `package.json` used for "static" deployments
const staticPackage = Object.freeze({
scripts: {
start: `NODE_ENV='production' serve ./content`
},
dependencies: {
serve: serveVersion
}
})
module.exports = readMetaData
async function readMetaData(
path,
{ deploymentType, deploymentName, quiet = false, strict = true }
) {
let description
let type = deploymentType
let name = deploymentName
let isStatic = false
let pkg = await readJSON(path, 'package.json')
let nowConfig = await readJSON(path, 'now.json')
const dockerfile = await readDockerfile(path)
const hasNowJson = Boolean(nowConfig)
if (pkg && pkg.now) {
// If the project has both a `now.json` and `now` Object in the `package.json`
// file, then fail hard and let the user know that they need to pick one or the
// other
if (nowConfig) {
const err = new Error(
'You have a `now` configuration field inside `package.json` ' +
'but configuration is also present in `now.json`! ' +
"Please ensure there's a single source of configuration by removing one."
)
err.userError = true
throw err
} else {
nowConfig = pkg.now
}
}
if (!type) {
// `now.json` / `pkg.now` get default type preference
if (nowConfig) {
type = nowConfig.type
}
// Both `package.json` and `Dockerfile` exist! Prompt the user to pick one.
if (!type && pkg && dockerfile) {
const err = new Error(
'Ambiguous deployment (`package.json` and `Dockerfile` found). ' +
'Please supply `--npm` or `--docker` to disambiguate.'
)
err.userError = true
err.code = 'MULTIPLE_MANIFESTS'
throw err
}
if (!type && pkg) {
type = 'npm'
}
if (!type && dockerfile) {
type = 'docker'
}
if (!type) {
type = 'static'
}
}
// "static" deployments are (not-so) secretly just
// npm deployments under the hood
if (type === 'static') {
isStatic = true
type = 'npm'
pkg = Object.assign({ name: pkg && pkg.name }, staticPackage)
}
if (!name && nowConfig) {
name = nowConfig.name
}
if (type === 'npm') {
if (pkg) {
if (!name && pkg.name) {
name = String(pkg.name)
}
description = pkg.description
}
} else if (type === 'docker') {
if (strict && dockerfile.length <= 0) {
const err = new Error('No commands found in `Dockerfile`')
err.userError = true
throw err
}
const labels = {}
dockerfile.filter(cmd => cmd.name === 'LABEL').forEach(({ args }) => {
for (const key in args) {
if (!{}.hasOwnProperty.call(args, key)) {
continue
}
// Unescape and convert into string
try {
labels[key] = args[key]
} catch (err) {
const e = new Error(
`Error parsing value for LABEL ${key} in \`Dockerfile\``
)
e.userError = true
throw e
}
}
})
if (!name) {
name = labels.name
}
description = labels.description
} else {
throw new TypeError(`Unsupported "deploymentType": ${type}`)
}
// No name in `package.json` / `now.json`, or "name" label in Dockerfile.
// Default to the basename of the root dir
if (!name) {
name = basename(path)
if (!quiet && !isStatic) {
if (type === 'docker') {
console.log(`> No \`name\` LABEL in \`Dockerfile\`, using ${chalk.bold(name)}`)
} else {
console.log(`> No \`name\` in \`package.json\`, using ${chalk.bold(name)}`)
}
}
}
return {
name,
description,
type,
isStatic,
pkg,
nowConfig,
hasNowJson,
// XXX: legacy
deploymentType: type
}
}
async function readJSON(path, name) {
try {
const contents = await readFile(resolvePath(path, name), 'utf8')
return JSON.parse(contents)
} catch (err) {
// If the file doesn't exist then that's fine; any other error bubbles up
if (err.code !== 'ENOENT') {
err.userError = true
throw err
}
}
}
async function readDockerfile(path, name = 'Dockerfile') {
try {
const contents = await readFile(resolvePath(path, name), 'utf8')
return parseDockerfile(contents, { includeComments: true })
} catch (err) {
// If the file doesn't exist then that's fine; any other error bubbles up
if (err.code !== 'ENOENT') {
err.userError = true
throw err
}
}
}