2017-12-20 22:46:01 +01:00
|
|
|
/*
|
2020-06-26 13:46:58 +02:00
|
|
|
Copyright 2016-2020 Balena Ltd.
|
2017-12-20 22:46:01 +01:00
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
2021-12-19 23:04:16 +00:00
|
|
|
|
2023-05-20 00:58:32 +03:00
|
|
|
import type {
|
|
|
|
Application,
|
|
|
|
BalenaSDK,
|
|
|
|
Device,
|
|
|
|
Organization,
|
2023-07-19 19:22:17 +03:00
|
|
|
PineFilter,
|
2023-05-20 00:58:32 +03:00
|
|
|
PineOptions,
|
|
|
|
PineTypedResult,
|
|
|
|
} from 'balena-sdk';
|
2017-12-20 22:46:01 +01:00
|
|
|
|
2021-08-27 00:49:54 +01:00
|
|
|
import { instanceOf, NotLoggedInError, ExpectedError } from '../errors';
|
2020-07-08 18:03:10 +01:00
|
|
|
import { getBalenaSdk, getVisuals, stripIndent, getCliForm } from './lazy';
|
2019-03-12 22:07:57 +00:00
|
|
|
import validation = require('./validation');
|
2020-07-01 15:26:40 +01:00
|
|
|
import { delay } from './helpers';
|
2017-12-20 22:46:01 +01:00
|
|
|
|
2023-10-27 10:57:07 -04:00
|
|
|
export function authenticate(options: object): Promise<void> {
|
2019-01-12 21:20:19 +00:00
|
|
|
const balena = getBalenaSdk();
|
2020-07-08 18:03:10 +01:00
|
|
|
return getCliForm()
|
2018-01-09 16:05:24 +01:00
|
|
|
.run(
|
|
|
|
[
|
|
|
|
{
|
|
|
|
message: 'Email:',
|
|
|
|
name: 'email',
|
|
|
|
type: 'input',
|
|
|
|
validate: validation.validateEmail,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
message: 'Password:',
|
|
|
|
name: 'password',
|
|
|
|
type: 'password',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
{ override: options },
|
|
|
|
)
|
2018-10-19 16:38:50 +02:00
|
|
|
.then(balena.auth.login)
|
|
|
|
.then(balena.auth.twoFactor.isPassed)
|
2018-01-09 16:05:24 +01:00
|
|
|
.then((isTwoFactorAuthPassed: boolean) => {
|
|
|
|
if (isTwoFactorAuthPassed) {
|
|
|
|
return;
|
|
|
|
}
|
2017-12-20 22:46:01 +01:00
|
|
|
|
2020-07-08 18:03:10 +01:00
|
|
|
return getCliForm()
|
2018-01-09 16:05:24 +01:00
|
|
|
.ask({
|
|
|
|
message: 'Two factor auth challenge:',
|
|
|
|
name: 'code',
|
|
|
|
type: 'input',
|
|
|
|
})
|
2018-10-19 16:38:50 +02:00
|
|
|
.then(balena.auth.twoFactor.challenge)
|
2018-01-09 16:05:24 +01:00
|
|
|
.catch((error: any) => {
|
2018-10-19 16:38:50 +02:00
|
|
|
return balena.auth.logout().then(() => {
|
2018-01-09 16:05:24 +01:00
|
|
|
if (
|
2018-10-19 16:38:50 +02:00
|
|
|
error.name === 'BalenaRequestError' &&
|
2018-01-09 16:05:24 +01:00
|
|
|
error.statusCode === 401
|
|
|
|
) {
|
2020-06-26 13:46:58 +02:00
|
|
|
throw new ExpectedError('Invalid two factor authentication code');
|
2018-01-09 16:05:24 +01:00
|
|
|
}
|
|
|
|
throw error;
|
|
|
|
});
|
|
|
|
});
|
2018-01-04 16:17:43 +00:00
|
|
|
});
|
2018-01-04 14:07:55 +00:00
|
|
|
}
|
2017-12-20 22:46:01 +01:00
|
|
|
|
2020-03-24 09:51:19 +01:00
|
|
|
/**
|
|
|
|
* Check if logged in, and throw `NotLoggedInError` if not.
|
|
|
|
* Note: `NotLoggedInError` is an `ExpectedError`.
|
|
|
|
*/
|
2019-10-31 01:40:57 +00:00
|
|
|
export async function checkLoggedIn(): Promise<void> {
|
2019-03-25 15:16:23 +00:00
|
|
|
const balena = getBalenaSdk();
|
|
|
|
if (!(await balena.auth.isLoggedIn())) {
|
2020-06-26 13:46:58 +02:00
|
|
|
throw new NotLoggedInError(stripIndent`
|
|
|
|
Login required: use the “balena login” command to log in.
|
|
|
|
`);
|
2019-03-25 15:16:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-20 22:46:01 +01:00
|
|
|
export function askLoginType() {
|
2020-07-08 18:03:10 +01:00
|
|
|
return getCliForm().ask<'web' | 'credentials' | 'token' | 'register'>({
|
2017-12-20 22:46:01 +01:00
|
|
|
message: 'How would you like to login?',
|
|
|
|
name: 'loginType',
|
|
|
|
type: 'list',
|
2018-01-09 16:05:24 +01:00
|
|
|
choices: [
|
|
|
|
{
|
|
|
|
name: 'Web authorization (recommended)',
|
|
|
|
value: 'web',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Credentials',
|
|
|
|
value: 'credentials',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Authentication token',
|
|
|
|
value: 'token',
|
|
|
|
},
|
|
|
|
{
|
2018-10-19 16:38:50 +02:00
|
|
|
name: "I don't have a balena account!",
|
2018-01-09 16:05:24 +01:00
|
|
|
value: 'register',
|
|
|
|
},
|
|
|
|
],
|
2018-01-04 14:07:55 +00:00
|
|
|
});
|
2017-12-20 22:46:01 +01:00
|
|
|
}
|
|
|
|
|
2023-07-07 19:51:43 +03:00
|
|
|
export async function selectDeviceType() {
|
|
|
|
const sdk = getBalenaSdk();
|
|
|
|
let deviceTypes = await sdk.models.deviceType.getAllSupported();
|
|
|
|
if (deviceTypes.length === 0) {
|
|
|
|
// Without this open-balena users would get an empty list
|
|
|
|
// until we add a hostApps import in open-balena.
|
|
|
|
deviceTypes = await sdk.models.deviceType.getAll();
|
|
|
|
}
|
|
|
|
return getCliForm().ask({
|
|
|
|
message: 'Device Type',
|
|
|
|
type: 'list',
|
2023-07-07 20:21:44 +03:00
|
|
|
choices: deviceTypes.map(({ slug: value, name }) => ({
|
2023-07-07 19:51:43 +03:00
|
|
|
name,
|
|
|
|
value,
|
|
|
|
})),
|
|
|
|
});
|
2017-12-20 22:46:01 +01:00
|
|
|
}
|
|
|
|
|
2020-07-15 21:13:18 +02:00
|
|
|
/**
|
|
|
|
* Display interactive confirmation prompt.
|
2021-08-27 00:49:54 +01:00
|
|
|
* Throw ExpectedError if the user declines.
|
2020-07-15 21:13:18 +02:00
|
|
|
* @param yesOption - automatically confirm if true
|
|
|
|
* @param message - message to display with prompt
|
|
|
|
* @param yesMessage - message to display if automatically confirming
|
|
|
|
*/
|
2020-07-01 15:26:40 +01:00
|
|
|
export async function confirm(
|
2018-10-25 14:57:45 +02:00
|
|
|
yesOption: boolean,
|
2018-01-09 16:05:24 +01:00
|
|
|
message: string,
|
2018-10-25 14:57:45 +02:00
|
|
|
yesMessage?: string,
|
2021-08-27 00:49:54 +01:00
|
|
|
defaultValue = false,
|
2018-01-09 16:05:24 +01:00
|
|
|
) {
|
2020-07-01 15:26:40 +01:00
|
|
|
if (yesOption) {
|
|
|
|
if (yesMessage) {
|
2021-12-19 23:04:16 +00:00
|
|
|
console.log(yesMessage);
|
2017-12-20 22:46:01 +01:00
|
|
|
}
|
2020-07-01 15:26:40 +01:00
|
|
|
return;
|
|
|
|
}
|
2017-12-20 22:46:01 +01:00
|
|
|
|
2020-07-08 18:03:10 +01:00
|
|
|
const confirmed = await getCliForm().ask<boolean>({
|
2020-07-01 15:26:40 +01:00
|
|
|
message,
|
|
|
|
type: 'confirm',
|
2021-08-27 00:49:54 +01:00
|
|
|
default: defaultValue,
|
2018-01-04 14:07:55 +00:00
|
|
|
});
|
2020-07-01 15:26:40 +01:00
|
|
|
|
|
|
|
if (!confirmed) {
|
2021-08-27 00:49:54 +01:00
|
|
|
throw new ExpectedError('Aborted');
|
2020-07-01 15:26:40 +01:00
|
|
|
}
|
2017-12-20 22:46:01 +01:00
|
|
|
}
|
|
|
|
|
2023-05-20 00:58:32 +03:00
|
|
|
const selectApplicationPineOptions = {
|
|
|
|
$select: ['id', 'slug', 'app_name'],
|
|
|
|
$expand: {
|
|
|
|
is_for__device_type: {
|
|
|
|
$select: 'slug',
|
|
|
|
},
|
|
|
|
},
|
|
|
|
} satisfies PineOptions<Application>;
|
|
|
|
|
|
|
|
type SelectApplicationResult = PineTypedResult<
|
|
|
|
Application,
|
|
|
|
typeof selectApplicationPineOptions
|
|
|
|
>;
|
|
|
|
|
2023-05-19 20:59:49 +03:00
|
|
|
export async function selectApplication(
|
2023-07-19 19:22:17 +03:00
|
|
|
filter?:
|
|
|
|
| PineFilter<Application>
|
|
|
|
| ((app: SelectApplicationResult) => boolean),
|
2020-07-15 08:26:31 -06:00
|
|
|
errorOnEmptySelection = false,
|
2018-10-19 13:14:28 +02:00
|
|
|
) {
|
2019-01-12 21:20:19 +00:00
|
|
|
const balena = getBalenaSdk();
|
2023-07-19 19:22:17 +03:00
|
|
|
let apps = (await balena.models.application.getAllDirectlyAccessible({
|
|
|
|
...selectApplicationPineOptions,
|
|
|
|
...(filter != null && typeof filter === 'object' && { $filter: filter }),
|
|
|
|
})) as SelectApplicationResult[];
|
2017-12-20 22:46:01 +01:00
|
|
|
|
2023-05-19 20:59:49 +03:00
|
|
|
if (!apps.length) {
|
|
|
|
throw new ExpectedError('No fleets found');
|
|
|
|
}
|
|
|
|
|
2023-07-19 19:22:17 +03:00
|
|
|
if (filter != null && typeof filter === 'function') {
|
2023-07-07 20:21:44 +03:00
|
|
|
apps = apps.filter(filter);
|
|
|
|
}
|
2023-05-19 20:59:49 +03:00
|
|
|
|
2023-07-07 20:21:44 +03:00
|
|
|
if (errorOnEmptySelection && apps.length === 0) {
|
2023-05-19 20:59:49 +03:00
|
|
|
throw new ExpectedError('No suitable fleets found for selection');
|
|
|
|
}
|
|
|
|
return getCliForm().ask({
|
|
|
|
message: 'Select an application',
|
|
|
|
type: 'list',
|
2023-07-07 20:21:44 +03:00
|
|
|
choices: apps.map((application) => ({
|
2023-05-19 20:59:49 +03:00
|
|
|
name: `${application.app_name} (${application.slug}) [${application.is_for__device_type[0].slug}]`,
|
|
|
|
value: application,
|
|
|
|
})),
|
|
|
|
});
|
2017-12-20 22:46:01 +01:00
|
|
|
}
|
|
|
|
|
2023-05-19 20:59:24 +03:00
|
|
|
export async function selectOrganization(
|
|
|
|
organizations?: Array<Pick<Organization, 'handle' | 'name'>>,
|
|
|
|
) {
|
2020-12-09 16:05:12 +01:00
|
|
|
// Use either provided orgs (if e.g. already loaded) or load from cloud
|
2023-05-20 00:19:10 +03:00
|
|
|
organizations ??= await getBalenaSdk().models.organization.getAll({
|
|
|
|
$select: ['name', 'handle'],
|
|
|
|
});
|
2020-12-09 16:05:12 +01:00
|
|
|
return getCliForm().ask({
|
|
|
|
message: 'Select an organization',
|
|
|
|
type: 'list',
|
|
|
|
choices: organizations.map((org) => ({
|
|
|
|
name: `${org.name} (${org.handle})`,
|
|
|
|
value: org.handle,
|
|
|
|
})),
|
|
|
|
});
|
2017-12-20 22:46:01 +01:00
|
|
|
}
|
|
|
|
|
2023-07-20 10:47:09 +03:00
|
|
|
export async function getAndSelectOrganization() {
|
|
|
|
const { getOwnOrganizations } = await import('./sdk');
|
|
|
|
const organizations = await getOwnOrganizations(getBalenaSdk(), {
|
|
|
|
$select: ['name', 'handle'],
|
|
|
|
});
|
|
|
|
|
|
|
|
if (organizations.length === 0) {
|
|
|
|
// User is not a member of any organizations (should not happen).
|
|
|
|
throw new Error('This account is not a member of any organizations');
|
|
|
|
} else if (organizations.length === 1) {
|
|
|
|
// User is a member of only one organization - use this.
|
|
|
|
return organizations[0].handle;
|
|
|
|
} else {
|
|
|
|
// User is a member of multiple organizations -
|
|
|
|
return selectOrganization(organizations);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-01 15:26:40 +01:00
|
|
|
export async function awaitDeviceOsUpdate(
|
|
|
|
uuid: string,
|
|
|
|
targetOsVersion: string,
|
|
|
|
) {
|
2019-06-11 15:08:15 +03:00
|
|
|
const balena = getBalenaSdk();
|
|
|
|
|
2020-07-01 15:26:40 +01:00
|
|
|
const deviceName = await balena.models.device.getName(uuid);
|
|
|
|
const visuals = getVisuals();
|
|
|
|
const progressBar = new visuals.Progress(
|
|
|
|
`Updating the OS of ${deviceName} to v${targetOsVersion}`,
|
|
|
|
);
|
|
|
|
progressBar.update({ percentage: 0 });
|
|
|
|
|
|
|
|
const poll = async (): Promise<void> => {
|
2021-07-20 14:57:00 +01:00
|
|
|
const [osUpdateStatus, { overall_progress: osUpdateProgress }] =
|
|
|
|
await Promise.all([
|
|
|
|
balena.models.device.getOsUpdateStatus(uuid),
|
|
|
|
balena.models.device.get(uuid, { $select: 'overall_progress' }),
|
|
|
|
]);
|
2020-07-01 15:26:40 +01:00
|
|
|
if (osUpdateStatus.status === 'done') {
|
|
|
|
console.info(
|
|
|
|
`The device ${deviceName} has been updated to v${targetOsVersion} and will restart shortly!`,
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
2019-06-11 15:08:15 +03:00
|
|
|
|
2020-07-01 15:26:40 +01:00
|
|
|
if (osUpdateStatus.error) {
|
2021-08-27 00:49:54 +01:00
|
|
|
throw new ExpectedError(
|
|
|
|
`Failed to complete Host OS update on device ${deviceName}\n${osUpdateStatus.error}`,
|
2020-07-01 15:26:40 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (osUpdateProgress !== null) {
|
|
|
|
// Avoid resetting to 0% at end of process when device goes offline.
|
|
|
|
progressBar.update({ percentage: osUpdateProgress });
|
|
|
|
}
|
|
|
|
|
|
|
|
await delay(3000);
|
|
|
|
await poll();
|
|
|
|
};
|
|
|
|
|
|
|
|
await poll();
|
|
|
|
return uuid;
|
2019-06-11 15:08:15 +03:00
|
|
|
}
|
|
|
|
|
2020-12-16 16:57:25 +01:00
|
|
|
/*
|
2022-07-15 15:01:37 +00:00
|
|
|
* Given fleetOrDevice, which may be
|
|
|
|
* - a fleet name
|
|
|
|
* - a fleet slug
|
2020-12-16 16:57:25 +01:00
|
|
|
* - a device uuid
|
|
|
|
* Either:
|
|
|
|
* - in case of device uuid, return uuid of device after verifying that it exists and is online.
|
2022-07-15 15:01:37 +00:00
|
|
|
* - in case of fleet, return uuid of device user selects from list of online devices.
|
2020-12-16 16:57:25 +01:00
|
|
|
*/
|
|
|
|
export async function getOnlineTargetDeviceUuid(
|
2021-12-19 23:04:16 +00:00
|
|
|
sdk: BalenaSDK,
|
2022-07-15 15:01:37 +00:00
|
|
|
fleetOrDevice: string,
|
2020-08-25 13:45:39 +02:00
|
|
|
) {
|
2020-12-16 16:57:25 +01:00
|
|
|
const logger = (await import('../utils/logger')).getLogger();
|
|
|
|
|
|
|
|
// If looks like UUID, probably device
|
2022-07-15 15:01:37 +00:00
|
|
|
if (validation.validateUuid(fleetOrDevice)) {
|
2020-12-16 16:57:25 +01:00
|
|
|
let device: Device;
|
2020-08-25 13:45:39 +02:00
|
|
|
try {
|
2020-12-16 16:57:25 +01:00
|
|
|
logger.logDebug(
|
2022-07-15 15:01:37 +00:00
|
|
|
`Trying to fetch device by UUID ${fleetOrDevice} (${typeof fleetOrDevice})`,
|
2020-12-16 16:57:25 +01:00
|
|
|
);
|
2022-07-15 15:01:37 +00:00
|
|
|
device = await sdk.models.device.get(fleetOrDevice, {
|
2020-12-16 16:57:25 +01:00
|
|
|
$select: ['uuid', 'is_online'],
|
|
|
|
});
|
2020-08-25 13:45:39 +02:00
|
|
|
|
2020-12-16 16:57:25 +01:00
|
|
|
if (!device.is_online) {
|
2022-07-15 15:01:37 +00:00
|
|
|
throw new ExpectedError(`Device with UUID ${fleetOrDevice} is offline`);
|
2020-12-16 16:57:25 +01:00
|
|
|
}
|
2019-06-06 11:24:44 +01:00
|
|
|
|
2020-12-16 16:57:25 +01:00
|
|
|
return device.uuid;
|
|
|
|
} catch (err) {
|
|
|
|
const { BalenaDeviceNotFound } = await import('balena-errors');
|
|
|
|
if (instanceOf(err, BalenaDeviceNotFound)) {
|
2022-07-15 15:01:37 +00:00
|
|
|
logger.logDebug(`Device with UUID ${fleetOrDevice} not found`);
|
|
|
|
// Now try application
|
2020-12-16 16:57:25 +01:00
|
|
|
} else {
|
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
}
|
2019-06-06 11:24:44 +01:00
|
|
|
}
|
|
|
|
|
2022-07-15 15:01:37 +00:00
|
|
|
// Not a device UUID, try application
|
2023-05-19 20:59:24 +03:00
|
|
|
const application = await (async () => {
|
|
|
|
try {
|
|
|
|
logger.logDebug(`Fetching fleet ${fleetOrDevice}`);
|
|
|
|
const { getApplication } = await import('./sdk');
|
2023-05-20 00:20:24 +03:00
|
|
|
return await getApplication(sdk, fleetOrDevice, {
|
|
|
|
$select: ['id', 'slug'],
|
2023-05-23 19:14:27 +03:00
|
|
|
$expand: {
|
|
|
|
owns__device: {
|
|
|
|
$select: ['device_name', 'uuid'],
|
|
|
|
$filter: { is_online: true },
|
|
|
|
},
|
|
|
|
},
|
2023-05-20 00:20:24 +03:00
|
|
|
});
|
2023-05-19 20:59:24 +03:00
|
|
|
} catch (err) {
|
|
|
|
const { BalenaApplicationNotFound } = await import('balena-errors');
|
|
|
|
if (instanceOf(err, BalenaApplicationNotFound)) {
|
|
|
|
throw new ExpectedError(`Fleet or Device not found: ${fleetOrDevice}`);
|
|
|
|
} else {
|
|
|
|
throw err;
|
|
|
|
}
|
2019-06-06 11:24:44 +01:00
|
|
|
}
|
2023-05-19 20:59:24 +03:00
|
|
|
})();
|
2019-06-06 11:24:44 +01:00
|
|
|
|
2020-12-16 16:57:25 +01:00
|
|
|
// App found, load its devices
|
2023-05-23 19:14:27 +03:00
|
|
|
const devices = application.owns__device;
|
2020-12-16 16:57:25 +01:00
|
|
|
|
|
|
|
// Throw if no devices online
|
2023-07-07 20:21:44 +03:00
|
|
|
if (!devices.length) {
|
2020-12-16 16:57:25 +01:00
|
|
|
throw new ExpectedError(
|
2022-07-15 15:01:37 +00:00
|
|
|
`Fleet ${application.slug} found, but has no devices online.`,
|
2020-12-16 16:57:25 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-07-15 15:01:37 +00:00
|
|
|
// Ask user to select from online devices for fleet
|
2020-12-16 16:57:25 +01:00
|
|
|
return getCliForm().ask({
|
2022-07-15 15:01:37 +00:00
|
|
|
message: `Select a device on fleet ${application.slug}`,
|
2020-12-16 16:57:25 +01:00
|
|
|
type: 'list',
|
|
|
|
default: devices[0].uuid,
|
2023-07-07 20:21:44 +03:00
|
|
|
choices: devices.map((device) => ({
|
2020-12-16 16:57:25 +01:00
|
|
|
name: `${device.device_name || 'Untitled'} (${device.uuid.slice(0, 7)})`,
|
|
|
|
value: device.uuid,
|
|
|
|
})),
|
|
|
|
});
|
2019-06-06 11:24:44 +01:00
|
|
|
}
|
|
|
|
|
2018-05-22 18:12:51 +03:00
|
|
|
export function selectFromList<T>(
|
|
|
|
message: string,
|
|
|
|
choices: Array<T & { name: string }>,
|
2020-07-01 15:26:40 +01:00
|
|
|
): Promise<T> {
|
2020-07-08 18:03:10 +01:00
|
|
|
return getCliForm().ask<T>({
|
2018-04-25 15:20:07 +01:00
|
|
|
message,
|
|
|
|
type: 'list',
|
2023-07-07 20:21:44 +03:00
|
|
|
choices: choices.map((s) => ({
|
2018-04-25 15:20:07 +01:00
|
|
|
name: s.name,
|
|
|
|
value: s,
|
|
|
|
})),
|
|
|
|
});
|
|
|
|
}
|