|
|
|
// Native
|
|
|
|
const { basename, resolve: resolvePath } = require('path')
|
|
|
|
|
|
|
|
// Packages
|
|
|
|
const chalk = require('chalk')
|
|
|
|
const { readFile } = require('fs-extra')
|
|
|
|
const { parse: parseDockerfile } = require('docker-file-parser')
|
|
|
|
const determineType = require('deployment-type')
|
|
|
|
|
|
|
|
module.exports = readMetaData
|
|
|
|
|
|
|
|
async function readMetaData(
|
|
|
|
path,
|
|
|
|
{ deploymentType, deploymentName, quiet = false, strict = true }
|
|
|
|
) {
|
|
|
|
let description
|
|
|
|
let type = deploymentType
|
|
|
|
let name = deploymentName
|
|
|
|
|
|
|
|
const 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
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We can remove this once the prompt for choosing `--npm` or `--docker` is gone
|
|
|
|
if (pkg && pkg.now && pkg.now.type) {
|
|
|
|
type = nowConfig.type
|
|
|
|
}
|
|
|
|
|
|
|
|
// The same goes for this
|
|
|
|
if (nowConfig && nowConfig.type) {
|
|
|
|
type = nowConfig.type
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!type) {
|
|
|
|
type = await determineType(path)
|
|
|
|
|
|
|
|
// Both `package.json` and `Dockerfile` exist! Prompt the user to pick one.
|
|
|
|
// We can remove this soon (details are internal) - also read the comment paragraph above
|
|
|
|
if (type === 'docker' && (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 (!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 if (type === 'static') {
|
|
|
|
// Do nothing
|
|
|
|
} 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 && type !== 'static') {
|
|
|
|
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,
|
|
|
|
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
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|