2020-07-09 13:38:04 +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';
|
2021-01-21 19:27:07 +00:00
|
|
|
import type { BlockDevice } from 'etcher-sdk/build/source-destination';
|
2020-07-09 13:38:04 +00:00
|
|
|
import Command from '../../command';
|
|
|
|
import { ExpectedError } from '../../errors';
|
|
|
|
import * as cf from '../../utils/common-flags';
|
|
|
|
import {
|
|
|
|
getChalk,
|
|
|
|
getCliForm,
|
|
|
|
getVisuals,
|
|
|
|
stripIndent,
|
|
|
|
} from '../../utils/lazy';
|
|
|
|
|
|
|
|
interface FlagsDef {
|
|
|
|
yes: boolean;
|
|
|
|
drive?: string;
|
|
|
|
help: void;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface ArgsDef {
|
|
|
|
image: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
export default class LocalFlashCmd extends Command {
|
|
|
|
public static description = stripIndent`
|
|
|
|
Flash an image to a drive.
|
|
|
|
|
|
|
|
Flash a balenaOS image to a drive.
|
|
|
|
Image file may be one of: .img|.zip|.gz|.bz2|.xz
|
|
|
|
|
|
|
|
If --drive is not specified, then it will interactively
|
|
|
|
show a list of available drives for selection.
|
|
|
|
`;
|
|
|
|
|
|
|
|
public static examples = [
|
|
|
|
'$ balena local flash path/to/balenaos.img',
|
|
|
|
'$ balena local flash path/to/balenaos.img --drive /dev/disk2',
|
|
|
|
'$ balena local flash path/to/balenaos.img --drive /dev/disk2 --yes',
|
|
|
|
];
|
|
|
|
|
|
|
|
public static args = [
|
|
|
|
{
|
|
|
|
name: 'image',
|
|
|
|
description: 'path to OS image',
|
|
|
|
required: true,
|
|
|
|
},
|
|
|
|
];
|
|
|
|
|
|
|
|
public static usage = 'local flash <image>';
|
|
|
|
|
|
|
|
public static flags: flags.Input<FlagsDef> = {
|
2020-07-15 19:13:18 +00:00
|
|
|
drive: cf.drive,
|
2020-07-09 13:38:04 +00:00
|
|
|
yes: cf.yes,
|
|
|
|
help: cf.help,
|
|
|
|
};
|
|
|
|
|
|
|
|
public async run() {
|
|
|
|
const { args: params, flags: options } = this.parse<FlagsDef, ArgsDef>(
|
|
|
|
LocalFlashCmd,
|
|
|
|
);
|
|
|
|
|
2021-01-21 19:27:07 +00:00
|
|
|
if (process.platform === 'linux') {
|
|
|
|
const { promisify } = await import('util');
|
|
|
|
const { exec } = await import('child_process');
|
|
|
|
const execAsync = promisify(exec);
|
|
|
|
let distroVersion = '';
|
|
|
|
try {
|
|
|
|
const info = await execAsync('cat /proc/version');
|
|
|
|
distroVersion = info.stdout.toLowerCase();
|
|
|
|
// tslint:disable-next-line: no-empty
|
|
|
|
} catch {}
|
|
|
|
if (distroVersion.includes('microsoft')) {
|
|
|
|
throw new ExpectedError(stripIndent`
|
|
|
|
This command is known not to work on WSL. Please use a CLI release
|
|
|
|
for Windows (not WSL), or balenaEtcher.`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-09 13:38:04 +00:00
|
|
|
const { sourceDestination, multiWrite } = await import('etcher-sdk');
|
|
|
|
|
|
|
|
const drive = await this.getDrive(options);
|
|
|
|
|
|
|
|
const yes =
|
|
|
|
options.yes ||
|
|
|
|
(await getCliForm().ask({
|
|
|
|
message: 'This will erase the selected drive. Are you sure?',
|
|
|
|
type: 'confirm',
|
|
|
|
name: 'yes',
|
|
|
|
default: false,
|
|
|
|
}));
|
|
|
|
|
|
|
|
if (!yes) {
|
|
|
|
console.log(getChalk().red.bold('Aborted image flash'));
|
|
|
|
process.exit(0);
|
|
|
|
}
|
|
|
|
|
2021-01-21 19:27:07 +00:00
|
|
|
const file = new sourceDestination.File({
|
|
|
|
path: params.image,
|
|
|
|
});
|
2020-07-09 13:38:04 +00:00
|
|
|
const source = await file.getInnerSource();
|
|
|
|
|
|
|
|
const visuals = getVisuals();
|
|
|
|
const progressBars: { [key: string]: any } = {
|
|
|
|
flashing: new visuals.Progress('Flashing'),
|
|
|
|
verifying: new visuals.Progress('Validating'),
|
|
|
|
};
|
|
|
|
|
2021-01-21 19:27:07 +00:00
|
|
|
await multiWrite.pipeSourceToDestinations({
|
2020-07-09 13:38:04 +00:00
|
|
|
source,
|
2021-01-21 19:27:07 +00:00
|
|
|
destinations: [drive],
|
|
|
|
onFail: (_, error) => {
|
|
|
|
console.error(getChalk().red.bold(error.message));
|
|
|
|
if (error.message.includes('EACCES')) {
|
|
|
|
console.error(
|
|
|
|
getChalk().red.bold(
|
|
|
|
'Try running this command with elevated privileges, with sudo or in a shell running with admininstrator privileges.',
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
2020-07-09 13:38:04 +00:00
|
|
|
},
|
2021-01-21 19:27:07 +00:00
|
|
|
onProgress: (progress) => {
|
2020-07-09 13:38:04 +00:00
|
|
|
progressBars[progress.type].update(progress);
|
|
|
|
},
|
2021-01-21 19:27:07 +00:00
|
|
|
verify: true,
|
|
|
|
});
|
2020-07-09 13:38:04 +00:00
|
|
|
}
|
|
|
|
|
2021-01-21 19:27:07 +00:00
|
|
|
async getDrive(options: { drive?: string }): Promise<BlockDevice> {
|
2020-07-09 13:38:04 +00:00
|
|
|
const drive = options.drive || (await getVisuals().drive('Select a drive'));
|
|
|
|
|
|
|
|
const sdk = await import('etcher-sdk');
|
|
|
|
|
2021-01-21 19:27:07 +00:00
|
|
|
const adapter = new sdk.scanner.adapters.BlockDeviceAdapter({
|
|
|
|
includeSystemDrives: () => false,
|
|
|
|
unmountOnSuccess: false,
|
|
|
|
write: true,
|
|
|
|
direct: true,
|
|
|
|
});
|
2020-07-09 13:38:04 +00:00
|
|
|
const scanner = new sdk.scanner.Scanner([adapter]);
|
|
|
|
await scanner.start();
|
|
|
|
try {
|
|
|
|
const d = scanner.getBy('device', drive);
|
|
|
|
if (
|
|
|
|
d === undefined ||
|
|
|
|
!(d instanceof sdk.sourceDestination.BlockDevice)
|
|
|
|
) {
|
|
|
|
throw new ExpectedError(`Drive not found: ${options.drive}`);
|
|
|
|
}
|
|
|
|
return d;
|
|
|
|
} finally {
|
|
|
|
scanner.stop();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|