diff --git a/automation/capitanodoc/capitanodoc.ts b/automation/capitanodoc/capitanodoc.ts index 10dcdf4a41..8f25b370e8 100644 --- a/automation/capitanodoc/capitanodoc.ts +++ b/automation/capitanodoc/capitanodoc.ts @@ -73,6 +73,8 @@ const capitanoDoc = { 'build/commands/device/restart.js', 'build/commands/device/rm.js', 'build/commands/device/shutdown.js', + 'build/commands/device/startservice.js', + 'build/commands/device/stopservice.js', ], }, { diff --git a/completion/_balena b/completion/_balena index 71fe71249c..c06cc6ff9c 100644 --- a/completion/_balena +++ b/completion/_balena @@ -14,7 +14,7 @@ _balena() { app_cmds=( create ) block_cmds=( create ) config_cmds=( generate inject read reconfigure write ) - device_cmds=( deactivate identify init local-mode move os-update pin public-url purge reboot register rename restart rm shutdown track-fleet ) + device_cmds=( deactivate identify init local-mode move os-update pin public-url purge reboot register rename restart rm shutdown startservice stopservice track-fleet ) devices_cmds=( supported ) env_cmds=( add rename rm ) fleet_cmds=( create pin purge rename restart rm track-latest ) diff --git a/completion/balena-completion.bash b/completion/balena-completion.bash index 272a928f91..a71304e738 100644 --- a/completion/balena-completion.bash +++ b/completion/balena-completion.bash @@ -13,7 +13,7 @@ _balena_complete() app_cmds="create" block_cmds="create" config_cmds="generate inject read reconfigure write" - device_cmds="deactivate identify init local-mode move os-update pin public-url purge reboot register rename restart rm shutdown track-fleet" + device_cmds="deactivate identify init local-mode move os-update pin public-url purge reboot register rename restart rm shutdown startservice stopservice track-fleet" devices_cmds="supported" env_cmds="add rename rm" fleet_cmds="create pin purge rename restart rm track-latest" diff --git a/docs/balena-cli.md b/docs/balena-cli.md index 65c167d457..b7e16c4d63 100644 --- a/docs/balena-cli.md +++ b/docs/balena-cli.md @@ -199,6 +199,8 @@ are encouraged to regularly update the balena CLI to the latest version. - [device restart <uuid>](#device-restart-uuid) - [device rm <uuid(s)>](#device-rm-uuid-s) - [device shutdown <uuid>](#device-shutdown-uuid) + - [device startservice <uuid>](#device-startservice-uuid) + - [device stopservice <uuid>](#device-stopservice-uuid) - Releases @@ -1157,6 +1159,58 @@ the uuid of the device to shutdown force action if the update lock is set +## device startservice <uuid> + +Start containers on a device. +If the --service flag is provided, then only those services' containers +will be started. + +Multiple devices and services may be specified with a comma-separated list +of values (no spaces). + +Examples: + + $ balena device startservice 23c73a1 --service myService + $ balena device startservice 23c73a1 -s myService1,myService2 + +### Arguments + +#### UUID + +comma-separated list (no blank spaces) of device UUIDs to start + +### Options + +#### -s, --service SERVICE + +comma-separated list (no blank spaces) of service names to start + +## device stopservice <uuid> + +Stop containers on a device. +If the --service flag is provided, then only those services' containers +will be stopped. + +Multiple devices and services may be specified with a comma-separated list +of values (no spaces). + +Examples: + + $ balena device stopservice 23c73a1 --service myService + $ balena device stopservice 23c73a1 -s myService1,myService2 + +### Arguments + +#### UUID + +comma-separated list (no blank spaces) of device UUIDs to stop + +### Options + +#### -s, --service SERVICE + +comma-separated list (no blank spaces) of service names to stop + # Releases ## releases <fleet> diff --git a/lib/commands/device/startservice.ts b/lib/commands/device/startservice.ts new file mode 100644 index 0000000000..9fccb9faf2 --- /dev/null +++ b/lib/commands/device/startservice.ts @@ -0,0 +1,155 @@ +/** + * @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, Args } from '@oclif/core'; +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'; + +export default class DeviceStartServiceCmd extends Command { + public static description = stripIndent` + Start containers on a device. + + Start containers on a device. + If the --service flag is provided, then only those services' containers + will be started. + + Multiple devices and services may be specified with a comma-separated list + of values (no spaces). + `; + public static examples = [ + '$ balena device startservice 23c73a1 --service myService', + '$ balena device startservice 23c73a1 -s myService1,myService2', + ]; + + public static args = { + uuid: Args.string({ + description: + 'comma-separated list (no blank spaces) of device UUIDs to start', + required: true, + }), + }; + + public static usage = 'device startservice '; + + public static flags = { + service: Flags.string({ + description: + 'comma-separated list (no blank spaces) of service names to start', + char: 's', + }), + help: cf.help, + }; + + public static authenticated = true; + + public async run() { + const { args: params, flags: options } = await this.parse( + DeviceStartServiceCmd, + ); + + const balena = getBalenaSdk(); + const ux = getCliUx(); + + const deviceUuids = params.uuid.split(','); + const serviceNames = options.service?.split(','); + + // Iterate sequentially through deviceUuids. + // We may later want to add a batching feature, + // so that n devices are processed in parallel + for (const uuid of deviceUuids) { + ux.action.start(`Starting services on device ${uuid}`); + if (serviceNames) { + await this.startServices(balena, uuid, serviceNames); + } + ux.action.stop(); + } + } + + async startServices( + balena: BalenaSDK, + deviceUuid: string, + serviceNames: string[], + ) { + const { ExpectedError, instanceOf } = await import('../../errors'); + const { getExpandedProp } = await import('../../utils/pine'); + + // Get device + let device: DeviceWithServiceDetails; + try { + device = await balena.models.device.getWithServiceDetails(deviceUuid, { + $expand: { + is_running__release: { $select: 'commit' }, + }, + }); + } catch (e) { + const { BalenaDeviceNotFound } = await import('balena-errors'); + if (instanceOf(e, BalenaDeviceNotFound)) { + throw new ExpectedError(`Device ${deviceUuid} not found.`); + } else { + throw e; + } + } + + const activeRelease = getExpandedProp(device.is_running__release, 'commit'); + + // Check specified services exist on this device before startinganything + serviceNames.forEach((service) => { + if (!device.current_services[service]) { + throw new ExpectedError( + `Service ${service} not found on device ${deviceUuid}.`, + ); + } + }); + + // Start services + const startPromises: Array> = []; + 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) { + startPromises.push( + balena.models.device.startService( + deviceUuid, + serviceContainer.image_id, + ), + ); + } + } + + try { + await Promise.all(startPromises); + } catch (e) { + if (e.message.toLowerCase().includes('no online device')) { + throw new ExpectedError(`Device ${deviceUuid} is not online.`); + } else { + throw e; + } + } + } +} diff --git a/lib/commands/device/stopservice.ts b/lib/commands/device/stopservice.ts new file mode 100644 index 0000000000..44f223aaff --- /dev/null +++ b/lib/commands/device/stopservice.ts @@ -0,0 +1,155 @@ +/** + * @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, Args } from '@oclif/core'; +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'; + +export default class DeviceStopServiceCmd extends Command { + public static description = stripIndent` + Stop containers on a device. + + Stop containers on a device. + If the --service flag is provided, then only those services' containers + will be stopped. + + Multiple devices and services may be specified with a comma-separated list + of values (no spaces). + `; + public static examples = [ + '$ balena device stopservice 23c73a1 --service myService', + '$ balena device stopservice 23c73a1 -s myService1,myService2', + ]; + + public static args = { + uuid: Args.string({ + description: + 'comma-separated list (no blank spaces) of device UUIDs to stop', + required: true, + }), + }; + + public static usage = 'device stopservice '; + + public static flags = { + service: Flags.string({ + description: + 'comma-separated list (no blank spaces) of service names to stop', + char: 's', + }), + help: cf.help, + }; + + public static authenticated = true; + + public async run() { + const { args: params, flags: options } = await this.parse( + DeviceStopServiceCmd, + ); + + const balena = getBalenaSdk(); + const ux = getCliUx(); + + const deviceUuids = params.uuid.split(','); + const serviceNames = options.service?.split(','); + + // Iterate sequentially through deviceUuids. + // We may later want to add a batching feature, + // so that n devices are processed in parallel + for (const uuid of deviceUuids) { + ux.action.start(`Stopping services on device ${uuid}`); + if (serviceNames) { + await this.stopServices(balena, uuid, serviceNames); + } + ux.action.stop(); + } + } + + async stopServices( + balena: BalenaSDK, + deviceUuid: string, + serviceNames: string[], + ) { + const { ExpectedError, instanceOf } = await import('../../errors'); + const { getExpandedProp } = await import('../../utils/pine'); + + // Get device + let device: DeviceWithServiceDetails; + try { + device = await balena.models.device.getWithServiceDetails(deviceUuid, { + $expand: { + is_running__release: { $select: 'commit' }, + }, + }); + } catch (e) { + const { BalenaDeviceNotFound } = await import('balena-errors'); + if (instanceOf(e, BalenaDeviceNotFound)) { + throw new ExpectedError(`Device ${deviceUuid} not found.`); + } else { + throw e; + } + } + + const activeRelease = getExpandedProp(device.is_running__release, 'commit'); + + // Check specified services exist on this device before stoppinganything + serviceNames.forEach((service) => { + if (!device.current_services[service]) { + throw new ExpectedError( + `Service ${service} not found on device ${deviceUuid}.`, + ); + } + }); + + // Stop services + const stopPromises: Array> = []; + 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) { + stopPromises.push( + balena.models.device.stopService( + deviceUuid, + serviceContainer.image_id, + ), + ); + } + } + + try { + await Promise.all(stopPromises); + } catch (e) { + if (e.message.toLowerCase().includes('no online device')) { + throw new ExpectedError(`Device ${deviceUuid} is not online.`); + } else { + throw e; + } + } + } +}