mirror of
https://github.com/balena-io/balena-cli.git
synced 2025-01-05 13:34:08 +00:00
0ca1faba09
Change-type: minor Resolves: #1432 #1697 #1670 Signed-off-by: Scott Lowe <scott@balena.io>
423 lines
12 KiB
TypeScript
423 lines
12 KiB
TypeScript
/**
|
|
* @license
|
|
* Copyright 2019 Balena Ltd.
|
|
*
|
|
* 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 type * as BalenaSdk from 'balena-sdk';
|
|
|
|
import { ExpectedError, printErrorMessage } from '../errors';
|
|
import { getVisuals, stripIndent, getCliForm } from './lazy';
|
|
import Logger = require('./logger');
|
|
import { exec, execBuffered, getDeviceOsRelease } from './ssh';
|
|
|
|
const MIN_BALENAOS_VERSION = 'v2.14.0';
|
|
|
|
export async function join(
|
|
logger: Logger,
|
|
sdk: BalenaSdk.BalenaSDK,
|
|
deviceHostnameOrIp?: string,
|
|
appName?: string,
|
|
appUpdatePollInterval?: number,
|
|
): Promise<void> {
|
|
logger.logDebug('Determining device...');
|
|
const deviceIp = await getOrSelectLocalDevice(deviceHostnameOrIp);
|
|
await assertDeviceIsCompatible(deviceIp);
|
|
logger.logDebug(`Using device: ${deviceIp}`);
|
|
|
|
logger.logDebug('Determining device type...');
|
|
const deviceType = await getDeviceType(deviceIp);
|
|
logger.logDebug(`Device type: ${deviceType}`);
|
|
|
|
logger.logDebug('Determining application...');
|
|
const app = await getOrSelectApplication(sdk, deviceType, appName);
|
|
logger.logDebug(
|
|
`Using application: ${app.app_name} (${app.is_for__device_type[0].slug})`,
|
|
);
|
|
if (app.is_for__device_type[0].slug !== deviceType) {
|
|
logger.logDebug(`Forcing device type to: ${deviceType}`);
|
|
app.is_for__device_type[0].slug = deviceType;
|
|
}
|
|
|
|
logger.logDebug('Determining device OS version...');
|
|
const deviceOsVersion = await getOsVersion(deviceIp);
|
|
logger.logDebug(`Device OS version: ${deviceOsVersion}`);
|
|
|
|
logger.logDebug('Generating application config...');
|
|
const config = await generateApplicationConfig(sdk, app, {
|
|
version: deviceOsVersion,
|
|
appUpdatePollInterval,
|
|
});
|
|
logger.logDebug(`Using config: ${JSON.stringify(config, null, 2)}`);
|
|
|
|
logger.logDebug('Configuring...');
|
|
await configure(deviceIp, config);
|
|
logger.logDebug('All done.');
|
|
|
|
const platformUrl = await sdk.settings.get('balenaUrl');
|
|
logger.logSuccess(`Device successfully joined ${platformUrl}!`);
|
|
}
|
|
|
|
export async function leave(
|
|
logger: Logger,
|
|
_sdk: BalenaSdk.BalenaSDK,
|
|
deviceHostnameOrIp?: string,
|
|
): Promise<void> {
|
|
logger.logDebug('Determining device...');
|
|
const deviceIp = await getOrSelectLocalDevice(deviceHostnameOrIp);
|
|
await assertDeviceIsCompatible(deviceIp);
|
|
logger.logDebug(`Using device: ${deviceIp}`);
|
|
|
|
logger.logDebug('Deconfiguring...');
|
|
await deconfigure(deviceIp);
|
|
logger.logDebug('All done.');
|
|
|
|
logger.logSuccess('Device successfully left the platform.');
|
|
}
|
|
|
|
async function execCommand(
|
|
deviceIp: string,
|
|
cmd: string,
|
|
msg: string,
|
|
): Promise<void> {
|
|
const through = await import('through2');
|
|
const visuals = getVisuals();
|
|
|
|
const spinner = new visuals.Spinner(`[${deviceIp}] Connecting...`);
|
|
const innerSpinner = spinner.spinner;
|
|
|
|
const stream = through(function (data, _enc, cb) {
|
|
innerSpinner.setSpinnerTitle(`%s [${deviceIp}] ${msg}`);
|
|
cb(null, data);
|
|
});
|
|
|
|
spinner.start();
|
|
await exec(deviceIp, cmd, stream);
|
|
spinner.stop();
|
|
}
|
|
|
|
async function configure(deviceIp: string, config: any): Promise<void> {
|
|
// Passing the JSON is slightly tricky due to the many layers of indirection
|
|
// so we just base64-encode it here and decode it at the other end, when invoking
|
|
// os-config.
|
|
const json = JSON.stringify(config);
|
|
const b64 = Buffer.from(json).toString('base64');
|
|
const str = `"$(base64 -d <<< ${b64})"`;
|
|
await execCommand(deviceIp, `os-config join ${str}`, 'Configuring...');
|
|
}
|
|
|
|
async function deconfigure(deviceIp: string): Promise<void> {
|
|
await execCommand(deviceIp, 'os-config leave', 'Configuring...');
|
|
}
|
|
|
|
async function assertDeviceIsCompatible(deviceIp: string): Promise<void> {
|
|
const cmd = 'os-config --version';
|
|
try {
|
|
await execBuffered(deviceIp, cmd);
|
|
} catch (err) {
|
|
if (err instanceof ExpectedError) {
|
|
throw err;
|
|
}
|
|
console.error(`${err}\n`);
|
|
throw new ExpectedError(stripIndent`
|
|
Failed to execute "${cmd}" on device "${deviceIp}".
|
|
Depending on more specific error messages above, this may mean that the device
|
|
is incompatible. Please ensure that the device is running a balenaOS release
|
|
newer than ${MIN_BALENAOS_VERSION}.`);
|
|
}
|
|
}
|
|
|
|
async function getDeviceType(deviceIp: string): Promise<string> {
|
|
const output = await getDeviceOsRelease(deviceIp);
|
|
const match = /^SLUG="([^"]+)"$/m.exec(output);
|
|
if (!match) {
|
|
throw new Error('Failed to determine device type');
|
|
}
|
|
return match[1];
|
|
}
|
|
|
|
async function getOsVersion(deviceIp: string): Promise<string> {
|
|
const output = await getDeviceOsRelease(deviceIp);
|
|
const match = /^VERSION_ID="([^"]+)"$/m.exec(output);
|
|
if (!match) {
|
|
throw new Error('Failed to determine OS version ID');
|
|
}
|
|
return match[1];
|
|
}
|
|
|
|
async function getOrSelectLocalDevice(deviceIp?: string): Promise<string> {
|
|
if (deviceIp) {
|
|
return deviceIp;
|
|
}
|
|
|
|
const through = await import('through2');
|
|
|
|
let ip: string | null = null;
|
|
const stream = through(function (data, _enc, cb) {
|
|
const match = /^==> Selected device: (.*)$/m.exec(data.toString());
|
|
if (match) {
|
|
ip = match[1];
|
|
cb();
|
|
} else {
|
|
cb(null, data);
|
|
}
|
|
});
|
|
|
|
stream.pipe(process.stderr);
|
|
|
|
const { sudo } = await import('../utils/helpers');
|
|
const command = ['internal', 'scandevices'];
|
|
await sudo(command, {
|
|
stderr: stream,
|
|
msg:
|
|
'Scanning for local devices. If asked, please type your computer password.',
|
|
});
|
|
|
|
if (!ip) {
|
|
throw new ExpectedError('No device selected');
|
|
}
|
|
|
|
return ip;
|
|
}
|
|
|
|
async function selectAppFromList(
|
|
applications: ApplicationWithDeviceType[],
|
|
): Promise<ApplicationWithDeviceType> {
|
|
const _ = await import('lodash');
|
|
const { selectFromList } = await import('../utils/patterns');
|
|
|
|
// Present a list to the user which shows the fully qualified application
|
|
// name (user/appname) and allows them to select.
|
|
return selectFromList(
|
|
'Select application',
|
|
_.map(applications, (app) => {
|
|
return { name: app.slug, ...app };
|
|
}),
|
|
);
|
|
}
|
|
|
|
async function getOrSelectApplication(
|
|
sdk: BalenaSdk.BalenaSDK,
|
|
deviceType: string,
|
|
appName?: string,
|
|
): Promise<ApplicationWithDeviceType> {
|
|
const _ = await import('lodash');
|
|
|
|
const allDeviceTypes = await sdk.models.config.getDeviceTypes();
|
|
const deviceTypeManifest = _.find(allDeviceTypes, { slug: deviceType });
|
|
if (!deviceTypeManifest) {
|
|
throw new ExpectedError(`"${deviceType}" is not a valid device type`);
|
|
}
|
|
const compatibleDeviceTypes = _(allDeviceTypes)
|
|
.filter(
|
|
(dt) =>
|
|
sdk.models.os.isArchitectureCompatibleWith(
|
|
deviceTypeManifest.arch,
|
|
dt.arch,
|
|
) &&
|
|
!!dt.isDependent === !!deviceTypeManifest.isDependent &&
|
|
dt.state !== 'DISCONTINUED',
|
|
)
|
|
.map((type) => type.slug)
|
|
.value();
|
|
|
|
if (!appName) {
|
|
return createOrSelectAppOrExit(sdk, compatibleDeviceTypes, deviceType);
|
|
}
|
|
|
|
const options: BalenaSdk.PineOptions<BalenaSdk.Application> = {
|
|
$expand: {
|
|
is_for__device_type: { $select: 'slug' },
|
|
},
|
|
};
|
|
|
|
// Check for an app of the form `user/application` and update the API query.
|
|
let name: string;
|
|
const match = appName.split('/');
|
|
if (match.length > 1) {
|
|
// These will match at most one app
|
|
options.$filter = { slug: appName.toLowerCase() };
|
|
name = match[1];
|
|
} else {
|
|
// We're given an application; resolve it if it's ambiguous and also validate
|
|
// it's of appropriate device type.
|
|
options.$filter = { app_name: appName };
|
|
name = appName;
|
|
}
|
|
|
|
const applications = (await sdk.models.application.getAll(
|
|
options,
|
|
)) as ApplicationWithDeviceType[];
|
|
|
|
if (applications.length === 0) {
|
|
const shouldCreateApp = await getCliForm().ask({
|
|
message:
|
|
`No application found with name "${appName}".\n` +
|
|
'Would you like to create it now?',
|
|
type: 'confirm',
|
|
default: true,
|
|
});
|
|
if (shouldCreateApp) {
|
|
return createApplication(sdk, deviceType, name);
|
|
}
|
|
process.exit(1);
|
|
}
|
|
|
|
// We've found at least one app with the given name.
|
|
// Filter out apps for non-matching device types and see what we're left with.
|
|
const validApplications = applications.filter((app) =>
|
|
_.includes(compatibleDeviceTypes, app.is_for__device_type[0].slug),
|
|
);
|
|
|
|
if (validApplications.length === 0) {
|
|
throw new ExpectedError('No application found with a matching device type');
|
|
}
|
|
|
|
if (validApplications.length === 1) {
|
|
return validApplications[0];
|
|
}
|
|
|
|
return selectAppFromList(applications);
|
|
}
|
|
|
|
// TODO: revisit this function's purpose. It was refactored out of
|
|
// `getOrSelectApplication` above in order to satisfy some resin-lint v3
|
|
// rules, but it looks like there's a fair amount of duplicate logic.
|
|
async function createOrSelectAppOrExit(
|
|
sdk: BalenaSdk.BalenaSDK,
|
|
compatibleDeviceTypes: string[],
|
|
deviceType: string,
|
|
): Promise<ApplicationWithDeviceType> {
|
|
// No application specified, show a list to select one.
|
|
const applications = (await sdk.models.application.getAll({
|
|
$expand: { is_for__device_type: { $select: 'slug' } },
|
|
$filter: {
|
|
is_for__device_type: {
|
|
$any: {
|
|
$alias: 'dt',
|
|
$expr: { dt: { slug: { $in: compatibleDeviceTypes } } },
|
|
},
|
|
},
|
|
},
|
|
})) as ApplicationWithDeviceType[];
|
|
|
|
if (applications.length === 0) {
|
|
const shouldCreateApp = await getCliForm().ask({
|
|
message:
|
|
'You have no applications this device can join.\n' +
|
|
'Would you like to create one now?',
|
|
type: 'confirm',
|
|
default: true,
|
|
});
|
|
if (shouldCreateApp) {
|
|
return createApplication(sdk, deviceType);
|
|
}
|
|
process.exit(1);
|
|
}
|
|
|
|
return selectAppFromList(applications);
|
|
}
|
|
|
|
async function createApplication(
|
|
sdk: BalenaSdk.BalenaSDK,
|
|
deviceType: string,
|
|
name?: string,
|
|
): Promise<ApplicationWithDeviceType> {
|
|
const validation = await import('./validation');
|
|
|
|
const username = await sdk.auth.whoami();
|
|
if (!username) {
|
|
throw new sdk.errors.BalenaNotLoggedIn();
|
|
}
|
|
|
|
const applicationName = await new Promise<string>(async (resolve, reject) => {
|
|
while (true) {
|
|
try {
|
|
const appName = await getCliForm().ask({
|
|
message: 'Enter a name for your new application:',
|
|
type: 'input',
|
|
default: name,
|
|
validate: validation.validateApplicationName,
|
|
});
|
|
|
|
try {
|
|
await sdk.models.application.get(appName, {
|
|
$filter: {
|
|
$or: [
|
|
{ slug: { $startswith: `${username!.toLowerCase()}/` } },
|
|
{ $not: { slug: { $contains: '/' } } },
|
|
],
|
|
},
|
|
});
|
|
// TODO: This is the only example in the codebase where `printErrorMessage()`
|
|
// is called directly. Consider refactoring.
|
|
printErrorMessage(
|
|
'You already have an application with that name; please choose another.',
|
|
);
|
|
continue;
|
|
} catch (err) {
|
|
return resolve(appName);
|
|
}
|
|
} catch (err) {
|
|
return reject(err);
|
|
}
|
|
}
|
|
});
|
|
|
|
const app = await sdk.models.application.create({
|
|
name: applicationName,
|
|
deviceType,
|
|
organization: username,
|
|
});
|
|
return (await sdk.models.application.get(app.id, {
|
|
$expand: {
|
|
is_for__device_type: { $select: 'slug' },
|
|
},
|
|
})) as ApplicationWithDeviceType;
|
|
}
|
|
|
|
async function generateApplicationConfig(
|
|
sdk: BalenaSdk.BalenaSDK,
|
|
app: ApplicationWithDeviceType,
|
|
options: {
|
|
version: string;
|
|
appUpdatePollInterval?: number;
|
|
},
|
|
) {
|
|
const { generateApplicationConfig: configGen } = await import('./config');
|
|
|
|
const manifest = await sdk.models.device.getManifestBySlug(
|
|
app.is_for__device_type[0].slug,
|
|
);
|
|
const opts =
|
|
manifest.options &&
|
|
manifest.options.filter((opt) => opt.name !== 'network');
|
|
|
|
const override = {
|
|
appUpdatePollInterval: options.appUpdatePollInterval,
|
|
};
|
|
|
|
const values = {
|
|
...(opts ? await getCliForm().run(opts, { override }) : {}),
|
|
...options,
|
|
};
|
|
|
|
const config = await configGen(app, values);
|
|
if (config.connectivity === 'connman') {
|
|
delete config.connectivity;
|
|
delete config.files;
|
|
}
|
|
|
|
return config;
|
|
}
|