mirror of
https://github.com/balena-io/balena-cli.git
synced 2025-01-11 15:33:06 +00:00
7c62e34455
This also avoids current unnecessary requiring of resin-cli-visuals for every command Change-type: patch
350 lines
11 KiB
CoffeeScript
350 lines
11 KiB
CoffeeScript
###
|
|
Copyright 2016-2018 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.
|
|
###
|
|
|
|
commandOptions = require('./command-options')
|
|
{ normalizeUuidProp } = require('../utils/normalization')
|
|
{ getBalenaSdk, getVisuals } = require('../utils/lazy')
|
|
|
|
exports.read =
|
|
signature: 'config read'
|
|
description: 'read a device configuration'
|
|
help: '''
|
|
Use this command to read the config.json file from the mounted filesystem (e.g. SD card) of a provisioned device"
|
|
|
|
Examples:
|
|
|
|
$ balena config read --type raspberry-pi
|
|
$ balena config read --type raspberry-pi --drive /dev/disk2
|
|
'''
|
|
options: [
|
|
{
|
|
signature: 'type'
|
|
description: 'device type (Check available types with `balena devices supported`)'
|
|
parameter: 'type'
|
|
alias: 't'
|
|
required: 'You have to specify a device type'
|
|
}
|
|
{
|
|
signature: 'drive'
|
|
description: 'drive'
|
|
parameter: 'drive'
|
|
alias: 'd'
|
|
}
|
|
]
|
|
permission: 'user'
|
|
root: true
|
|
action: (params, options) ->
|
|
Promise = require('bluebird')
|
|
config = require('balena-config-json')
|
|
umountAsync = Promise.promisify(require('umount').umount)
|
|
prettyjson = require('prettyjson')
|
|
|
|
Promise.try ->
|
|
return options.drive or getVisuals().drive('Select the device drive')
|
|
.tap(umountAsync)
|
|
.then (drive) ->
|
|
return config.read(drive, options.type)
|
|
.tap (configJSON) ->
|
|
console.info(prettyjson.render(configJSON))
|
|
|
|
exports.write =
|
|
signature: 'config write <key> <value>'
|
|
description: 'write a device configuration'
|
|
help: '''
|
|
Use this command to write the config.json file to the mounted filesystem (e.g. SD card) of a provisioned device
|
|
|
|
Examples:
|
|
|
|
$ balena config write --type raspberry-pi username johndoe
|
|
$ balena config write --type raspberry-pi --drive /dev/disk2 username johndoe
|
|
$ balena config write --type raspberry-pi files.network/settings "..."
|
|
'''
|
|
options: [
|
|
{
|
|
signature: 'type'
|
|
description: 'device type (Check available types with `balena devices supported`)'
|
|
parameter: 'type'
|
|
alias: 't'
|
|
required: 'You have to specify a device type'
|
|
}
|
|
{
|
|
signature: 'drive'
|
|
description: 'drive'
|
|
parameter: 'drive'
|
|
alias: 'd'
|
|
}
|
|
]
|
|
permission: 'user'
|
|
root: true
|
|
action: (params, options) ->
|
|
Promise = require('bluebird')
|
|
_ = require('lodash')
|
|
config = require('balena-config-json')
|
|
umountAsync = Promise.promisify(require('umount').umount)
|
|
|
|
Promise.try ->
|
|
return options.drive or getVisuals().drive('Select the device drive')
|
|
.tap(umountAsync)
|
|
.then (drive) ->
|
|
config.read(drive, options.type).then (configJSON) ->
|
|
console.info("Setting #{params.key} to #{params.value}")
|
|
_.set(configJSON, params.key, params.value)
|
|
return configJSON
|
|
.tap ->
|
|
return umountAsync(drive)
|
|
.then (configJSON) ->
|
|
return config.write(drive, options.type, configJSON)
|
|
.tap ->
|
|
console.info('Done')
|
|
|
|
exports.inject =
|
|
signature: 'config inject <file>'
|
|
description: 'inject a device configuration file'
|
|
help: '''
|
|
Use this command to inject a config.json file to the mounted filesystem
|
|
(e.g. SD card or mounted balenaOS image) of a provisioned device"
|
|
|
|
Examples:
|
|
|
|
$ balena config inject my/config.json --type raspberry-pi
|
|
$ balena config inject my/config.json --type raspberry-pi --drive /dev/disk2
|
|
'''
|
|
options: [
|
|
{
|
|
signature: 'type'
|
|
description: 'device type (Check available types with `balena devices supported`)'
|
|
parameter: 'type'
|
|
alias: 't'
|
|
required: 'You have to specify a device type'
|
|
}
|
|
{
|
|
signature: 'drive'
|
|
description: 'drive'
|
|
parameter: 'drive'
|
|
alias: 'd'
|
|
}
|
|
]
|
|
permission: 'user'
|
|
root: true
|
|
action: (params, options) ->
|
|
Promise = require('bluebird')
|
|
config = require('balena-config-json')
|
|
umountAsync = Promise.promisify(require('umount').umount)
|
|
readFileAsync = Promise.promisify(require('fs').readFile)
|
|
|
|
Promise.try ->
|
|
return options.drive or getVisuals().drive('Select the device drive')
|
|
.tap(umountAsync)
|
|
.then (drive) ->
|
|
readFileAsync(params.file, 'utf8').then(JSON.parse).then (configJSON) ->
|
|
return config.write(drive, options.type, configJSON)
|
|
.tap ->
|
|
console.info('Done')
|
|
|
|
exports.reconfigure =
|
|
signature: 'config reconfigure'
|
|
description: 'reconfigure a provisioned device'
|
|
help: '''
|
|
Use this command to reconfigure a provisioned device
|
|
|
|
Examples:
|
|
|
|
$ balena config reconfigure --type raspberry-pi
|
|
$ balena config reconfigure --type raspberry-pi --advanced
|
|
$ balena config reconfigure --type raspberry-pi --drive /dev/disk2
|
|
'''
|
|
options: [
|
|
{
|
|
signature: 'type'
|
|
description: 'device type (Check available types with `balena devices supported`)'
|
|
parameter: 'type'
|
|
alias: 't'
|
|
required: 'You have to specify a device type'
|
|
}
|
|
{
|
|
signature: 'drive'
|
|
description: 'drive'
|
|
parameter: 'drive'
|
|
alias: 'd'
|
|
}
|
|
{
|
|
signature: 'advanced'
|
|
description: 'show advanced commands'
|
|
boolean: true
|
|
alias: 'v'
|
|
}
|
|
]
|
|
permission: 'user'
|
|
root: true
|
|
action: (params, options) ->
|
|
Promise = require('bluebird')
|
|
config = require('balena-config-json')
|
|
{ runCommand } = require('../utils/helpers')
|
|
umountAsync = Promise.promisify(require('umount').umount)
|
|
|
|
Promise.try ->
|
|
return options.drive or getVisuals().drive('Select the device drive')
|
|
.tap(umountAsync)
|
|
.then (drive) ->
|
|
config.read(drive, options.type).get('uuid')
|
|
.tap ->
|
|
umountAsync(drive)
|
|
.then (uuid) ->
|
|
configureCommand = "os configure #{drive} --device #{uuid}"
|
|
if options.advanced
|
|
configureCommand += ' --advanced'
|
|
return runCommand(configureCommand)
|
|
.then ->
|
|
console.info('Done')
|
|
|
|
exports.generate =
|
|
signature: 'config generate'
|
|
description: 'generate a config.json file'
|
|
help: '''
|
|
Use this command to generate a config.json for a device or application.
|
|
|
|
Calling this command with the exact version number of the targeted image is required.
|
|
|
|
This is interactive by default, but you can do this automatically without interactivity
|
|
by specifying an option for each question on the command line, if you know the questions
|
|
that will be asked for the relevant device type.
|
|
|
|
In case that you want to configure an image for an application with mixed device types,
|
|
you can pass the --device-type argument along with --app to specify the target device type.
|
|
|
|
Examples:
|
|
|
|
$ balena config generate --device 7cf02a6 --version 2.12.7
|
|
$ balena config generate --device 7cf02a6 --version 2.12.7 --generate-device-api-key
|
|
$ balena config generate --device 7cf02a6 --version 2.12.7 --device-api-key <existingDeviceKey>
|
|
$ balena config generate --device 7cf02a6 --version 2.12.7 --output config.json
|
|
$ balena config generate --app MyApp --version 2.12.7
|
|
$ balena config generate --app MyApp --version 2.12.7 --device-type fincm3
|
|
$ balena config generate --app MyApp --version 2.12.7 --output config.json
|
|
$ balena config generate --app MyApp --version 2.12.7 \
|
|
--network wifi --wifiSsid mySsid --wifiKey abcdefgh --appUpdatePollInterval 1
|
|
'''
|
|
options: [
|
|
commandOptions.osVersion
|
|
commandOptions.optionalApplication
|
|
commandOptions.optionalDevice
|
|
commandOptions.optionalDeviceApiKey
|
|
commandOptions.optionalDeviceType
|
|
{
|
|
signature: 'generate-device-api-key'
|
|
description: 'generate a fresh device key for the device'
|
|
boolean: true
|
|
}
|
|
{
|
|
signature: 'output'
|
|
description: 'output'
|
|
parameter: 'output'
|
|
alias: 'o'
|
|
}
|
|
# Options for non-interactive configuration
|
|
{
|
|
signature: 'network'
|
|
description: 'the network type to use: ethernet or wifi'
|
|
parameter: 'network'
|
|
}
|
|
{
|
|
signature: 'wifiSsid'
|
|
description: 'the wifi ssid to use (used only if --network is set to wifi)'
|
|
parameter: 'wifiSsid'
|
|
}
|
|
{
|
|
signature: 'wifiKey'
|
|
description: 'the wifi key to use (used only if --network is set to wifi)'
|
|
parameter: 'wifiKey'
|
|
}
|
|
{
|
|
signature: 'appUpdatePollInterval'
|
|
description: 'how frequently (in minutes) to poll for application updates'
|
|
parameter: 'appUpdatePollInterval'
|
|
}
|
|
]
|
|
permission: 'user'
|
|
action: (params, options) ->
|
|
normalizeUuidProp(options, 'device')
|
|
Promise = require('bluebird')
|
|
writeFileAsync = Promise.promisify(require('fs').writeFile)
|
|
balena = getBalenaSdk()
|
|
form = require('resin-cli-form')
|
|
prettyjson = require('prettyjson')
|
|
|
|
{ generateDeviceConfig, generateApplicationConfig } = require('../utils/config')
|
|
helpers = require('../utils/helpers')
|
|
{ exitWithExpectedError } = require('../utils/patterns')
|
|
|
|
if not options.device? and not options.application?
|
|
exitWithExpectedError '''
|
|
You have to pass either a device or an application.
|
|
|
|
See the help page for examples:
|
|
|
|
$ balena help config generate
|
|
'''
|
|
|
|
if !options.application and options.deviceType
|
|
exitWithExpectedError '''
|
|
Specifying a different device type is only supported when
|
|
generating a config for an application:
|
|
|
|
* An application, with --app <appname>
|
|
* A specific device type, with --device-type <deviceTypeSlug>
|
|
|
|
See the help page for examples:
|
|
|
|
$ balena help config generate
|
|
'''
|
|
|
|
Promise.try ->
|
|
if options.device?
|
|
return balena.models.device.get(options.device)
|
|
return balena.models.application.get(options.application)
|
|
.then (resource) ->
|
|
deviceType = options.deviceType || resource.device_type
|
|
manifestPromise = balena.models.device.getManifestBySlug(deviceType)
|
|
|
|
if options.application && options.deviceType
|
|
app = resource
|
|
appManifestPromise = balena.models.device.getManifestBySlug(app.device_type)
|
|
manifestPromise = manifestPromise.tap (paramDeviceType) ->
|
|
appManifestPromise.then (appDeviceType) ->
|
|
if not helpers.areDeviceTypesCompatible(appDeviceType, paramDeviceType)
|
|
throw new balena.errors.BalenaInvalidDeviceType(
|
|
"Device type #{options.deviceType} is incompatible with application #{options.application}"
|
|
)
|
|
|
|
manifestPromise.get('options')
|
|
.then (formOptions) ->
|
|
# Pass params as an override: if there is any param with exactly the same name as a
|
|
# required option, that value is used (and the corresponding question is not asked)
|
|
form.run(formOptions, override: options)
|
|
.then (answers) ->
|
|
answers.version = options.version
|
|
|
|
if resource.uuid?
|
|
generateDeviceConfig(resource, options.deviceApiKey || options['generate-device-api-key'], answers)
|
|
else
|
|
answers.deviceType = deviceType
|
|
generateApplicationConfig(resource, answers)
|
|
.then (config) ->
|
|
if options.output?
|
|
return writeFileAsync(options.output, JSON.stringify(config))
|
|
|
|
console.log(prettyjson.render(config))
|