Leo Lamprecht
8 years ago
5 changed files with 1001 additions and 1 deletions
@ -0,0 +1,533 @@ |
|||
#!/usr/bin/env node
|
|||
|
|||
// Packages
|
|||
const chalk = require('chalk') |
|||
const minimist = require('minimist') |
|||
const table = require('text-table') |
|||
const ms = require('ms') |
|||
const printf = require('printf') |
|||
require('epipebomb')() |
|||
const supportsColor = require('supports-color') |
|||
|
|||
// Ours
|
|||
const strlen = require('../lib/strlen') |
|||
const NowAlias = require('../lib/alias') |
|||
const NowDomains = require('../lib/domains') |
|||
const login = require('../lib/login') |
|||
const cfg = require('../lib/cfg') |
|||
const { handleError, error } = require('../lib/error') |
|||
const toHost = require('../lib/to-host') |
|||
const { reAlias } = require('../lib/re-alias') |
|||
const exit = require('../lib/utils/exit') |
|||
const info = require('../lib/utils/output/info') |
|||
const logo = require('../lib/utils/output/logo') |
|||
const promptBool = require('../lib/utils/input/prompt-bool') |
|||
|
|||
const grayWidth = 10 |
|||
const underlineWidth = 11 |
|||
|
|||
// Options
|
|||
const help = () => { |
|||
console.log(` |
|||
${chalk.bold(`${logo} now alias`)} <ls | set | rm> <deployment> <alias> |
|||
|
|||
${chalk.dim('Options:')} |
|||
|
|||
-h, --help Output usage information |
|||
-c ${chalk.bold.underline('FILE')}, --config=${chalk.bold.underline( |
|||
'FILE' |
|||
)} Config file |
|||
-r ${chalk.bold.underline('RULES_FILE')}, --rules=${chalk.bold.underline( |
|||
'RULES_FILE' |
|||
)} Rules file |
|||
-d, --debug Debug mode [off] |
|||
-t ${chalk.bold.underline('TOKEN')}, --token=${chalk.bold.underline( |
|||
'TOKEN' |
|||
)} Login token |
|||
|
|||
${chalk.dim('Examples:')} |
|||
|
|||
${chalk.gray('–')} Lists all your aliases: |
|||
|
|||
${chalk.cyan('$ now alias ls')} |
|||
|
|||
${chalk.gray('–')} Adds a new alias to ${chalk.underline('my-api.now.sh')}: |
|||
|
|||
${chalk.cyan( |
|||
`$ now alias set ${chalk.underline( |
|||
'api-ownv3nc9f8.now.sh' |
|||
)} ${chalk.underline('my-api.now.sh')}` |
|||
)} |
|||
|
|||
The ${chalk.dim('`.now.sh`')} suffix can be ommited: |
|||
|
|||
${chalk.cyan('$ now alias set api-ownv3nc9f8 my-api')} |
|||
|
|||
The deployment id can be used as the source: |
|||
|
|||
${chalk.cyan('$ now alias set deploymentId my-alias')} |
|||
|
|||
Custom domains work as alias targets: |
|||
|
|||
${chalk.cyan( |
|||
`$ now alias set ${chalk.underline( |
|||
'api-ownv3nc9f8.now.sh' |
|||
)} ${chalk.underline('my-api.com')}` |
|||
)} |
|||
|
|||
${chalk.dim('–')} The subcommand ${chalk.dim( |
|||
'`set`' |
|||
)} is the default and can be skipped. |
|||
${chalk.dim('–')} ${chalk.dim( |
|||
'`http(s)://`' |
|||
)} in the URLs is unneeded / ignored. |
|||
|
|||
${chalk.gray('–')} Add and modify path based aliases for ${chalk.underline( |
|||
'zeit.ninja' |
|||
)}: |
|||
|
|||
${chalk.cyan( |
|||
`$ now alias ${chalk.underline('zeit.ninja')} -r ${chalk.underline( |
|||
'rules.json' |
|||
)}` |
|||
)} |
|||
|
|||
Export effective routing rules: |
|||
|
|||
${chalk.cyan( |
|||
`$ now alias ls aliasId --json > ${chalk.underline('rules.json')}` |
|||
)} |
|||
|
|||
${chalk.cyan(`$ now alias ls zeit.ninja`)} |
|||
|
|||
${chalk.gray('–')} Removing an alias: |
|||
|
|||
${chalk.cyan('$ now alias rm aliasId')} |
|||
|
|||
To get the list of alias ids, use ${chalk.dim('`now alias ls`')}. |
|||
|
|||
${chalk.dim('Alias:')} ln |
|||
`)
|
|||
} |
|||
|
|||
// Options
|
|||
const debug = false |
|||
const apiUrl = 'https://api.zeit.co' |
|||
|
|||
let argv |
|||
let subcommand |
|||
|
|||
const main = async ctx => { |
|||
argv = minimist(ctx.argv.slice(2), { |
|||
string: ['config', 'token', 'rules'], |
|||
boolean: ['help', 'debug'], |
|||
alias: { |
|||
help: 'h', |
|||
config: 'c', |
|||
rules: 'r', |
|||
debug: 'd', |
|||
token: 't' |
|||
} |
|||
}) |
|||
|
|||
argv._ = argv._.slice(1) |
|||
subcommand = argv._[0] |
|||
|
|||
if (argv.help) { |
|||
help() |
|||
process.exit(0) |
|||
} |
|||
|
|||
const config = await cfg.read({ token: argv.token }) |
|||
|
|||
let token |
|||
try { |
|||
token = config.token || (await login(apiUrl)) |
|||
} catch (err) { |
|||
error(`Authentication error – ${err.message}`) |
|||
exit(1) |
|||
} |
|||
|
|||
try { |
|||
await run({ token, config }) |
|||
} catch (err) { |
|||
if (err.userError) { |
|||
error(err.message) |
|||
} else { |
|||
error(`Unknown error: ${err}\n${err.stack}`) |
|||
} |
|||
exit(1) |
|||
} |
|||
} |
|||
|
|||
module.exports = async ctx => { |
|||
try { |
|||
await main(ctx) |
|||
} catch (err) { |
|||
handleError(err) |
|||
process.exit(1) |
|||
} |
|||
} |
|||
|
|||
async function run({ token, config: { currentTeam, user } }) { |
|||
const alias = new NowAlias({ apiUrl, token, debug, currentTeam }) |
|||
const domains = new NowDomains({ apiUrl, token, debug, currentTeam }) |
|||
const args = argv._.slice(1) |
|||
|
|||
switch (subcommand) { |
|||
case 'ls': |
|||
case 'list': { |
|||
if (args.length === 1) { |
|||
const list = await alias.listAliases() |
|||
const item = list.find( |
|||
e => e.uid === argv._[1] || e.alias === argv._[1] |
|||
) |
|||
if (!item || !item.rules) { |
|||
error(`Could not match path alias for: ${argv._[1]}`) |
|||
return exit(1) |
|||
} |
|||
|
|||
if (argv.json) { |
|||
console.log(JSON.stringify({ rules: item.rules }, null, 2)) |
|||
} else { |
|||
const header = [ |
|||
['', 'pathname', 'method', 'dest'].map(s => chalk.dim(s)) |
|||
] |
|||
const text = |
|||
list.length === 0 |
|||
? null |
|||
: table( |
|||
header.concat( |
|||
item.rules.map(rule => { |
|||
return [ |
|||
'', |
|||
rule.pathname ? rule.pathname : '', |
|||
rule.method ? rule.method : '*', |
|||
rule.dest |
|||
] |
|||
}) |
|||
), |
|||
{ |
|||
align: ['l', 'l', 'l', 'l'], |
|||
hsep: ' '.repeat(2), |
|||
stringLength: strlen |
|||
} |
|||
) |
|||
|
|||
console.log(text) |
|||
} |
|||
break |
|||
} else if (args.length !== 0) { |
|||
error( |
|||
`Invalid number of arguments. Usage: ${chalk.cyan('`now alias ls`')}` |
|||
) |
|||
return exit(1) |
|||
} |
|||
|
|||
const start_ = new Date() |
|||
const aliases = await alias.ls() |
|||
aliases.sort((a, b) => new Date(b.created) - new Date(a.created)) |
|||
const current = new Date() |
|||
const sourceUrlLength = |
|||
aliases.reduce((acc, i) => { |
|||
return Math.max(acc, (i.deployment && i.deployment.url.length) || 0) |
|||
}, 0) + 9 |
|||
const aliasLength = |
|||
aliases.reduce((acc, i) => { |
|||
return Math.max(acc, (i.alias && i.alias.length) || 0) |
|||
}, 0) + 8 |
|||
const elapsed_ = ms(new Date() - start_) |
|||
console.log( |
|||
`> ${aliases.length} alias${aliases.length === 1 |
|||
? '' |
|||
: 'es'} found ${chalk.gray(`[${elapsed_}]`)} under ${chalk.bold( |
|||
(currentTeam && currentTeam.slug) || user.username || user.email |
|||
)}` |
|||
) |
|||
console.log() |
|||
|
|||
if (supportsColor) { |
|||
const urlSpecHeader = `%-${sourceUrlLength + 1}s` |
|||
const aliasSpecHeader = `%-${aliasLength + 1}s` |
|||
console.log( |
|||
printf( |
|||
` ${chalk.gray(urlSpecHeader + ' ' + aliasSpecHeader + ' %5s')}`, |
|||
'source', |
|||
'url', |
|||
'age' |
|||
) |
|||
) |
|||
} else { |
|||
const urlSpecHeader = `%-${sourceUrlLength}s` |
|||
const aliasSpecHeader = `%-${aliasLength}s` |
|||
console.log( |
|||
printf( |
|||
` ${urlSpecHeader} ${aliasSpecHeader} %5s`, |
|||
'source', |
|||
'url', |
|||
'age' |
|||
) |
|||
) |
|||
} |
|||
|
|||
let text = '' |
|||
aliases.forEach(_alias => { |
|||
let urlSpec = sourceUrlLength |
|||
let aliasSpec = aliasLength |
|||
let ageSpec = 5 |
|||
const _url = chalk.underline(_alias.alias) |
|||
let _sourceUrl |
|||
if (supportsColor) { |
|||
aliasSpec += underlineWidth |
|||
ageSpec += grayWidth |
|||
} |
|||
if (_alias.deployment) { |
|||
_sourceUrl = chalk.underline(_alias.deployment.url) |
|||
if (supportsColor) { |
|||
urlSpec += grayWidth |
|||
} |
|||
} else if (_alias.rules) { |
|||
_sourceUrl = chalk.gray( |
|||
`[${_alias.rules.length} custom rule${_alias.rules.length > 1 |
|||
? 's' |
|||
: ''}]` |
|||
) |
|||
if (supportsColor) { |
|||
urlSpec += underlineWidth |
|||
} |
|||
} else { |
|||
_sourceUrl = chalk.gray('<null>') |
|||
} |
|||
|
|||
const time = chalk.gray(ms(current - new Date(_alias.created))) |
|||
text += printf( |
|||
` %-${urlSpec}s %-${aliasSpec}s %${ageSpec}s\n`, |
|||
_sourceUrl, |
|||
_url, |
|||
time |
|||
) |
|||
}) |
|||
|
|||
console.log(text) |
|||
break |
|||
} |
|||
case 'rm': |
|||
case 'remove': { |
|||
const _target = String(args[0]) |
|||
if (!_target) { |
|||
const err = new Error('No alias id specified') |
|||
err.userError = true |
|||
throw err |
|||
} |
|||
|
|||
if (args.length !== 1) { |
|||
error( |
|||
`Invalid number of arguments. Usage: ${chalk.cyan( |
|||
'`now alias rm <id>`' |
|||
)}` |
|||
) |
|||
return exit(1) |
|||
} |
|||
|
|||
const _aliases = await alias.ls() |
|||
const _alias = findAlias(_target, _aliases) |
|||
|
|||
if (!_alias) { |
|||
const err = new Error( |
|||
`Alias not found by "${_target}" under ${chalk.bold( |
|||
(currentTeam && currentTeam.slug) || user.username || user.email |
|||
)}. Run ${chalk.dim('`now alias ls`')} to see your aliases.` |
|||
) |
|||
err.userError = true |
|||
throw err |
|||
} |
|||
|
|||
try { |
|||
const confirmation = await confirmDeploymentRemoval(alias, _alias) |
|||
if (!confirmation) { |
|||
info('Aborted') |
|||
return process.exit(0) |
|||
} |
|||
|
|||
const start = new Date() |
|||
await alias.rm(_alias) |
|||
const elapsed = ms(new Date() - start) |
|||
console.log( |
|||
`${chalk.cyan('> Success!')} Alias ${chalk.bold( |
|||
_alias.uid |
|||
)} removed [${elapsed}]` |
|||
) |
|||
} catch (err) { |
|||
error(err) |
|||
exit(1) |
|||
} |
|||
|
|||
break |
|||
} |
|||
case 'add': |
|||
case 'set': { |
|||
if (argv.rules) { |
|||
await updatePathAlias(alias, argv._[0], argv.rules, domains) |
|||
break |
|||
} |
|||
if (args.length !== 2) { |
|||
error( |
|||
`Invalid number of arguments. Usage: ${chalk.cyan( |
|||
'`now alias set <id> <domain>`' |
|||
)}` |
|||
) |
|||
return exit(1) |
|||
} |
|||
await alias.set( |
|||
String(args[0]), |
|||
String(args[1]), |
|||
domains, |
|||
currentTeam, |
|||
user |
|||
) |
|||
break |
|||
} |
|||
default: { |
|||
if (argv._.length === 0) { |
|||
await reAlias( |
|||
token, |
|||
null, |
|||
null, |
|||
help, |
|||
exit, |
|||
apiUrl, |
|||
debug, |
|||
alias, |
|||
currentTeam, |
|||
user |
|||
) |
|||
break |
|||
} |
|||
|
|||
if (argv.rules) { |
|||
await updatePathAlias(alias, argv._[0], argv.rules, domains) |
|||
break |
|||
} |
|||
|
|||
if (argv._.length === 1) { |
|||
await reAlias( |
|||
token, |
|||
null, |
|||
String(argv._[0]), |
|||
help, |
|||
exit, |
|||
apiUrl, |
|||
debug, |
|||
alias, |
|||
currentTeam, |
|||
user |
|||
) |
|||
break |
|||
} else if (argv._.length === 2) { |
|||
await alias.set( |
|||
String(argv._[0]), |
|||
String(argv._[1]), |
|||
domains, |
|||
currentTeam, |
|||
user |
|||
) |
|||
} else if (argv._.length >= 3) { |
|||
error('Invalid number of arguments') |
|||
help() |
|||
exit(1) |
|||
} else { |
|||
error('Please specify a valid subcommand: ls | set | rm') |
|||
help() |
|||
exit(1) |
|||
} |
|||
} |
|||
} |
|||
|
|||
domains.close() |
|||
alias.close() |
|||
} |
|||
|
|||
async function confirmDeploymentRemoval(alias, _alias) { |
|||
const time = chalk.gray(ms(new Date() - new Date(_alias.created)) + ' ago') |
|||
const _sourceUrl = _alias.deployment |
|||
? chalk.underline(_alias.deployment.url) |
|||
: null |
|||
const tbl = table( |
|||
[ |
|||
[ |
|||
_alias.uid, |
|||
...(_sourceUrl ? [_sourceUrl] : []), |
|||
chalk.underline(_alias.alias), |
|||
time |
|||
] |
|||
], |
|||
{ hsep: ' '.repeat(6) } |
|||
) |
|||
|
|||
const msg = |
|||
'> The following alias will be removed permanently\n' + |
|||
` ${tbl} \nAre you sure?` |
|||
|
|||
return promptBool(msg, { |
|||
trailing: '\n' |
|||
}) |
|||
} |
|||
|
|||
function findAlias(alias, list) { |
|||
let key |
|||
let val |
|||
|
|||
if (/\./.test(alias)) { |
|||
val = toHost(alias) |
|||
key = 'alias' |
|||
} else { |
|||
val = alias |
|||
key = 'uid' |
|||
} |
|||
|
|||
const _alias = list.find(d => { |
|||
if (d[key] === val) { |
|||
if (debug) { |
|||
console.log(`> [debug] matched alias ${d.uid} by ${key} ${val}`) |
|||
} |
|||
|
|||
return true |
|||
} |
|||
|
|||
// Match prefix
|
|||
if (`${val}.now.sh` === d.alias) { |
|||
if (debug) { |
|||
console.log(`> [debug] matched alias ${d.uid} by url ${d.host}`) |
|||
} |
|||
|
|||
return true |
|||
} |
|||
|
|||
return false |
|||
}) |
|||
|
|||
return _alias |
|||
} |
|||
|
|||
async function updatePathAlias(alias, aliasName, rules, domains) { |
|||
const start = new Date() |
|||
const res = await alias.updatePathBasedroutes( |
|||
String(aliasName), |
|||
rules, |
|||
domains |
|||
) |
|||
const elapsed = ms(new Date() - start) |
|||
if (res.error) { |
|||
const err = new Error(res.error.message) |
|||
err.userError = true |
|||
throw err |
|||
} else { |
|||
console.log( |
|||
`${chalk.cyan( |
|||
'> Success!' |
|||
)} ${res.ruleCount} rules configured for ${chalk.underline( |
|||
res.alias |
|||
)} [${elapsed}]` |
|||
) |
|||
} |
|||
} |
@ -0,0 +1,392 @@ |
|||
#!/usr/bin/env node
|
|||
|
|||
// Packages
|
|||
const chalk = require('chalk') |
|||
const isURL = require('is-url') |
|||
const minimist = require('minimist') |
|||
const ms = require('ms') |
|||
const printf = require('printf') |
|||
require('epipebomb')() |
|||
const supportsColor = require('supports-color') |
|||
|
|||
// Ours
|
|||
const cfg = require('../lib/cfg') |
|||
const { handleError, error } = require('../lib/error') |
|||
const NowScale = require('../lib/scale') |
|||
const login = require('../lib/login') |
|||
const exit = require('../lib/utils/exit') |
|||
const logo = require('../lib/utils/output/logo') |
|||
const info = require('../lib/scale-info') |
|||
const sort = require('../lib/sort-deployments') |
|||
const success = require('../lib/utils/output/success') |
|||
|
|||
let id |
|||
let scaleArg |
|||
let optionalScaleArg |
|||
|
|||
// Options
|
|||
const help = () => { |
|||
console.log(` |
|||
${chalk.bold(`${logo} now scale`)} ls |
|||
${chalk.bold(`${logo} now scale`)} <url> |
|||
${chalk.bold(`${logo} now scale`)} <url> <min> [max] |
|||
|
|||
${chalk.dim('Options:')} |
|||
|
|||
-h, --help Output usage information |
|||
-c ${chalk.bold.underline('FILE')}, --config=${chalk.bold.underline( |
|||
'FILE' |
|||
)} Config file |
|||
-d, --debug Debug mode [off] |
|||
|
|||
${chalk.dim('Examples:')} |
|||
|
|||
${chalk.gray('–')} Create a deployment with 3 instances, never sleeps: |
|||
|
|||
${chalk.cyan('$ now scale my-deployment-ntahoeato.now.sh 3')} |
|||
|
|||
${chalk.gray('–')} Create an automatically scaling deployment: |
|||
|
|||
${chalk.cyan('$ now scale my-deployment-ntahoeato.now.sh 1 5')} |
|||
|
|||
${chalk.gray( |
|||
'–' |
|||
)} Create an automatically scaling deployment without specifying max: |
|||
|
|||
${chalk.cyan('$ now scale my-deployment-ntahoeato.now.sh 1 auto')} |
|||
|
|||
${chalk.gray( |
|||
'–' |
|||
)} Create an automatically scaling deployment without specifying min or max: |
|||
|
|||
${chalk.cyan('$ now scale my-deployment-ntahoeato.now.sh auto')} |
|||
|
|||
${chalk.gray( |
|||
'–' |
|||
)} Create a deployment that is always active and never "sleeps": |
|||
|
|||
${chalk.cyan('$ now scale my-deployment-ntahoeato.now.sh 1')} |
|||
`)
|
|||
} |
|||
|
|||
// Options
|
|||
const debug = false |
|||
const apiUrl = 'https://api.zeit.co' |
|||
|
|||
let argv |
|||
|
|||
const main = async ctx => { |
|||
argv = minimist(ctx.argv.slice(2), { |
|||
string: ['config', 'token'], |
|||
boolean: ['help', 'debug'], |
|||
alias: { help: 'h', config: 'c', debug: 'd', token: 't' } |
|||
}) |
|||
|
|||
argv._ = argv._.slice(1) |
|||
|
|||
id = argv._[0] |
|||
scaleArg = argv._[1] |
|||
optionalScaleArg = argv._[2] |
|||
|
|||
if (argv.help) { |
|||
help() |
|||
process.exit(0) |
|||
} |
|||
|
|||
const config = await cfg.read({ token: argv.token }) |
|||
|
|||
let token |
|||
try { |
|||
token = config.token || (await login(apiUrl)) |
|||
} catch (err) { |
|||
error(`Authentication error – ${err.message}`) |
|||
exit(1) |
|||
} |
|||
|
|||
try { |
|||
await run({ token, config }) |
|||
} catch (err) { |
|||
if (err.userError) { |
|||
error(err.message) |
|||
} else { |
|||
error(`Unknown error: ${err}\n${err.stack}`) |
|||
} |
|||
exit(1) |
|||
} |
|||
} |
|||
|
|||
module.exports = async ctx => { |
|||
try { |
|||
await main(ctx) |
|||
} catch (err) { |
|||
handleError(err) |
|||
process.exit(1) |
|||
} |
|||
} |
|||
|
|||
function guessParams() { |
|||
if (Number.isInteger(scaleArg) && !optionalScaleArg) { |
|||
return { min: scaleArg, max: scaleArg } |
|||
} else if (Number.isInteger(scaleArg) && Number.isInteger(optionalScaleArg)) { |
|||
return { min: scaleArg, max: optionalScaleArg } |
|||
} else if (Number.isInteger(scaleArg) && optionalScaleArg === 'auto') { |
|||
return { min: scaleArg, max: 'auto' } |
|||
} else if ( |
|||
(!scaleArg && !optionalScaleArg) || |
|||
(scaleArg === 'auto' && !optionalScaleArg) |
|||
) { |
|||
return { min: 1, max: 'auto' } |
|||
} |
|||
help() |
|||
process.exit(1) |
|||
} |
|||
|
|||
function isHostNameOrId(str) { |
|||
return ( |
|||
/(https?:\/\/)?((?:(?=[a-z0-9-]{1,63}\.)(?:xn--)?[a-z0-9]+(?:-[a-z0-9]+)*\.)+[a-z]{2,63})/.test( |
|||
str |
|||
) || str.length === 28 |
|||
) |
|||
} |
|||
|
|||
async function run({ token, config: { currentTeam } }) { |
|||
const scale = new NowScale({ apiUrl, token, debug, currentTeam }) |
|||
const start = Date.now() |
|||
|
|||
if (id === 'ls') { |
|||
await list(scale) |
|||
process.exit(0) |
|||
} else if (id === 'info') { |
|||
await info(scale) |
|||
process.exit(0) |
|||
} else if (id && isHostNameOrId(id)) { |
|||
// Normalize URL by removing slash from the end
|
|||
if (isURL(id)) { |
|||
id = id.replace(/^https:\/\//i, '') |
|||
if (id.slice(-1) === '/') { |
|||
id = id.slice(0, -1) |
|||
} |
|||
} |
|||
} else { |
|||
error('Please specify a deployment: now scale <id|url>') |
|||
help() |
|||
exit(1) |
|||
} |
|||
|
|||
const deployments = await scale.list() |
|||
|
|||
let match = deployments.find(d => { |
|||
// `url` should match the hostname of the deployment
|
|||
let u = id.replace(/^https:\/\//i, '') |
|||
|
|||
if (u.indexOf('.') === -1) { |
|||
// `.now.sh` domain is implied if just the subdomain is given
|
|||
u += '.now.sh' |
|||
} |
|||
return d.uid === id || d.name === id || d.url === u |
|||
}) |
|||
|
|||
if (!match) { |
|||
// Maybe it's an alias
|
|||
const aliasDeployment = (await scale.listAliases()).find( |
|||
e => e.alias === id |
|||
) |
|||
if (!aliasDeployment) { |
|||
error(`Could not find any deployments matching ${id}`) |
|||
return process.exit(1) |
|||
} |
|||
match = deployments.find(d => { |
|||
return d.uid === aliasDeployment.deploymentId |
|||
}) |
|||
} |
|||
|
|||
const { min, max } = guessParams() |
|||
|
|||
if ( |
|||
!(Number.isInteger(min) || min === 'auto') && |
|||
!(Number.isInteger(max) || max === 'auto') |
|||
) { |
|||
help() |
|||
return exit(1) |
|||
} |
|||
|
|||
if (match.type === 'STATIC') { |
|||
if (min === 0 && max === 0) { |
|||
error("Static deployments can't be FROZEN. Use `now rm` to remove") |
|||
return process.exit(1) |
|||
} |
|||
console.log('> Static deployments are automatically scaled!') |
|||
return process.exit(0) |
|||
} |
|||
|
|||
const { |
|||
max: currentMax, |
|||
min: currentMin, |
|||
current: currentCurrent |
|||
} = match.scale |
|||
if ( |
|||
max === currentMax && |
|||
min === currentMin && |
|||
Number.isInteger(min) && |
|||
currentCurrent >= min && |
|||
Number.isInteger(max) && |
|||
currentCurrent <= max |
|||
) { |
|||
// Nothing to do, let's print the rules
|
|||
printScaleingRules(match.url, currentCurrent, min, max) |
|||
return |
|||
} |
|||
|
|||
if ((match.state === 'FROZEN' || match.scale.current === 0) && min > 0) { |
|||
console.log( |
|||
`> Deployment is currently in 0 replicas, preparing deployment for scaling...` |
|||
) |
|||
if (match.scale.max < 1) { |
|||
await scale.setScale(match.uid, { min: 0, max: 1 }) |
|||
} |
|||
await scale.unfreeze(match) |
|||
} |
|||
|
|||
const { min: newMin, max: newMax } = await scale.setScale(match.uid, { |
|||
min, |
|||
max |
|||
}) |
|||
|
|||
const elapsed = ms(new Date() - start) |
|||
|
|||
const currentReplicas = match.scale.current |
|||
printScaleingRules(match.url, currentReplicas, newMin, newMax, elapsed) |
|||
await info(scale, match.url) |
|||
|
|||
scale.close() |
|||
} |
|||
function printScaleingRules(url, currentReplicas, min, max, elapsed) { |
|||
const log = console.log |
|||
success( |
|||
`Configured scaling rules ${chalk.gray(elapsed ? '[' + elapsed + ']' : '')}` |
|||
) |
|||
log() |
|||
log( |
|||
`${chalk.bold(url)} (${chalk.gray(currentReplicas)} ${chalk.gray( |
|||
'current' |
|||
)})` |
|||
) |
|||
log(printf('%6s %s', 'min', chalk.bold(min))) |
|||
log(printf('%6s %s', 'max', chalk.bold(max))) |
|||
log(printf('%6s %s', 'auto', chalk.bold(min === max ? '✖' : '✔'))) |
|||
log() |
|||
} |
|||
|
|||
async function list(scale) { |
|||
let deployments |
|||
try { |
|||
const app = argv._[1] |
|||
deployments = await scale.list(app) |
|||
} catch (err) { |
|||
handleError(err) |
|||
process.exit(1) |
|||
} |
|||
|
|||
scale.close() |
|||
|
|||
const apps = new Map() |
|||
|
|||
for (const dep of deployments) { |
|||
const deps = apps.get(dep.name) || [] |
|||
apps.set(dep.name, deps.concat(dep)) |
|||
} |
|||
|
|||
const sorted = await sort([...apps]) |
|||
|
|||
const timeNow = new Date() |
|||
const urlLength = |
|||
deployments.reduce((acc, i) => { |
|||
return Math.max(acc, (i.url && i.url.length) || 0) |
|||
}, 0) + 5 |
|||
|
|||
for (const app of sorted) { |
|||
const depls = argv.all ? app[1] : app[1].slice(0, 5) |
|||
console.log( |
|||
`${chalk.bold(app[0])} ${chalk.gray( |
|||
'(' + depls.length + ' of ' + app[1].length + ' total)' |
|||
)}` |
|||
) |
|||
console.log() |
|||
const urlSpec = `%-${urlLength}s` |
|||
console.log( |
|||
printf( |
|||
` ${chalk.grey(urlSpec + ' %8s %8s %8s %8s %8s')}`, |
|||
'url', |
|||
'cur', |
|||
'min', |
|||
'max', |
|||
'auto', |
|||
'age' |
|||
) |
|||
) |
|||
for (const instance of depls) { |
|||
if (!instance.scale) { |
|||
let spec |
|||
if (supportsColor) { |
|||
spec = ` %-${urlLength + 10}s %8s %8s %8s %8s %8s` |
|||
} else { |
|||
spec = ` %-${urlLength + 1}s %8s %8s %8s %8s %8s` |
|||
} |
|||
const infinite = '∞' |
|||
console.log( |
|||
printf( |
|||
spec, |
|||
chalk.underline(instance.url), |
|||
infinite, |
|||
1, |
|||
infinite, |
|||
'✔', |
|||
ms(timeNow - instance.created) |
|||
) |
|||
) |
|||
} else if (instance.scale.current > 0) { |
|||
let spec |
|||
if (supportsColor) { |
|||
spec = ` %-${urlLength + 10}s %8s %8s %8s %8s %8s` |
|||
} else { |
|||
spec = ` %-${urlLength + 1}s %8s %8s %8s %8s %8s` |
|||
} |
|||
console.log( |
|||
printf( |
|||
spec, |
|||
chalk.underline(instance.url), |
|||
instance.scale.current, |
|||
instance.scale.min, |
|||
instance.scale.max, |
|||
instance.scale.max === instance.scale.min ? '✖' : '✔', |
|||
ms(timeNow - instance.created) |
|||
) |
|||
) |
|||
} else { |
|||
let spec |
|||
if (supportsColor) { |
|||
spec = ` %-${urlLength + 10}s ${chalk.gray('%8s %8s %8s %8s %8s')}` |
|||
} else { |
|||
spec = ` %-${urlLength + 1}s ${chalk.gray('%8s %8s %8s %8s %8s')}` |
|||
} |
|||
console.log( |
|||
printf( |
|||
spec, |
|||
chalk.underline(instance.url), |
|||
instance.scale.current, |
|||
instance.scale.min, |
|||
instance.scale.max, |
|||
instance.scale.max === instance.scale.min ? '✖' : '✔', |
|||
ms(timeNow - instance.created) |
|||
) |
|||
) |
|||
} |
|||
} |
|||
console.log() |
|||
} |
|||
} |
|||
|
|||
process.on('uncaughtException', err => { |
|||
handleError(err) |
|||
exit(1) |
|||
}) |
Loading…
Reference in new issue