2020-07-02 13:58:07 +00:00
|
|
|
/**
|
|
|
|
* @license
|
|
|
|
* Copyright 2016-2020 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 { flags } from '@oclif/command';
|
|
|
|
import Command from '../../command';
|
|
|
|
import * as cf from '../../utils/common-flags';
|
|
|
|
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
2021-12-19 20:50:03 +00:00
|
|
|
import { applicationIdInfo } from '../../utils/messages';
|
2020-07-02 13:58:07 +00:00
|
|
|
import { runCommand } from '../../utils/helpers';
|
|
|
|
|
|
|
|
interface FlagsDef {
|
2021-07-15 13:41:38 +00:00
|
|
|
fleet?: string;
|
2020-07-02 13:58:07 +00:00
|
|
|
yes: boolean;
|
|
|
|
advanced: boolean;
|
|
|
|
'os-version'?: string;
|
|
|
|
drive?: string;
|
|
|
|
config?: string;
|
|
|
|
help: void;
|
2021-07-20 06:42:15 +00:00
|
|
|
'provisioning-key-name'?: string;
|
2022-05-10 18:02:45 +00:00
|
|
|
'provisioning-key-expiry-date'?: string;
|
2020-07-02 13:58:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export default class DeviceInitCmd extends Command {
|
|
|
|
public static description = stripIndent`
|
2020-12-10 12:30:17 +00:00
|
|
|
Initialize a device with balenaOS.
|
2020-07-02 13:58:07 +00:00
|
|
|
|
2021-11-24 18:24:14 +00:00
|
|
|
Register a new device in the selected fleet, download the OS image for the
|
|
|
|
fleet's default device type, configure the image and write it to an SD card.
|
|
|
|
This command effectively combines several other balena CLI commands in one,
|
|
|
|
namely:
|
2020-07-02 13:58:07 +00:00
|
|
|
|
2021-11-24 18:24:14 +00:00
|
|
|
'balena device register'
|
|
|
|
'balena os download'
|
|
|
|
'balena os build-config' or 'balena config generate'
|
|
|
|
'balena os configure'
|
|
|
|
'balena os local flash'
|
|
|
|
|
|
|
|
Possible arguments for the '--fleet', '--os-version' and '--drive' options can
|
|
|
|
be listed respectively with the commands:
|
|
|
|
|
|
|
|
'balena fleets'
|
|
|
|
'balena os versions'
|
|
|
|
'balena util available-drives'
|
|
|
|
|
|
|
|
If the '--fleet' or '--drive' options are omitted, interactive menus will be
|
|
|
|
presented with values to choose from. If the '--os-version' option is omitted,
|
|
|
|
the latest released OS version for the fleet's default device type will be used.
|
2020-12-10 12:30:17 +00:00
|
|
|
|
|
|
|
${applicationIdInfo.split('\n').join('\n\t\t')}
|
2021-11-24 18:24:14 +00:00
|
|
|
|
|
|
|
Image configuration questions will be asked interactively unless a pre-configured
|
|
|
|
'config.json' file is provided with the '--config' option. The file can be
|
|
|
|
generated with the 'balena config generate' or 'balena os build-config' commands.
|
2020-12-10 12:30:17 +00:00
|
|
|
`;
|
|
|
|
|
2020-07-02 13:58:07 +00:00
|
|
|
public static examples = [
|
|
|
|
'$ balena device init',
|
2021-07-15 13:41:38 +00:00
|
|
|
'$ balena device init -f myorg/myfleet',
|
2022-09-06 10:26:26 +00:00
|
|
|
'$ balena device init --fleet myFleet --os-version 2.101.7 --drive /dev/disk5 --config config.json --yes',
|
2021-11-24 18:24:14 +00:00
|
|
|
'$ balena device init --fleet myFleet --os-version 2.83.21+rev1.prod --drive /dev/disk5 --config config.json --yes',
|
2020-07-02 13:58:07 +00:00
|
|
|
];
|
|
|
|
|
|
|
|
public static usage = 'device init';
|
|
|
|
|
|
|
|
public static flags: flags.Input<FlagsDef> = {
|
2021-07-15 13:41:38 +00:00
|
|
|
fleet: cf.fleet,
|
2020-07-02 13:58:07 +00:00
|
|
|
yes: cf.yes,
|
|
|
|
advanced: flags.boolean({
|
|
|
|
char: 'v',
|
|
|
|
description: 'show advanced configuration options',
|
|
|
|
}),
|
|
|
|
'os-version': flags.string({
|
|
|
|
description: stripIndent`
|
|
|
|
exact version number, or a valid semver range,
|
|
|
|
or 'latest' (includes pre-releases),
|
|
|
|
or 'default' (excludes pre-releases if at least one stable version is available),
|
|
|
|
or 'recommended' (excludes pre-releases, will fail if only pre-release versions are available),
|
|
|
|
or 'menu' (will show the interactive menu)
|
|
|
|
`,
|
|
|
|
}),
|
2020-07-15 19:13:18 +00:00
|
|
|
drive: cf.drive,
|
2020-07-02 13:58:07 +00:00
|
|
|
config: flags.string({
|
|
|
|
description: 'path to the config JSON file, see `balena os build-config`',
|
|
|
|
}),
|
2021-07-20 06:42:15 +00:00
|
|
|
'provisioning-key-name': flags.string({
|
|
|
|
description: 'custom key name assigned to generated provisioning api key',
|
|
|
|
}),
|
2022-05-10 18:02:45 +00:00
|
|
|
'provisioning-key-expiry-date': flags.string({
|
|
|
|
description:
|
|
|
|
'expiry date assigned to generated provisioning api key (format: YYYY-MM-DD)',
|
|
|
|
}),
|
2020-07-02 13:58:07 +00:00
|
|
|
help: cf.help,
|
|
|
|
};
|
|
|
|
|
|
|
|
public static authenticated = true;
|
|
|
|
|
|
|
|
public async run() {
|
|
|
|
const { flags: options } = this.parse<FlagsDef, {}>(DeviceInitCmd);
|
|
|
|
|
|
|
|
// Imports
|
2020-08-04 15:24:02 +00:00
|
|
|
const { promisify } = await import('util');
|
2020-07-02 13:58:07 +00:00
|
|
|
const rimraf = promisify(await import('rimraf'));
|
|
|
|
const tmp = await import('tmp');
|
|
|
|
const tmpNameAsync = promisify(tmp.tmpName);
|
|
|
|
tmp.setGracefulCleanup();
|
2020-07-14 09:24:41 +00:00
|
|
|
const { downloadOSImage } = await import('../../utils/cloud');
|
2020-11-04 15:42:31 +00:00
|
|
|
const { getApplication } = await import('../../utils/sdk');
|
2020-07-02 13:58:07 +00:00
|
|
|
|
2020-11-04 15:42:31 +00:00
|
|
|
const logger = await Command.getLogger();
|
|
|
|
const balena = getBalenaSdk();
|
2020-07-02 13:58:07 +00:00
|
|
|
|
|
|
|
// Get application and
|
2020-11-04 15:42:31 +00:00
|
|
|
const application = (await getApplication(
|
|
|
|
balena,
|
2021-12-19 20:50:03 +00:00
|
|
|
options.fleet ||
|
2021-07-20 13:57:00 +00:00
|
|
|
(
|
|
|
|
await (await import('../../utils/patterns')).selectApplication()
|
2022-07-15 15:01:37 +00:00
|
|
|
).slug,
|
2020-07-31 14:35:20 +00:00
|
|
|
{
|
|
|
|
$expand: {
|
|
|
|
is_for__device_type: {
|
|
|
|
$select: 'slug',
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)) as ApplicationWithDeviceType;
|
2020-07-02 13:58:07 +00:00
|
|
|
|
|
|
|
// Register new device
|
|
|
|
const deviceUuid = balena.models.device.generateUniqueKey();
|
2021-12-21 00:02:45 +00:00
|
|
|
console.info(`Registering to ${application.slug}: ${deviceUuid}`);
|
2020-07-02 13:58:07 +00:00
|
|
|
await balena.models.device.register(application.id, deviceUuid);
|
|
|
|
const device = await balena.models.device.get(deviceUuid);
|
|
|
|
|
|
|
|
// Download OS, configure, and flash
|
|
|
|
const tmpPath = (await tmpNameAsync()) as string;
|
|
|
|
try {
|
|
|
|
logger.logDebug(`Downloading OS image...`);
|
2020-07-14 09:24:41 +00:00
|
|
|
const osVersion = options['os-version'] || 'default';
|
2020-07-31 14:35:20 +00:00
|
|
|
const deviceType = application.is_for__device_type[0].slug;
|
|
|
|
await downloadOSImage(deviceType, tmpPath, osVersion);
|
2020-07-02 13:58:07 +00:00
|
|
|
|
|
|
|
logger.logDebug(`Configuring OS image...`);
|
|
|
|
await this.configureOsImage(tmpPath, device.uuid, options);
|
|
|
|
|
|
|
|
logger.logDebug(`Writing OS image...`);
|
2020-07-31 14:35:20 +00:00
|
|
|
await this.writeOsImage(tmpPath, deviceType, options);
|
2020-07-02 13:58:07 +00:00
|
|
|
} catch (e) {
|
|
|
|
// Remove device in failed cases
|
|
|
|
try {
|
|
|
|
logger.logDebug(`Process failed, removing device ${device.uuid}`);
|
|
|
|
await balena.models.device.remove(device.uuid);
|
|
|
|
} catch (e) {
|
|
|
|
// Ignore removal failures, and throw original error
|
|
|
|
}
|
|
|
|
throw e;
|
|
|
|
} finally {
|
|
|
|
// Remove temp download
|
|
|
|
logger.logDebug(`Removing temporary OS image download...`);
|
|
|
|
await rimraf(tmpPath);
|
|
|
|
}
|
|
|
|
|
|
|
|
console.log('Done');
|
|
|
|
return device.uuid;
|
|
|
|
}
|
|
|
|
|
|
|
|
async configureOsImage(path: string, uuid: string, options: FlagsDef) {
|
2020-07-03 08:05:40 +00:00
|
|
|
const configureCommand = ['os', 'configure', path, '--device', uuid];
|
2020-07-02 13:58:07 +00:00
|
|
|
if (options.config) {
|
2020-07-03 08:05:40 +00:00
|
|
|
configureCommand.push('--config', options.config);
|
2020-07-02 13:58:07 +00:00
|
|
|
} else if (options.advanced) {
|
2020-07-03 08:05:40 +00:00
|
|
|
configureCommand.push('--advanced');
|
2020-07-02 13:58:07 +00:00
|
|
|
}
|
2021-07-20 06:42:15 +00:00
|
|
|
|
|
|
|
if (options['provisioning-key-name']) {
|
|
|
|
configureCommand.push(
|
|
|
|
'--provisioning-key-name',
|
|
|
|
options['provisioning-key-name'],
|
|
|
|
);
|
|
|
|
}
|
2022-05-10 18:02:45 +00:00
|
|
|
|
|
|
|
if (options['provisioning-key-expiry-date']) {
|
|
|
|
configureCommand.push(
|
|
|
|
'--provisioning-key-expiry-date',
|
|
|
|
options['provisioning-key-expiry-date'],
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-07-02 13:58:07 +00:00
|
|
|
await runCommand(configureCommand);
|
|
|
|
}
|
|
|
|
|
|
|
|
async writeOsImage(path: string, deviceType: string, options: FlagsDef) {
|
2020-07-03 08:05:40 +00:00
|
|
|
const osInitCommand = ['os', 'initialize', path, '--type', deviceType];
|
2020-07-02 13:58:07 +00:00
|
|
|
if (options.yes) {
|
2020-07-03 08:05:40 +00:00
|
|
|
osInitCommand.push('--yes');
|
2020-07-02 13:58:07 +00:00
|
|
|
}
|
|
|
|
if (options.drive) {
|
2020-07-03 08:05:40 +00:00
|
|
|
osInitCommand.push('--drive', options.drive);
|
2020-07-02 13:58:07 +00:00
|
|
|
}
|
|
|
|
await runCommand(osInitCommand);
|
|
|
|
}
|
|
|
|
}
|