Promise = require('bluebird') dockerUtils = require('../utils/docker') getBuilderPushEndpoint = (baseUrl, owner, app) -> querystring = require('querystring') args = querystring.stringify({ owner, app }) "https://builder.#{baseUrl}/v1/push?#{args}" getBuilderLogPushEndpoint = (baseUrl, buildId, owner, app) -> querystring = require('querystring') args = querystring.stringify({ owner, app, buildId }) "https://builder.#{baseUrl}/v1/pushLogs?#{args}" formatImageName = (image) -> image.split('/').pop() parseInput = Promise.method (params, options) -> if not params.appName? throw new Error('Need an application to deploy to!') appName = params.appName image = undefined if params.image? if options.build or options.source? throw new Error('Build and source parameters are not applicable when specifying an image') options.build = false image = params.image else if options.build source = options.source || '.' else throw new Error('Need either an image or a build flag!') return [appName, options.build, source, image] # Builds and returns a Docker-like progress bar like this: # [==================================> ] 64% renderProgress = (percentage, stepCount = 50) -> _ = require('lodash') percentage = Math.max(0, Math.min(percentage, 100)) barCount = stepCount * percentage // 100 spaceCount = stepCount - barCount bar = "[#{_.repeat('=', barCount)}>#{_.repeat(' ', spaceCount)}]" return "#{bar} #{percentage.toFixed(1)}%" pushProgress = (imageSize, request, logStreams, timeout = 250) -> logging = require('../utils/logging') ansiEscapes = require('ansi-escapes') logging.logInfo(logStreams, 'Initializing...') progressReporter = setInterval -> sent = request.req.connection._bytesDispatched percent = (sent / imageSize) * 100 if percent >= 100 clearInterval(progressReporter) percent = 100 process.stdout.write(ansiEscapes.cursorUp(1)) process.stdout.clearLine() process.stdout.cursorTo(0) logging.logInfo(logStreams, renderProgress(percent)) , timeout getBundleInfo = (options) -> helpers = require('../utils/helpers') helpers.getAppInfo(options.appName) .then (app) -> [app.arch, app.device_type] performUpload = (image, token, username, url, size, appName, logStreams) -> request = require('request') post = request.post url: getBuilderPushEndpoint(url, username, appName) auth: bearer: token body: image uploadToPromise(post, size, logStreams) uploadLogs = (logs, token, url, buildId, username, appName) -> request = require('request') request.post json: true url: getBuilderLogPushEndpoint(url, buildId, username, appName) auth: bearer: token body: Buffer.from(logs) uploadToPromise = (request, size, logStreams) -> logging = require('../utils/logging') new Promise (resolve, reject) -> handleMessage = (data) -> data = data.toString() logging.logDebug(logStreams, "Received data: #{data}") try obj = JSON.parse(data) catch e logging.logError(logStreams, 'Error parsing reply from remote side') reject(e) return if obj.type? switch obj.type when 'error' then reject(new Error("Remote error: #{obj.error}")) when 'success' then resolve(obj) when 'status' then logging.logInfo(logStreams, "Remote: #{obj.message}") else reject(new Error("Received unexpected reply from remote: #{data}")) else reject(new Error("Received unexpected reply from remote: #{data}")) request .on('error', reject) .on('data', handleMessage) # Set up upload reporting pushProgress(size, request, logStreams) module.exports = signature: 'deploy [image]' description: 'Deploy a container to a resin.io application' help: ''' Use this command to deploy and optionally build an image to an application. Usage: deploy ([image] | --build [--source build-dir]) Note: If building with this command, all options supported by `resin build` are also supported with this command. Examples: $ resin deploy myApp --build --source myBuildDir/ $ resin deploy myApp myApp/myImage ''' permission: 'user' options: dockerUtils.appendOptions [ { signature: 'build' boolean: true description: 'Build image then deploy' alias: 'b' }, { signature: 'source' parameter: 'source' description: 'The source directory to use when building the image' alias: 's' }, { signature: 'nologupload' description: "Don't upload build logs to the dashboard with image (if building)" boolean: true } ] action: (params, options, done) -> _ = require('lodash') tmp = require('tmp') tmpNameAsync = Promise.promisify(tmp.tmpName) resin = require('resin-sdk-preconfigured') logging = require('../utils/logging') logStreams = logging.getLogStreams() # Ensure the tmp files gets deleted tmp.setGracefulCleanup() logs = '' upload = (token, username, url) -> docker = dockerUtils.getDocker(options) # Check input parameters parseInput(params, options) .then ([appName, build, source, imageName]) -> tmpNameAsync() .then (tmpPath) -> # Setup the build args for how the build routine expects them options = _.assign({}, options, { appName }) params = _.assign({}, params, { source }) Promise.try -> if build dockerUtils.runBuild(params, options, getBundleInfo, logStreams) else { image: imageName, log: '' } .then ({ image: imageName, log: buildLogs }) -> logs = buildLogs Promise.join( dockerUtils.bufferImage(docker, imageName, tmpPath) token username url dockerUtils.getImageSize(docker, imageName) params.appName logStreams performUpload ) .finally -> # If the file was never written to (for instance because an error # has occured before any data was written) this call will throw an # ugly error, just suppress it require('mz/fs').unlink(tmpPath) .catch(_.noop) .tap ({ image: imageName, buildId }) -> logging.logSuccess(logStreams, "Successfully deployed image: #{formatImageName(imageName)}") return buildId .then ({ image: imageName, buildId }) -> if logs is '' or options.nologupload? return '' logging.logInfo(logStreams, 'Uploading logs to dashboard...') Promise.join( logs token url buildId username params.appName uploadLogs ) .return('Successfully uploaded logs') .then (msg) -> logging.logSuccess(logStreams, msg) if msg isnt '' .asCallback(done) Promise.join( resin.auth.getToken() resin.auth.whoami() resin.settings.get('resinUrl') upload )