mirror of
https://github.com/balena-io/balena-cli.git
synced 2025-01-04 13:04:09 +00:00
b4439b7d78
The suggestion happens if the UUID supplied is not found. Because of that function, it's impossible to do an atomic connect to a device in non-interactive mode. The auto-suggestion results connecting to the first available device, which is likely not the intended action. The current workaround is running a `balena device UUID` and check its exit code before running `balena ssh UUID`, but since these are independent steps, still can connect to another device, if between the two commands anything changes. With this flag used, one could never connect accidentally to the wrong device due to suggestions. Change-type: minor Signed-off-by: Gergely Imreh <gergely@balena.io>
150 lines
4.8 KiB
CoffeeScript
150 lines
4.8 KiB
CoffeeScript
###
|
|
Copyright 2016-2017 Balena
|
|
|
|
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')
|
|
|
|
module.exports =
|
|
signature: 'ssh [uuid]'
|
|
description: '(beta) get a shell into the running app container of a device'
|
|
help: '''
|
|
Warning: 'balena ssh' requires an openssh-compatible client to be correctly
|
|
installed in your shell environment. For more information (including Windows
|
|
support) please check the README here: https://github.com/balena-io/balena-cli
|
|
|
|
Use this command to get a shell into the running application container of
|
|
your device.
|
|
|
|
Examples:
|
|
|
|
$ balena ssh MyApp
|
|
$ balena ssh 7cf02a6
|
|
$ balena ssh 7cf02a6 --port 8080
|
|
$ balena ssh 7cf02a6 -v
|
|
$ balena ssh 7cf02a6 -s
|
|
$ balena ssh 7cf02a6 --noninteractive
|
|
'''
|
|
permission: 'user'
|
|
primary: true
|
|
options: [
|
|
signature: 'port'
|
|
parameter: 'port'
|
|
description: 'ssh gateway port'
|
|
alias: 'p'
|
|
,
|
|
signature: 'verbose'
|
|
boolean: true
|
|
description: 'increase verbosity'
|
|
alias: 'v'
|
|
commandOptions.hostOSAccess,
|
|
signature: 'noproxy'
|
|
boolean: true
|
|
description: "don't use the proxy configuration for this connection.
|
|
Only makes sense if you've configured proxy globally."
|
|
,
|
|
signature: 'noninteractive'
|
|
boolean: true
|
|
description: 'run command non-interactively, do not automatically suggest devices to connect to if UUID not found'
|
|
]
|
|
action: (params, options, done) ->
|
|
normalizeUuidProp(params)
|
|
child_process = require('child_process')
|
|
Promise = require('bluebird')
|
|
balena = require('balena-sdk').fromSharedOptions()
|
|
_ = require('lodash')
|
|
bash = require('bash')
|
|
hasbin = require('hasbin')
|
|
{ getSubShellCommand } = require('../utils/helpers')
|
|
patterns = require('../utils/patterns')
|
|
|
|
options.port ?= 22
|
|
|
|
verbose = if options.verbose then '-vvv' else ''
|
|
|
|
proxyConfig = global.PROXY_CONFIG
|
|
useProxy = !!proxyConfig and not options.noproxy
|
|
|
|
getSshProxyCommand = (hasTunnelBin) ->
|
|
return '' if not useProxy
|
|
|
|
if not hasTunnelBin
|
|
console.warn('''
|
|
Proxy is enabled but the `proxytunnel` binary cannot be found.
|
|
Please install it if you want to route the `balena ssh` requests through the proxy.
|
|
Alternatively you can pass `--noproxy` param to the `balena ssh` command to ignore the proxy config
|
|
for the `ssh` requests.
|
|
|
|
Attemmpting the unproxied request for now.
|
|
''')
|
|
return ''
|
|
|
|
tunnelOptions =
|
|
proxy: "#{proxyConfig.host}:#{proxyConfig.port}"
|
|
dest: '%h:%p'
|
|
{ proxyAuth } = proxyConfig
|
|
if proxyAuth
|
|
i = proxyAuth.indexOf(':')
|
|
_.assign tunnelOptions,
|
|
user: proxyAuth.substring(0, i)
|
|
pass: proxyAuth.substring(i + 1)
|
|
proxyCommand = "proxytunnel #{bash.args(tunnelOptions, '--', '=')}"
|
|
return "-o #{bash.args({ ProxyCommand: proxyCommand }, '', '=')}"
|
|
|
|
Promise.try ->
|
|
return false if not params.uuid
|
|
return balena.models.device.has(params.uuid)
|
|
.then (uuidExists) ->
|
|
return params.uuid if uuidExists
|
|
if options.noninteractive
|
|
console.error("Could not find device: #{params.uuid}")
|
|
process.exit(1)
|
|
return patterns.inferOrSelectDevice()
|
|
.then (uuid) ->
|
|
console.info("Connecting to: #{uuid}")
|
|
balena.models.device.get(uuid)
|
|
.then (device) ->
|
|
patterns.exitWithExpectedError('Device is not online') if not device.is_online
|
|
|
|
Promise.props
|
|
username: balena.auth.whoami()
|
|
uuid: device.uuid
|
|
# get full uuid
|
|
containerId: if options.host then '' else balena.models.device.getApplicationInfo(device.uuid).get('containerId')
|
|
proxyUrl: balena.settings.get('proxyUrl')
|
|
|
|
hasTunnelBin: if useProxy then hasbin('proxytunnel') else null
|
|
.then ({ username, uuid, containerId, proxyUrl, hasTunnelBin }) ->
|
|
throw new Error('Did not find running application container') if not containerId?
|
|
Promise.try ->
|
|
sshProxyCommand = getSshProxyCommand(hasTunnelBin)
|
|
|
|
if options.host
|
|
accessCommand = "host #{uuid}"
|
|
else
|
|
accessCommand = "enter #{uuid} #{containerId}"
|
|
|
|
command = "ssh #{verbose} -t \
|
|
-o LogLevel=ERROR \
|
|
-o StrictHostKeyChecking=no \
|
|
-o UserKnownHostsFile=/dev/null \
|
|
#{sshProxyCommand} \
|
|
-p #{options.port} #{username}@ssh.#{proxyUrl} #{accessCommand}"
|
|
|
|
subShellCommand = getSubShellCommand(command)
|
|
child_process.spawn subShellCommand.program, subShellCommand.args,
|
|
stdio: 'inherit'
|
|
.nodeify(done)
|