mirror of
https://github.com/balena-io/balena-cli.git
synced 2025-01-15 09:20:31 +00:00
198 lines
5.8 KiB
TypeScript
198 lines
5.8 KiB
TypeScript
|
/**
|
||
|
* @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 type { IArg } from '@oclif/parser/lib/args';
|
||
|
import Command from '../../command';
|
||
|
import * as cf from '../../utils/common-flags';
|
||
|
import { getBalenaSdk, getCliUx, stripIndent } from '../../utils/lazy';
|
||
|
import type {
|
||
|
BalenaSDK,
|
||
|
DeviceWithServiceDetails,
|
||
|
CurrentServiceWithCommit,
|
||
|
} from 'balena-sdk';
|
||
|
|
||
|
interface FlagsDef {
|
||
|
help: void;
|
||
|
service?: string;
|
||
|
}
|
||
|
|
||
|
interface ArgsDef {
|
||
|
uuid: string;
|
||
|
}
|
||
|
|
||
|
export default class DeviceRestartCmd extends Command {
|
||
|
public static description = stripIndent`
|
||
|
Restart containers on a device.
|
||
|
|
||
|
Restart containers on a device.
|
||
|
If the --service flag is provided, then only those services' containers
|
||
|
will be restarted, otherwise all containers on the device will be restarted.
|
||
|
|
||
|
Multiple devices and services may be specified with a comma-separated list
|
||
|
of values (no spaces).
|
||
|
|
||
|
Note this does not reboot the device, to do so use instead \`balena device reboot\`.
|
||
|
`;
|
||
|
public static examples = [
|
||
|
'$ balena device restart 23c73a1',
|
||
|
'$ balena device restart 55d43b3,23c73a1',
|
||
|
'$ balena device restart 23c73a1 --service myService',
|
||
|
'$ balena device restart 23c73a1 -s myService1,myService2',
|
||
|
];
|
||
|
|
||
|
public static args: Array<IArg<any>> = [
|
||
|
{
|
||
|
name: 'uuid',
|
||
|
description:
|
||
|
'comma-separated list (no blank spaces) of device UUIDs to restart',
|
||
|
required: true,
|
||
|
},
|
||
|
];
|
||
|
|
||
|
public static usage = 'device restart <uuid>';
|
||
|
|
||
|
public static flags: flags.Input<FlagsDef> = {
|
||
|
service: flags.string({
|
||
|
description:
|
||
|
'comma-separated list (no blank spaces) of service names to restart',
|
||
|
char: 's',
|
||
|
}),
|
||
|
help: cf.help,
|
||
|
};
|
||
|
|
||
|
public static authenticated = true;
|
||
|
|
||
|
public async run() {
|
||
|
const { args: params, flags: options } = this.parse<FlagsDef, ArgsDef>(
|
||
|
DeviceRestartCmd,
|
||
|
);
|
||
|
|
||
|
const { tryAsInteger } = await import('../../utils/validation');
|
||
|
const balena = getBalenaSdk();
|
||
|
const ux = getCliUx();
|
||
|
|
||
|
const deviceIds = params.uuid.split(',').map((id) => {
|
||
|
return tryAsInteger(id);
|
||
|
});
|
||
|
const serviceNames = options.service?.split(',');
|
||
|
|
||
|
// Iterate sequentially through deviceIds.
|
||
|
// We may later want to add a batching feature,
|
||
|
// so that n devices are processed in parallel
|
||
|
for (const deviceId of deviceIds) {
|
||
|
ux.action.start(`Restarting services on device ${deviceId}`);
|
||
|
if (serviceNames) {
|
||
|
await this.restartServices(balena, deviceId, serviceNames);
|
||
|
} else {
|
||
|
await this.restartAllServices(balena, deviceId);
|
||
|
}
|
||
|
ux.action.stop();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
async restartServices(
|
||
|
balena: BalenaSDK,
|
||
|
deviceId: number | string,
|
||
|
serviceNames: string[],
|
||
|
) {
|
||
|
const { ExpectedError, instanceOf } = await import('../../errors');
|
||
|
const { getExpandedProp } = await import('../../utils/pine');
|
||
|
|
||
|
// Get device
|
||
|
let device: DeviceWithServiceDetails<CurrentServiceWithCommit>;
|
||
|
try {
|
||
|
device = await balena.models.device.getWithServiceDetails(deviceId, {
|
||
|
$expand: {
|
||
|
is_running__release: { $select: 'commit' },
|
||
|
},
|
||
|
});
|
||
|
} catch (e) {
|
||
|
const { BalenaDeviceNotFound } = await import('balena-errors');
|
||
|
if (instanceOf(e, BalenaDeviceNotFound)) {
|
||
|
throw new ExpectedError(`Device ${deviceId} not found.`);
|
||
|
} else {
|
||
|
throw e;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
const activeRelease = getExpandedProp(device.is_running__release, 'commit');
|
||
|
|
||
|
// Check specified services exist on this device before restarting anything
|
||
|
serviceNames.forEach((service) => {
|
||
|
if (!device.current_services[service]) {
|
||
|
throw new ExpectedError(
|
||
|
`Service ${service} not found on device ${deviceId}.`,
|
||
|
);
|
||
|
}
|
||
|
});
|
||
|
|
||
|
// Restart services
|
||
|
const restartPromises: Array<Promise<void>> = [];
|
||
|
for (const serviceName of serviceNames) {
|
||
|
const service = device.current_services[serviceName];
|
||
|
// Each service is an array of `CurrentServiceWithCommit`
|
||
|
// because when service is updating, it will actually hold 2 services
|
||
|
// Target commit matching `device.is_running__release`
|
||
|
const serviceContainer = service.find((s) => {
|
||
|
return s.commit === activeRelease;
|
||
|
});
|
||
|
|
||
|
if (serviceContainer) {
|
||
|
restartPromises.push(
|
||
|
balena.models.device.restartService(
|
||
|
deviceId,
|
||
|
serviceContainer.image_id,
|
||
|
),
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
try {
|
||
|
await Promise.all(restartPromises);
|
||
|
} catch (e) {
|
||
|
if (e.message.toLowerCase().includes('no online device')) {
|
||
|
throw new ExpectedError(`Device ${deviceId} is not online.`);
|
||
|
} else {
|
||
|
throw e;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
async restartAllServices(balena: BalenaSDK, deviceId: number | string) {
|
||
|
// Note: device.restartApplication throws `BalenaDeviceNotFound: Device not found` if device not online.
|
||
|
// Need to use device.get first to distinguish between non-existant and offline devices.
|
||
|
// Remove this workaround when SDK issue resolved: https://github.com/balena-io/balena-sdk/issues/649
|
||
|
const { instanceOf, ExpectedError } = await import('../../errors');
|
||
|
try {
|
||
|
const device = await balena.models.device.get(deviceId);
|
||
|
if (!device.is_online) {
|
||
|
throw new ExpectedError(`Device ${deviceId} is not online.`);
|
||
|
}
|
||
|
} catch (e) {
|
||
|
const { BalenaDeviceNotFound } = await import('balena-errors');
|
||
|
if (instanceOf(e, BalenaDeviceNotFound)) {
|
||
|
throw new ExpectedError(`Device ${deviceId} not found.`);
|
||
|
} else {
|
||
|
throw e;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
await balena.models.device.restartApplication(deviceId);
|
||
|
}
|
||
|
}
|