/** * @license * Copyright 2019 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 * as BalenaSdk from 'balena-sdk'; import { stripIndent } from 'common-tags'; import { runCommand } from './helpers'; import Logger = require('./logger'); import { exec, execBuffered } from './ssh'; const MIN_BALENAOS_VERSION = 'v2.14.0'; export async function join( logger: Logger, sdk: BalenaSdk.BalenaSDK, deviceHostnameOrIp?: string, appName?: string, ): Promise { logger.logDebug('Checking login...'); const isLoggedIn = await sdk.auth.isLoggedIn(); if (!isLoggedIn) { logger.logInfo("Looks like you're not logged in yet!"); logger.logInfo("Let's go through a quick wizard to get you started.\n"); await runCommand('login'); } logger.logDebug('Determining device...'); const deviceIp = await getOrSelectLocalDevice(deviceHostnameOrIp); await assertDeviceIsCompatible(deviceIp); logger.logDebug(`Using device: ${deviceIp}`); logger.logDebug('Determining device type...'); const deviceType = await getDeviceType(deviceIp); logger.logDebug(`Device type: ${deviceType}`); logger.logDebug('Determining application...'); const app = await getOrSelectApplication(sdk, deviceType, appName); logger.logDebug(`Using application: ${app.app_name} (${app.device_type})`); if (app.device_type !== deviceType) { logger.logDebug(`Forcing device type to: ${deviceType}`); app.device_type = deviceType; } logger.logDebug('Determining device OS version...'); const deviceOsVersion = await getOsVersion(deviceIp); logger.logDebug(`Device OS version: ${deviceOsVersion}`); logger.logDebug('Generating application config...'); const config = await generateApplicationConfig(sdk, app, { version: deviceOsVersion, }); logger.logDebug(`Using config: ${JSON.stringify(config, null, 2)}`); logger.logDebug('Configuring...'); await configure(deviceIp, config); logger.logDebug('All done.'); const platformUrl = await sdk.settings.get('balenaUrl'); logger.logSuccess(`Device successfully joined ${platformUrl}!`); } export async function leave( logger: Logger, _sdk: BalenaSdk.BalenaSDK, deviceHostnameOrIp?: string, ): Promise { logger.logDebug('Determining device...'); const deviceIp = await getOrSelectLocalDevice(deviceHostnameOrIp); await assertDeviceIsCompatible(deviceIp); logger.logDebug(`Using device: ${deviceIp}`); logger.logDebug('Deconfiguring...'); await deconfigure(deviceIp); logger.logDebug('All done.'); logger.logSuccess('Device successfully left the platform.'); } async function execCommand( deviceIp: string, cmd: string, msg: string, ): Promise { const through = await import('through2'); const visuals = await import('resin-cli-visuals'); const spinner = new visuals.Spinner(`[${deviceIp}] Connecting...`); const innerSpinner = spinner.spinner; const stream = through(function(data, _enc, cb) { innerSpinner.setSpinnerTitle(`%s [${deviceIp}] ${msg}`); cb(null, data); }); spinner.start(); await exec(deviceIp, cmd, stream); spinner.stop(); } async function configure(deviceIp: string, config: any): Promise { // Passing the JSON is slightly tricky due to the many layers of indirection // so we just base64-encode it here and decode it at the other end, when invoking // os-config. const json = JSON.stringify(config); const b64 = Buffer.from(json).toString('base64'); const str = `"$(base64 -d <<< ${b64})"`; await execCommand(deviceIp, `os-config join '${str}'`, 'Configuring...'); } async function deconfigure(deviceIp: string): Promise { await execCommand(deviceIp, 'os-config leave', 'Configuring...'); } async function assertDeviceIsCompatible(deviceIp: string): Promise { const { exitWithExpectedError } = await import('../utils/patterns'); try { await execBuffered(deviceIp, 'os-config --version'); } catch (err) { exitWithExpectedError(stripIndent` Device "${deviceIp}" is incompatible and cannot join or leave an application. Please select or provision device with balenaOS newer than ${MIN_BALENAOS_VERSION}.`); } } async function getDeviceType(deviceIp: string): Promise { const output = await execBuffered(deviceIp, 'cat /etc/os-release'); const match = /^SLUG="([^"]+)"$/m.exec(output); if (!match) { throw new Error('Failed to determine device type'); } return match[1]; } async function getOsVersion(deviceIp: string): Promise { const output = await execBuffered(deviceIp, 'cat /etc/os-release'); const match = /^VERSION_ID="([^"]+)"$/m.exec(output); if (!match) { throw new Error('Failed to determine OS version ID'); } return match[1]; } async function getOrSelectLocalDevice(deviceIp?: string): Promise { if (deviceIp) { return deviceIp; } const through = await import('through2'); let ip: string | null = null; const stream = through(function(data, _enc, cb) { const match = /^==> Selected device: (.*)$/m.exec(data.toString()); if (match) { ip = match[1]; cb(); } else { cb(null, data); } }); stream.pipe(process.stderr); const { sudo } = await import('../utils/helpers'); const command = process.argv.slice(0, 2).concat(['internal', 'scandevices']); await sudo(command, { stderr: stream, msg: 'Scanning for local devices. If asked, please type your computer password.', }); if (!ip) { throw new Error('No device selected'); } return ip; } async function selectAppFromList(applications: BalenaSdk.Application[]) { const _ = await import('lodash'); const { selectFromList } = await import('../utils/patterns'); // Present a list to the user which shows the fully qualified application // name (user/appname) and allows them to select. return selectFromList( 'Select application', _.map(applications, app => { return _.merge({ name: app.slug }, app); }), ); } async function getOrSelectApplication( sdk: BalenaSdk.BalenaSDK, deviceType: string, appName?: string, ): Promise { const _ = await import('lodash'); const form = await import('resin-cli-form'); const allDeviceTypes = await sdk.models.config.getDeviceTypes(); const deviceTypeManifest = _.find(allDeviceTypes, { slug: deviceType }); if (!deviceTypeManifest) { throw new Error(`"${deviceType}" is not a valid device type`); } const compatibleDeviceTypes = _(allDeviceTypes) .filter( dt => sdk.models.os.isArchitectureCompatibleWith( deviceTypeManifest.arch, dt.arch, ) && !!dt.isDependent === !!deviceTypeManifest.isDependent && dt.state !== 'DISCONTINUED', ) .map(type => type.slug) .value(); if (!appName) { return createOrSelectAppOrExit( form, sdk, compatibleDeviceTypes, deviceType, ); } const options: BalenaSdk.PineOptionsFor = {}; // Check for an app of the form `user/application` and update the API query. let name: string; const match = appName.split('/'); if (match.length > 1) { // These will match at most one app options.$filter = { slug: appName.toLowerCase() }; name = match[1]; } else { // We're given an application; resolve it if it's ambiguous and also validate // it's of appropriate device type. options.$filter = { app_name: appName }; name = appName; } const applications = await sdk.models.application.getAll(options); if (applications.length === 0) { const shouldCreateApp = await form.ask({ message: `No application found with name "${appName}".\n` + 'Would you like to create it now?', type: 'confirm', default: true, }); if (shouldCreateApp) { return createApplication(sdk, deviceType, name); } process.exit(1); } // We've found at least one app with the given name. // Filter out apps for non-matching device types and see what we're left with. const validApplications = applications.filter(app => _.includes(compatibleDeviceTypes, app.device_type), ); if (validApplications.length === 0) { throw new Error('No application found with a matching device type'); } if (validApplications.length === 1) { return validApplications[0]; } return selectAppFromList(applications); } // TODO: revisit this function's purpose. It was refactored out of // `getOrSelectApplication` above in order to satisfy some resin-lint v3 // rules, but it looks like there's a fair amount of duplicate logic. async function createOrSelectAppOrExit( form: any, sdk: BalenaSdk.BalenaSDK, compatibleDeviceTypes: string[], deviceType: string, ) { const options = { $filter: { device_type: { $in: compatibleDeviceTypes } }, }; // No application specified, show a list to select one. const applications = await sdk.models.application.getAll(options); if (applications.length === 0) { const shouldCreateApp = await form.ask({ message: 'You have no applications this device can join.\n' + 'Would you like to create one now?', type: 'confirm', default: true, }); if (shouldCreateApp) { return createApplication(sdk, deviceType); } process.exit(1); } return selectAppFromList(applications); } async function createApplication( sdk: BalenaSdk.BalenaSDK, deviceType: string, name?: string, ): Promise { const form = await import('resin-cli-form'); const validation = await import('./validation'); const patterns = await import('./patterns'); let username = await sdk.auth.whoami(); if (!username) { throw new sdk.errors.BalenaNotLoggedIn(); } username = username.toLowerCase(); const applicationName = await new Promise(async (resolve, reject) => { while (true) { try { const appName = await form.ask({ message: 'Enter a name for your new application:', type: 'input', default: name, validate: validation.validateApplicationName, }); try { await sdk.models.application.get(appName, { $filter: { $or: [ { slug: { $startswith: `${username}/` } }, { $not: { slug: { $contains: '/' } } }, ], }, }); patterns.printErrorMessage( 'You already have an application with that name; please choose another.', ); continue; } catch (err) { return resolve(appName); } } catch (err) { return reject(err); } } }); return sdk.models.application.create({ name: applicationName, deviceType, }); } async function generateApplicationConfig( sdk: BalenaSdk.BalenaSDK, app: BalenaSdk.Application, options: { version: string }, ) { const form = await import('resin-cli-form'); const { generateApplicationConfig: configGen } = await import('./config'); const manifest = await sdk.models.device.getManifestBySlug(app.device_type); const opts = manifest.options && manifest.options.filter(opt => opt.name !== 'network'); const values = { ...(opts ? await form.run(opts) : {}), ...options, }; const config = await configGen(app, values); if (config.connectivity === 'connman') { delete config.connectivity; delete config.files; } return config; }