mirror of
https://github.com/balena-io/balena-cli.git
synced 2025-01-26 22:29:22 +00:00
952d74207d
Signed-off-by: Thodoris Greasidis <thodoris@balena.io>
190 lines
5.1 KiB
TypeScript
190 lines
5.1 KiB
TypeScript
/*
|
|
Copyright 2016-2017 Balena
|
|
|
|
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 Bluebird = require('bluebird');
|
|
import _ = require('lodash');
|
|
import chalk from 'chalk';
|
|
import rindle = require('rindle');
|
|
import visuals = require('resin-cli-visuals');
|
|
import BalenaSdk = require('balena-sdk');
|
|
|
|
import { InitializeEmitter, OperationState } from 'balena-device-init';
|
|
|
|
const waitStreamAsync = Bluebird.promisify(rindle.wait);
|
|
|
|
const balena = BalenaSdk.fromSharedOptions();
|
|
|
|
export function getGroupDefaults(group: {
|
|
options: { name: string; default?: string }[];
|
|
}): { [name: string]: string | undefined } {
|
|
return _.chain(group)
|
|
.get('options')
|
|
.map(question => [question.name, question.default])
|
|
.fromPairs()
|
|
.value();
|
|
}
|
|
|
|
export function stateToString(state: OperationState) {
|
|
const percentage = _.padStart(`${state.percentage}`, 3, '0');
|
|
const result = `${chalk.blue(percentage + '%')} ${chalk.cyan(
|
|
state.operation.command,
|
|
)}`;
|
|
|
|
switch (state.operation.command) {
|
|
case 'copy':
|
|
return `${result} ${state.operation.from.path} -> ${
|
|
state.operation.to.path
|
|
}`;
|
|
case 'replace':
|
|
return `${result} ${state.operation.file.path}, ${
|
|
state.operation.copy
|
|
} -> ${state.operation.replace}`;
|
|
case 'run-script':
|
|
return `${result} ${state.operation.script}`;
|
|
default:
|
|
throw new Error(`Unsupported operation: ${state.operation.command}`);
|
|
}
|
|
}
|
|
|
|
export function sudo(
|
|
command: string[],
|
|
{ stderr, msg }: { stderr?: NodeJS.WritableStream; msg?: string } = {},
|
|
) {
|
|
const { executeWithPrivileges } = require('./sudo');
|
|
|
|
if (os.platform() !== 'win32') {
|
|
console.log(
|
|
msg || 'If asked please type your computer password to continue',
|
|
);
|
|
}
|
|
|
|
return executeWithPrivileges(command, stderr);
|
|
}
|
|
|
|
export function runCommand(command: string): Bluebird<void> {
|
|
const capitano = require('capitano');
|
|
return Bluebird.fromCallback(resolver => capitano.run(command, resolver));
|
|
}
|
|
|
|
export async function getManifest(
|
|
image: string,
|
|
deviceType: string,
|
|
): Promise<BalenaSdk.DeviceType> {
|
|
const init = await import('balena-device-init');
|
|
const manifest = await init.getImageManifest(image);
|
|
if (manifest != null) {
|
|
return manifest;
|
|
}
|
|
return balena.models.device.getManifestBySlug(deviceType);
|
|
}
|
|
|
|
export const areDeviceTypesCompatible = (
|
|
deviceTypeA: BalenaSdk.DeviceType,
|
|
deviceTypeB: BalenaSdk.DeviceType,
|
|
) =>
|
|
deviceTypeA.arch === deviceTypeB.arch &&
|
|
!!deviceTypeA.isDependent === !!deviceTypeB.isDependent;
|
|
|
|
export async function getOsVersion(
|
|
image: string,
|
|
manifest: BalenaSdk.DeviceType,
|
|
): Promise<string | null> {
|
|
const init = await import('balena-device-init');
|
|
return init.getImageOsVersion(image, manifest);
|
|
}
|
|
|
|
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) {
|
|
if (state.operation.command === 'burn') {
|
|
return;
|
|
}
|
|
console.log(exports.stateToString(state));
|
|
});
|
|
|
|
const progressBars = {
|
|
write: new visuals.Progress('Writing Device OS'),
|
|
check: new visuals.Progress('Validating Device OS'),
|
|
};
|
|
|
|
step.on('burn', state => progressBars[state.type].update(state));
|
|
|
|
return waitStreamAsync(step);
|
|
}
|
|
|
|
export function getArchAndDeviceType(
|
|
applicationName: string,
|
|
): Bluebird<{ arch: string; device_type: string }> {
|
|
return Bluebird.join(
|
|
getApplication(applicationName),
|
|
balena.models.config.getDeviceTypes(),
|
|
function(app, deviceTypes) {
|
|
const config = _.find(deviceTypes, { slug: app.device_type });
|
|
|
|
if (!config) {
|
|
throw new Error('Could not read application information!');
|
|
}
|
|
|
|
return { device_type: app.device_type, arch: config.arch };
|
|
},
|
|
);
|
|
}
|
|
|
|
export function getApplication(applicationName: string) {
|
|
// Check for an app of the form `user/application`, and send
|
|
// that off to a special handler (before importing any modules)
|
|
const match = applicationName.split('/');
|
|
|
|
const extraOptions = {
|
|
$expand: {
|
|
application_type: {
|
|
$select: ['name', 'slug', 'supports_multicontainer', 'is_legacy'],
|
|
},
|
|
},
|
|
};
|
|
|
|
if (match.length > 1) {
|
|
return balena.models.application.getAppByOwner(
|
|
match[1],
|
|
match[0],
|
|
extraOptions,
|
|
);
|
|
}
|
|
|
|
return balena.models.application.get(applicationName, extraOptions);
|
|
}
|
|
|
|
// 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',
|
|
args: ['/s', '/c', command],
|
|
};
|
|
} else {
|
|
return {
|
|
program: '/bin/sh',
|
|
args: ['-c', command],
|
|
};
|
|
}
|
|
}
|