// Generated by CoffeeScript 1.12.6 var Promise, dockerUtils, formatImageName, getBuilderLogPushEndpoint, getBuilderPushEndpoint, getBundleInfo, parseInput, performUpload, renderProgress, showPushProgress, updatePushProgress, uploadLogs, uploadToPromise; Promise = require('bluebird'); dockerUtils = require('../utils/docker'); getBuilderPushEndpoint = function(baseUrl, owner, app) { var args, querystring; querystring = require('querystring'); args = querystring.stringify({ owner: owner, app: app }); return "https://builder." + baseUrl + "/v1/push?" + args; }; getBuilderLogPushEndpoint = function(baseUrl, buildId, owner, app) { var args, querystring; querystring = require('querystring'); args = querystring.stringify({ owner: owner, app: app, buildId: buildId }); return "https://builder." + baseUrl + "/v1/pushLogs?" + args; }; formatImageName = function(image) { return image.split('/').pop(); }; parseInput = Promise.method(function(params, options) { var appName, image, source; if (params.appName == null) { throw new Error('Need an application to deploy to!'); } appName = params.appName; image = void 0; if (params.image != null) { if (options.build || (options.source != null)) { 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]; }); renderProgress = function(percentage, stepCount) { var _, bar, barCount, spaceCount; if (stepCount == null) { stepCount = 50; } _ = require('lodash'); percentage = Math.max(0, Math.min(percentage, 100)); barCount = Math.floor(stepCount * percentage / 100); spaceCount = stepCount - barCount; bar = "[" + (_.repeat('=', barCount)) + ">" + (_.repeat(' ', spaceCount)) + "]"; return bar + " " + (percentage.toFixed(1)) + "%"; }; showPushProgress = function(logStreams) { var logging; logging = require('../utils/logging'); return logging.logInfo(logStreams, renderProgress(0)); }; updatePushProgress = function(percentage, logStreams) { var ansiEscapes, logging; logging = require('../utils/logging'); ansiEscapes = require('ansi-escapes'); if (percentage >= 100) { percentage = 100; } process.stdout.write(ansiEscapes.cursorUp(1)); process.stdout.clearLine(); process.stdout.cursorTo(0); return logging.logInfo(logStreams, renderProgress(percentage)); }; getBundleInfo = function(options) { var helpers; helpers = require('../utils/helpers'); return helpers.getAppInfo(options.appName).then(function(app) { return [app.arch, app.device_type]; }); }; performUpload = function(imageStream, token, username, url, appName, logStreams) { var progressStream, request, streamWithProgress, uploadRequest, zlib; request = require('request'); progressStream = require('progress-stream'); zlib = require('zlib'); showPushProgress(logStreams); streamWithProgress = imageStream.pipe(progressStream({ time: 500, length: imageStream.length }, function(arg) { var percentage; percentage = arg.percentage; return updatePushProgress(percentage, logStreams); })); uploadRequest = request.post({ url: getBuilderPushEndpoint(url, username, appName), headers: { 'Content-Encoding': 'gzip' }, auth: { bearer: token }, body: streamWithProgress.pipe(zlib.createGzip({ level: 6 })) }); return uploadToPromise(uploadRequest, logStreams); }; uploadLogs = function(logs, token, url, buildId, username, appName) { var request; request = require('request'); return request.post({ json: true, url: getBuilderLogPushEndpoint(url, buildId, username, appName), auth: { bearer: token }, body: Buffer.from(logs) }); }; uploadToPromise = function(uploadRequest, logStreams) { var logging; logging = require('../utils/logging'); return new Promise(function(resolve, reject) { var handleMessage; handleMessage = function(data) { var e, obj; data = data.toString(); logging.logDebug(logStreams, "Received data: " + data); try { obj = JSON.parse(data); } catch (error) { e = error; logging.logError(logStreams, 'Error parsing reply from remote side'); reject(e); return; } if (obj.type != null) { switch (obj.type) { case 'error': return reject(new Error("Remote error: " + obj.error)); case 'success': return resolve(obj); case 'status': return logging.logInfo(logStreams, "Remote: " + obj.message); default: return reject(new Error("Received unexpected reply from remote: " + data)); } } else { return reject(new Error("Received unexpected reply from remote: " + data)); } }; return uploadRequest.on('error', reject).on('data', handleMessage); }); }; module.exports = { signature: 'deploy [image]', description: 'Deploy an image to a resin.io application', help: 'Use this command to deploy an image to an application, optionally building it first.\n\nUsage: deploy ([image] | --build [--source build-dir])\n\nNote: If building with this command, all options supported by `resin build`\nare also supported with this command.\n\nExamples:\n $ resin deploy myApp --build --source myBuildDir/\n $ 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: function(params, options, done) { var _, logStreams, logging, logs, resin, tmp, tmpNameAsync, upload; _ = require('lodash'); tmp = require('tmp'); tmpNameAsync = Promise.promisify(tmp.tmpName); resin = require('resin-sdk-preconfigured'); logging = require('../utils/logging'); logStreams = logging.getLogStreams(); tmp.setGracefulCleanup(); logs = ''; upload = function(token, username, url) { return dockerUtils.getDocker(options).then(function(docker) { return parseInput(params, options).then(function(arg) { var appName, build, imageName, source; appName = arg[0], build = arg[1], source = arg[2], imageName = arg[3]; return tmpNameAsync().then(function(bufferFile) { options = _.assign({}, options, { appName: appName }); params = _.assign({}, params, { source: source }); return Promise["try"](function() { if (build) { return dockerUtils.runBuild(params, options, getBundleInfo, logStreams); } else { return { image: imageName, log: '' }; } }).then(function(arg1) { var buildLogs, imageName; imageName = arg1.image, buildLogs = arg1.log; logging.logInfo(logStreams, 'Initializing deploy...'); logs = buildLogs; return Promise.all([dockerUtils.bufferImage(docker, imageName, bufferFile), token, username, url, params.appName, logStreams]).spread(performUpload); })["finally"](function() { return Promise["try"](function() { return require('mz/fs').unlink(bufferFile); })["catch"](_.noop); }); }); }).tap(function(arg) { var buildId, imageName; imageName = arg.image, buildId = arg.buildId; logging.logSuccess(logStreams, "Successfully deployed image: " + (formatImageName(imageName))); return buildId; }).then(function(arg) { var buildId, imageName; imageName = arg.image, buildId = arg.buildId; if (logs === '' || (options.nologupload != null)) { return ''; } logging.logInfo(logStreams, 'Uploading logs to dashboard...'); return Promise.join(logs, token, url, buildId, username, params.appName, uploadLogs)["return"]('Successfully uploaded logs'); }).then(function(msg) { if (msg !== '') { return logging.logSuccess(logStreams, msg); } }).asCallback(done); }); }; return Promise.join(resin.auth.getToken(), resin.auth.whoami(), resin.settings.get('resinUrl'), upload); } };