2020-06-17 13:46:25 +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 { expandForAppName } from '../../utils/helpers';
|
2020-05-04 14:57:23 +00:00
|
|
|
import { getBalenaSdk, getVisuals, stripIndent } from '../../utils/lazy';
|
2020-06-17 13:46:25 +00:00
|
|
|
import { tryAsInteger } from '../../utils/validation';
|
2020-07-31 14:35:20 +00:00
|
|
|
import type { Application } from 'balena-sdk';
|
2020-06-17 13:46:25 +00:00
|
|
|
|
2020-07-31 14:35:20 +00:00
|
|
|
interface ExtendedDevice extends DeviceWithDeviceType {
|
2020-06-17 13:46:25 +00:00
|
|
|
dashboard_url?: string;
|
2020-10-27 21:12:49 +00:00
|
|
|
application_name?: string | null;
|
|
|
|
device_type?: string | null;
|
2020-06-17 13:46:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
interface FlagsDef {
|
|
|
|
application?: string;
|
|
|
|
app?: string;
|
|
|
|
help: void;
|
2020-08-07 14:08:20 +00:00
|
|
|
json: boolean;
|
2020-06-17 13:46:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export default class DevicesCmd extends Command {
|
|
|
|
public static description = stripIndent`
|
|
|
|
List all devices.
|
|
|
|
|
|
|
|
list all devices that belong to you.
|
|
|
|
|
|
|
|
You can filter the devices by application by using the \`--application\` option.
|
2020-08-07 14:08:20 +00:00
|
|
|
|
|
|
|
The --json option is recommended when scripting the output of this command,
|
|
|
|
because field names are less likely to change in JSON format and because it
|
2020-10-27 21:12:49 +00:00
|
|
|
better represents data types like arrays, empty strings and null values.
|
|
|
|
The 'jq' utility may be helpful for querying JSON fields in shell scripts
|
|
|
|
(https://stedolan.github.io/jq/manual/).
|
2020-06-17 13:46:25 +00:00
|
|
|
`;
|
|
|
|
public static examples = [
|
|
|
|
'$ balena devices',
|
|
|
|
'$ balena devices --application MyApp',
|
|
|
|
'$ balena devices --app MyApp',
|
|
|
|
'$ balena devices -a MyApp',
|
|
|
|
];
|
|
|
|
|
|
|
|
public static usage = 'devices';
|
|
|
|
|
|
|
|
public static flags: flags.Input<FlagsDef> = {
|
|
|
|
application: cf.application,
|
|
|
|
app: cf.app,
|
|
|
|
help: cf.help,
|
2020-08-07 14:08:20 +00:00
|
|
|
json: flags.boolean({
|
|
|
|
char: 'j',
|
|
|
|
description: 'produce JSON output instead of tabular output',
|
|
|
|
}),
|
2020-06-17 13:46:25 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
public static primary = true;
|
|
|
|
|
|
|
|
public static authenticated = true;
|
|
|
|
|
|
|
|
public async run() {
|
|
|
|
const { flags: options } = this.parse<FlagsDef, {}>(DevicesCmd);
|
|
|
|
|
|
|
|
const balena = getBalenaSdk();
|
|
|
|
|
|
|
|
// Consolidate application options
|
|
|
|
options.application = options.application || options.app;
|
|
|
|
delete options.app;
|
|
|
|
|
2020-07-31 14:35:20 +00:00
|
|
|
let devices;
|
2020-06-17 13:46:25 +00:00
|
|
|
|
|
|
|
if (options.application != null) {
|
2020-07-31 14:35:20 +00:00
|
|
|
devices = (await balena.models.device.getAllByApplication(
|
2020-06-17 13:46:25 +00:00
|
|
|
tryAsInteger(options.application),
|
|
|
|
expandForAppName,
|
2020-07-31 14:35:20 +00:00
|
|
|
)) as ExtendedDevice[];
|
2020-06-17 13:46:25 +00:00
|
|
|
} else {
|
2020-07-31 14:35:20 +00:00
|
|
|
devices = (await balena.models.device.getAll(
|
|
|
|
expandForAppName,
|
|
|
|
)) as ExtendedDevice[];
|
2020-06-17 13:46:25 +00:00
|
|
|
}
|
|
|
|
|
2020-07-08 16:34:53 +00:00
|
|
|
devices = devices.map(function (device) {
|
2020-06-17 13:46:25 +00:00
|
|
|
device.dashboard_url = balena.models.device.getDashboardUrl(device.uuid);
|
|
|
|
|
|
|
|
const belongsToApplication = device.belongs_to__application as Application[];
|
2020-10-27 21:12:49 +00:00
|
|
|
device.application_name = belongsToApplication?.[0]?.app_name || null;
|
2020-06-17 13:46:25 +00:00
|
|
|
|
2020-12-03 20:54:08 +00:00
|
|
|
device.uuid = options.json ? device.uuid : device.uuid.slice(0, 7);
|
2020-07-31 14:35:20 +00:00
|
|
|
|
2020-10-27 21:12:49 +00:00
|
|
|
device.device_type = device.is_of__device_type?.[0]?.slug || null;
|
2020-06-17 13:46:25 +00:00
|
|
|
return device;
|
|
|
|
});
|
|
|
|
|
2020-08-07 14:08:20 +00:00
|
|
|
const fields = [
|
|
|
|
'id',
|
|
|
|
'uuid',
|
|
|
|
'device_name',
|
|
|
|
'device_type',
|
|
|
|
'application_name',
|
|
|
|
'status',
|
|
|
|
'is_online',
|
|
|
|
'supervisor_version',
|
|
|
|
'os_version',
|
|
|
|
'dashboard_url',
|
|
|
|
];
|
2020-10-27 21:12:49 +00:00
|
|
|
const _ = await import('lodash');
|
2020-08-07 14:08:20 +00:00
|
|
|
if (options.json) {
|
|
|
|
console.log(
|
|
|
|
JSON.stringify(
|
|
|
|
devices.map((device) => _.pick(device, fields)),
|
|
|
|
null,
|
|
|
|
4,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
} else {
|
2020-10-27 21:12:49 +00:00
|
|
|
console.log(
|
|
|
|
getVisuals().table.horizontal(
|
|
|
|
devices.map((dev) => _.mapValues(dev, (val) => val ?? 'N/a')),
|
|
|
|
fields,
|
|
|
|
),
|
|
|
|
);
|
2020-08-07 14:08:20 +00:00
|
|
|
}
|
2020-06-17 13:46:25 +00:00
|
|
|
}
|
|
|
|
}
|