2017-12-12 00:37:56 +02:00
|
|
|
# Imported here because it's needed for the setup
|
|
|
|
# of this action
|
2017-03-29 12:03:40 +01:00
|
|
|
Promise = require('bluebird')
|
2017-04-24 20:05:18 +01:00
|
|
|
dockerUtils = require('../utils/docker')
|
2017-12-12 00:37:56 +02:00
|
|
|
compose = require('../utils/compose')
|
|
|
|
|
|
|
|
###
|
|
|
|
Opts must be an object with the following keys:
|
|
|
|
|
|
|
|
app: the application instance to deploy to
|
|
|
|
image: the image to deploy; optional
|
|
|
|
shouldPerformBuild
|
|
|
|
shouldUploadLogs
|
|
|
|
buildEmulated
|
|
|
|
buildOpts: arguments to forward to docker build command
|
|
|
|
###
|
|
|
|
deployProject = (docker, logger, composeOpts, opts) ->
|
|
|
|
_ = require('lodash')
|
|
|
|
doodles = require('resin-doodles')
|
2018-10-19 16:38:50 +02:00
|
|
|
sdk = require('balena-sdk').fromSharedOptions()
|
2017-12-12 00:37:56 +02:00
|
|
|
|
|
|
|
compose.loadProject(
|
|
|
|
logger
|
|
|
|
composeOpts.projectPath
|
|
|
|
composeOpts.projectName
|
|
|
|
opts.image
|
|
|
|
)
|
|
|
|
.then (project) ->
|
2018-03-19 20:58:57 +02:00
|
|
|
if project.descriptors.length > 1 and !opts.app.application_type?[0]?.supports_multicontainer
|
|
|
|
throw new Error('Target application does not support multiple containers. Aborting!')
|
|
|
|
|
2017-12-12 00:37:56 +02:00
|
|
|
# find which services use images that already exist locally
|
|
|
|
Promise.map project.descriptors, (d) ->
|
|
|
|
# unconditionally build (or pull) if explicitly requested
|
|
|
|
return d if opts.shouldPerformBuild
|
|
|
|
docker.getImage(d.image.tag ? d.image).inspect()
|
|
|
|
.return(d.serviceName)
|
|
|
|
.catchReturn()
|
|
|
|
.filter (d) -> !!d
|
|
|
|
.then (servicesToSkip) ->
|
|
|
|
# multibuild takes in a composition and always attempts to
|
|
|
|
# build or pull all services. we workaround that here by
|
|
|
|
# passing a modified composition.
|
|
|
|
compositionToBuild = _.cloneDeep(project.composition)
|
|
|
|
compositionToBuild.services = _.omit(compositionToBuild.services, servicesToSkip)
|
|
|
|
if _.size(compositionToBuild.services) is 0
|
|
|
|
logger.logInfo('Everything is up to date (use --build to force a rebuild)')
|
|
|
|
return {}
|
|
|
|
compose.buildProject(
|
|
|
|
docker
|
|
|
|
logger
|
|
|
|
project.path
|
|
|
|
project.name
|
|
|
|
compositionToBuild
|
|
|
|
opts.app.arch
|
|
|
|
opts.app.device_type
|
|
|
|
opts.buildEmulated
|
|
|
|
opts.buildOpts
|
|
|
|
composeOpts.inlineLogs
|
|
|
|
)
|
|
|
|
.then (builtImages) ->
|
|
|
|
_.keyBy(builtImages, 'serviceName')
|
|
|
|
.then (builtImages) ->
|
|
|
|
project.descriptors.map (d) ->
|
|
|
|
builtImages[d.serviceName] ? {
|
|
|
|
serviceName: d.serviceName,
|
|
|
|
name: d.image.tag ? d.image
|
|
|
|
logs: 'Build skipped; image for service already exists.'
|
|
|
|
props: {}
|
|
|
|
}
|
|
|
|
.then (images) ->
|
2018-03-19 20:58:05 +02:00
|
|
|
if opts.app.application_type?[0]?.is_legacy
|
|
|
|
chalk = require('chalk')
|
|
|
|
legacyDeploy = require('../utils/deploy-legacy')
|
|
|
|
|
|
|
|
msg = chalk.yellow('Target application requires legacy deploy method.')
|
|
|
|
logger.logWarn(msg)
|
|
|
|
|
|
|
|
return Promise.join(
|
|
|
|
docker
|
|
|
|
logger
|
|
|
|
sdk.auth.getToken()
|
|
|
|
sdk.auth.whoami()
|
2018-10-19 16:38:50 +02:00
|
|
|
sdk.settings.get('balenaUrl')
|
2018-03-19 20:58:05 +02:00
|
|
|
{
|
|
|
|
appName: opts.app.app_name
|
|
|
|
imageName: images[0].name
|
|
|
|
buildLogs: images[0].logs
|
|
|
|
shouldUploadLogs: opts.shouldUploadLogs
|
|
|
|
}
|
|
|
|
legacyDeploy
|
|
|
|
)
|
|
|
|
.then (releaseId) ->
|
2018-03-20 14:05:14 +02:00
|
|
|
sdk.models.release.get(releaseId, $select: [ 'commit' ])
|
2017-12-12 00:37:56 +02:00
|
|
|
Promise.join(
|
|
|
|
sdk.auth.getUserId()
|
|
|
|
sdk.auth.getToken()
|
|
|
|
sdk.settings.get('apiUrl')
|
|
|
|
(userId, auth, apiEndpoint) ->
|
|
|
|
compose.deployProject(
|
|
|
|
docker
|
|
|
|
logger
|
|
|
|
project.composition
|
|
|
|
images
|
|
|
|
opts.app.id
|
|
|
|
userId
|
|
|
|
"Bearer #{auth}"
|
|
|
|
apiEndpoint
|
|
|
|
!opts.shouldUploadLogs
|
|
|
|
)
|
|
|
|
)
|
|
|
|
.then (release) ->
|
|
|
|
logger.logSuccess('Deploy succeeded!')
|
|
|
|
logger.logSuccess("Release: #{release.commit}")
|
|
|
|
console.log()
|
|
|
|
console.log(doodles.getDoodle()) # Show charlie
|
|
|
|
console.log()
|
|
|
|
.tapCatch (e) ->
|
|
|
|
logger.logError('Deploy failed')
|
2017-03-29 12:03:40 +01:00
|
|
|
|
|
|
|
module.exports =
|
|
|
|
signature: 'deploy <appName> [image]'
|
2018-10-19 16:38:50 +02:00
|
|
|
description: 'Deploy a single image or a multicontainer project to a balena application'
|
2017-03-29 12:03:40 +01:00
|
|
|
help: '''
|
2017-12-12 00:37:56 +02:00
|
|
|
Use this command to deploy an image or a complete multicontainer project
|
|
|
|
to an application, optionally building it first.
|
2017-03-29 12:03:40 +01:00
|
|
|
|
2017-10-09 19:10:50 +02:00
|
|
|
Usage: `deploy <appName> ([image] | --build [--source build-dir])`
|
|
|
|
|
2017-12-12 00:37:56 +02:00
|
|
|
Unless an image is specified, this command will look into the current directory
|
|
|
|
(or the one specified by --source) for a compose file. If one is found, this
|
|
|
|
command will deploy each service defined in the compose file, building it first
|
|
|
|
if an image for it doesn't exist. If a compose file isn't found, the command
|
|
|
|
will look for a Dockerfile, and if yet that isn't found, it will try to
|
|
|
|
generate one.
|
|
|
|
|
2017-10-09 19:10:50 +02:00
|
|
|
To deploy to an app on which you're a collaborator, use
|
2018-10-19 16:38:50 +02:00
|
|
|
`balena deploy <appOwnerUsername>/<appName>`.
|
2017-03-29 12:03:40 +01:00
|
|
|
|
2018-10-19 16:38:50 +02:00
|
|
|
Note: If building with this command, all options supported by `balena build`
|
2017-04-26 13:34:40 +01:00
|
|
|
are also supported with this command.
|
2017-03-29 12:03:40 +01:00
|
|
|
|
|
|
|
Examples:
|
2017-12-12 00:37:56 +02:00
|
|
|
|
2018-10-19 16:38:50 +02:00
|
|
|
$ balena deploy myApp
|
|
|
|
$ balena deploy myApp --build --source myBuildDir/
|
|
|
|
$ balena deploy myApp myApp/myImage
|
2017-03-29 12:03:40 +01:00
|
|
|
'''
|
|
|
|
permission: 'user'
|
2017-12-12 00:37:56 +02:00
|
|
|
primary: true
|
|
|
|
options: dockerUtils.appendOptions compose.appendOptions [
|
2017-03-29 12:03:40 +01:00
|
|
|
{
|
|
|
|
signature: 'source'
|
|
|
|
parameter: 'source'
|
2017-12-12 00:37:56 +02:00
|
|
|
description: 'Specify an alternate source directory; default is the working directory'
|
2017-03-29 12:03:40 +01:00
|
|
|
alias: 's'
|
2017-04-24 11:06:53 +01:00
|
|
|
},
|
2017-12-12 00:37:56 +02:00
|
|
|
{
|
|
|
|
signature: 'build'
|
|
|
|
boolean: true
|
|
|
|
description: 'Force a rebuild before deploy'
|
|
|
|
alias: 'b'
|
|
|
|
},
|
2017-04-24 11:06:53 +01:00
|
|
|
{
|
|
|
|
signature: 'nologupload'
|
|
|
|
description: "Don't upload build logs to the dashboard with image (if building)"
|
|
|
|
boolean: true
|
2017-03-29 12:03:40 +01:00
|
|
|
}
|
|
|
|
]
|
|
|
|
action: (params, options, done) ->
|
2017-12-12 00:37:56 +02:00
|
|
|
# compositions with many services trigger misleading warnings
|
|
|
|
require('events').defaultMaxListeners = 1000
|
2017-03-29 12:03:40 +01:00
|
|
|
|
2017-12-12 00:37:56 +02:00
|
|
|
helpers = require('../utils/helpers')
|
2017-06-28 18:30:37 +02:00
|
|
|
Logger = require('../utils/logger')
|
2017-12-12 00:37:56 +02:00
|
|
|
|
2017-06-28 18:30:37 +02:00
|
|
|
logger = new Logger()
|
2017-04-24 13:48:43 +01:00
|
|
|
|
2017-12-12 00:37:56 +02:00
|
|
|
logger.logDebug('Parsing input...')
|
|
|
|
|
|
|
|
Promise.try ->
|
|
|
|
{ appName, image } = params
|
|
|
|
|
2018-10-19 16:38:50 +02:00
|
|
|
# look into "balena build" options if appName isn't given
|
2017-12-12 00:37:56 +02:00
|
|
|
appName = options.application if not appName?
|
|
|
|
delete options.application
|
|
|
|
|
|
|
|
if not appName?
|
|
|
|
throw new Error('Please specify the name of the application to deploy')
|
|
|
|
|
|
|
|
if image? and options.build
|
|
|
|
throw new Error('Build option is not applicable when specifying an image')
|
|
|
|
|
|
|
|
Promise.join(
|
|
|
|
helpers.getApplication(appName)
|
|
|
|
helpers.getArchAndDeviceType(appName)
|
|
|
|
(app, { arch, device_type }) ->
|
|
|
|
app.arch = arch
|
|
|
|
app.device_type = device_type
|
|
|
|
return app
|
|
|
|
)
|
|
|
|
.then (app) ->
|
|
|
|
[ app, image, !!options.build, !options.nologupload ]
|
|
|
|
|
|
|
|
.then ([ app, image, shouldPerformBuild, shouldUploadLogs ]) ->
|
|
|
|
Promise.join(
|
|
|
|
dockerUtils.getDocker(options)
|
|
|
|
dockerUtils.generateBuildOpts(options)
|
|
|
|
compose.generateOpts(options)
|
|
|
|
(docker, buildOpts, composeOpts) ->
|
|
|
|
deployProject(docker, logger, composeOpts, {
|
|
|
|
app
|
|
|
|
image
|
|
|
|
shouldPerformBuild
|
|
|
|
shouldUploadLogs
|
|
|
|
buildEmulated: !!options.emulated
|
|
|
|
buildOpts
|
|
|
|
})
|
|
|
|
)
|
|
|
|
.asCallback(done)
|