2017-12-20 22:46:01 +01:00
|
|
|
/*
|
|
|
|
Copyright 2016-2017 Resin.io
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import os = require('os');
|
|
|
|
import Promise = require('bluebird');
|
|
|
|
import _ = require('lodash');
|
|
|
|
import chalk from 'chalk';
|
|
|
|
import rindle = require('rindle');
|
|
|
|
import imagefs = require('resin-image-fs');
|
|
|
|
import visuals = require('resin-cli-visuals');
|
|
|
|
import ResinSdk = require('resin-sdk');
|
|
|
|
|
|
|
|
import { execute } from 'president';
|
2018-01-04 14:07:55 +00:00
|
|
|
import { InitializeEmitter, OperationState } from 'resin-device-init';
|
2017-12-20 22:46:01 +01:00
|
|
|
|
2018-01-04 16:17:43 +00:00
|
|
|
const extractStreamAsync = Promise.promisify(rindle.extract);
|
|
|
|
const waitStreamAsync = Promise.promisify(rindle.wait);
|
|
|
|
const presidentExecuteAsync = Promise.promisify(execute);
|
|
|
|
|
2017-12-20 22:46:01 +01:00
|
|
|
const resin = ResinSdk.fromSharedOptions();
|
|
|
|
|
2018-01-09 16:05:24 +01:00
|
|
|
export function getGroupDefaults(group: {
|
|
|
|
options: { name: string; default?: string }[];
|
|
|
|
}): { [name: string]: string | undefined } {
|
2017-12-20 22:46:01 +01:00
|
|
|
return _.chain(group)
|
|
|
|
.get('options')
|
2018-01-09 16:05:24 +01:00
|
|
|
.map(question => [question.name, question.default])
|
2017-12-20 22:46:01 +01:00
|
|
|
.fromPairs()
|
|
|
|
.value();
|
2018-01-04 14:07:55 +00:00
|
|
|
}
|
2017-12-20 22:46:01 +01:00
|
|
|
|
|
|
|
export function stateToString(state: OperationState) {
|
2018-01-04 16:17:43 +00:00
|
|
|
const percentage = _.padStart(`${state.percentage}`, 3, '0');
|
2018-01-09 16:05:24 +01:00
|
|
|
const result = `${chalk.blue(percentage + '%')} ${chalk.cyan(
|
|
|
|
state.operation.command,
|
|
|
|
)}`;
|
2017-12-20 22:46:01 +01:00
|
|
|
|
|
|
|
switch (state.operation.command) {
|
|
|
|
case 'copy':
|
2018-01-09 16:05:24 +01:00
|
|
|
return `${result} ${state.operation.from.path} -> ${
|
|
|
|
state.operation.to.path
|
|
|
|
}`;
|
2017-12-20 22:46:01 +01:00
|
|
|
case 'replace':
|
2018-01-09 16:05:24 +01:00
|
|
|
return `${result} ${state.operation.file.path}, ${
|
|
|
|
state.operation.copy
|
|
|
|
} -> ${state.operation.replace}`;
|
2017-12-20 22:46:01 +01:00
|
|
|
case 'run-script':
|
|
|
|
return `${result} ${state.operation.script}`;
|
|
|
|
default:
|
|
|
|
throw new Error(`Unsupported operation: ${state.operation.command}`);
|
|
|
|
}
|
2018-01-04 14:07:55 +00:00
|
|
|
}
|
2017-12-20 22:46:01 +01:00
|
|
|
|
|
|
|
export function sudo(command: string[]) {
|
|
|
|
if (os.platform() !== 'win32') {
|
|
|
|
console.log('If asked please type your computer password to continue');
|
|
|
|
}
|
|
|
|
|
|
|
|
command = _.union(_.take(process.argv, 2), command);
|
|
|
|
|
|
|
|
return presidentExecuteAsync(command);
|
2018-01-04 14:07:55 +00:00
|
|
|
}
|
2017-12-20 22:46:01 +01:00
|
|
|
|
2018-01-09 16:05:24 +01:00
|
|
|
export function getManifest(
|
|
|
|
image: string,
|
|
|
|
deviceType: string,
|
|
|
|
): Promise<ResinSdk.DeviceType> {
|
2017-12-20 22:46:01 +01:00
|
|
|
// Attempt to read manifest from the first
|
|
|
|
// partition, but fallback to the API if
|
|
|
|
// we encounter any errors along the way.
|
2018-01-09 16:05:24 +01:00
|
|
|
return imagefs
|
|
|
|
.read({
|
|
|
|
image,
|
|
|
|
partition: {
|
|
|
|
primary: 1,
|
|
|
|
},
|
|
|
|
path: '/device-type.json',
|
|
|
|
})
|
|
|
|
.then(extractStreamAsync)
|
|
|
|
.then(JSON.parse)
|
|
|
|
.catch(() => resin.models.device.getManifestBySlug(deviceType));
|
2018-01-04 14:07:55 +00:00
|
|
|
}
|
2017-12-20 22:46:01 +01:00
|
|
|
|
|
|
|
export function osProgressHandler(step: InitializeEmitter) {
|
|
|
|
step.on('stdout', process.stdout.write.bind(process.stdout));
|
|
|
|
step.on('stderr', process.stderr.write.bind(process.stderr));
|
|
|
|
|
|
|
|
step.on('state', function(state) {
|
2018-01-09 16:05:24 +01:00
|
|
|
if (state.operation.command === 'burn') {
|
|
|
|
return;
|
|
|
|
}
|
2018-01-04 16:17:43 +00:00
|
|
|
console.log(exports.stateToString(state));
|
2017-12-20 22:46:01 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
const progressBars = {
|
|
|
|
write: new visuals.Progress('Writing Device OS'),
|
2018-01-04 14:07:55 +00:00
|
|
|
check: new visuals.Progress('Validating Device OS'),
|
2017-12-20 22:46:01 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
step.on('burn', state => progressBars[state.type].update(state));
|
|
|
|
|
2018-01-04 16:17:43 +00:00
|
|
|
return waitStreamAsync(step);
|
2018-01-04 14:07:55 +00:00
|
|
|
}
|
2017-12-20 22:46:01 +01:00
|
|
|
|
2018-01-09 16:05:24 +01:00
|
|
|
export function getArchAndDeviceType(
|
|
|
|
applicationName: string,
|
|
|
|
): Promise<{ arch: string; device_type: string }> {
|
2017-12-20 22:46:01 +01:00
|
|
|
return Promise.join(
|
|
|
|
getApplication(applicationName),
|
|
|
|
resin.models.config.getDeviceTypes(),
|
2018-01-09 16:05:24 +01:00
|
|
|
function(app, deviceTypes) {
|
2018-01-04 16:17:43 +00:00
|
|
|
const config = _.find(deviceTypes, { slug: app.device_type });
|
2017-12-20 22:46:01 +01:00
|
|
|
|
2018-01-04 16:17:43 +00:00
|
|
|
if (!config) {
|
2017-12-20 22:46:01 +01:00
|
|
|
throw new Error('Could not read application information!');
|
|
|
|
}
|
|
|
|
|
|
|
|
return { device_type: app.device_type, arch: config.arch };
|
2018-01-04 14:07:55 +00:00
|
|
|
},
|
2017-12-20 22:46:01 +01:00
|
|
|
);
|
2018-01-04 14:07:55 +00:00
|
|
|
}
|
2017-12-20 22:46:01 +01:00
|
|
|
|
2017-12-12 00:37:56 +02:00
|
|
|
export function getApplication(applicationName: string) {
|
2017-12-20 22:46:01 +01:00
|
|
|
// Check for an app of the form `user/application`, and send
|
2018-01-04 16:17:43 +00:00
|
|
|
// that off to a special handler (before importing any modules)
|
|
|
|
const match = /(\w+)\/(\w+)/.exec(applicationName);
|
|
|
|
|
2018-03-19 20:58:05 +02:00
|
|
|
const extraOptions = {
|
|
|
|
$expand: {
|
|
|
|
application_type: {
|
2018-03-20 13:06:44 +02:00
|
|
|
$select: ['name', 'slug', 'supports_multicontainer', 'is_legacy'],
|
2018-03-19 20:58:05 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2018-01-04 16:17:43 +00:00
|
|
|
if (match) {
|
2018-03-20 13:06:44 +02:00
|
|
|
return resin.models.application.getAppByOwner(
|
|
|
|
match[2],
|
|
|
|
match[1],
|
|
|
|
extraOptions,
|
|
|
|
);
|
2017-12-20 22:46:01 +01:00
|
|
|
}
|
|
|
|
|
2018-03-19 20:58:05 +02:00
|
|
|
return resin.models.application.get(applicationName, extraOptions);
|
2018-01-04 14:07:55 +00:00
|
|
|
}
|
2017-12-20 22:46:01 +01:00
|
|
|
|
|
|
|
// A function to reliably execute a command
|
|
|
|
// in all supported operating systems, including
|
|
|
|
// different Windows environments like `cmd.exe`
|
|
|
|
// and `Cygwin`.
|
|
|
|
export function getSubShellCommand(command: string) {
|
|
|
|
if (os.platform() === 'win32') {
|
|
|
|
return {
|
|
|
|
program: 'cmd.exe',
|
2018-01-09 16:05:24 +01:00
|
|
|
args: ['/s', '/c', command],
|
2017-12-20 22:46:01 +01:00
|
|
|
};
|
|
|
|
} else {
|
|
|
|
return {
|
|
|
|
program: '/bin/sh',
|
2018-01-09 16:05:24 +01:00
|
|
|
args: ['-c', command],
|
2017-12-20 22:46:01 +01:00
|
|
|
};
|
|
|
|
}
|
2018-01-04 14:07:55 +00:00
|
|
|
}
|