Interactive Shell for Remote Systems
Based on the Train ecosystem, provide a shell to manage systems via a multitude of transports.
Train default supports:
- Docker
- Local
- SSH (Unix, Windows, Cisco)
- WinRM (Windows)
3rd party plugins support:
- AWS Systems Manager
- LXD
- Telnet
- Serial/USB interfaces
- VMware Guest Operations (VMware Tools)
- ...
When specifying a password within an URL take care of special characters or the connection will fail.
trainsh connect local://
trainsh connect winrm://Administrator:[email protected]
trainsh connect awsssm://i-123456789abc0
trainsh connect vsphere-gom://example-vm
If the transport has additional options like key_files
, they need to be added as URL parameters (ssh://10.20.30.40/?key_files=~/.ssh/id_rsa2
).
There are three separate categories of commands when in the Shell:
- remote commands
- local commands, which are prefixed with
.
- built-in commands, wich are prefixed with
!
Anything you type without a prefix gets executed 1:1 on the remote system. Please notice that Train is headless, so any interactive programs will not work but lock up your shell! If you need to edit remote files or use a pager (like less
), please look into the Built-In Commands section.
Local commands get executed 1:1 on your local system, so you are able to check things locally or change paths, for example.
There is also the Sessionprefix @
, which is described briefly in the "Prompt and Sessions" section.
!!!
Quit TrainSH. Aliases: quit
, exit
, logout
, disconnect
.
!clear-history
Clear your TrainSH history, for example to remove clutter or sensitive information
!connect <uri>
Connect to another system. The URI needs to match the format of the used Train transport, which is usually transportname://host
but varies. See the Train transport's documentation for details.
!copy @<session>:/<path> @<session>:/<path>
Copy a file between two established sessions.
!detect
Re-runs the OS detection which is running automatically on start. This will determine the OS, OS-family and general platform information via Train.
!download <remote> <local>
Download a file to the local system. You need to specify the target, if you want to keep the name just use .
as local part.
!edit <remotefile>
Downloads the remote file as temporary file and opens the system default editor locally (EDITOR
and VISUAL
environment variables, with fallback to vi
). Upon exiting the file, it will be uploaded again and overwrite the remote file.
!env
Prints the environment variables of your remote shell. This will be filled on first command invocation to save IO. Currently unsupported for Windows remote systems
!help
Print out help
!history
Output your TrainSH command history. As this uses the popular Readline library, you can also navigate your history with the Up/Down arrows or use Ctrl-R for reverse search. You can do auto completion for built-in commands.
!host
Outputs the remote hostname.
!ping
Executes a simple command to measure roundtrip/overhead time.
!pwd
Output the remote working directory. This will be filled on first command invocation to save IO.
!read <remotefile>
Download the specified file and display it in the system default pager (PAGER
environment variable, with fallback to less
).
!reconnect
Quits and reopens the current session.
!sessions
List all currently active sessions (See "Prompt and Sessions" section). Password information is redacted for security reasons.
!session <session_id>
Change to another session by passing its ID (=number)
!upload <local> <remote>
Upload files to the target machine
OK trainsh(@0 local://trc4023)>
The prompt consists roughly of three areas:
- Exit code of last command ("OK" or the exit code in format
Exx
with xx being the numeric code) - Session indication
- Input area
The session indication includes the ID of the session, so @0
means session 0, @1
means session 1 and so on. The second part shows the Train URI for the remote system for easy identification.
Any TrainSH invocation has at minimum one active session but if you want, you can add more sessions with the !connect <uri>
built-in. Every session has its own Train backend and storage of current working directory, environment variables etc. You can list active sessions with !sessions
and switch between them using !session <session_id>
.
If you want to execute commands on another session ad-hoc, you can prefix it with the session id: @1 uname -a
will execute the uname -a
command on session ID 1.
Depending on the transport and target, the time between sending a command and retrieving the result will vary widely. In TrainSH this is called the "ping", which measures the minimum overhead/latency for the session. It varies massively between Train plugins and is not only related to the network distance, but also to the way of command execution - which might involve a number of HTTPS requests or the time to invoke a remote shell (examples: AWS SSM and VSphere GOM transports).
As Train is headless/stateless system, there will be various pre- and postfixed commands to make your life easier. This means that any state-related commands like changing the current directory or modification of environment variables would be unavailable for the next command - as it technically is a separate shell.
To make this easier, internal commands get attached to your input like this:
- Prefix: Get remote hostname (discovery task, just on first execution)
- Prefix: Set previous environment variables
- Prefix: Set previous path
- Command
- Postfix: Retrieve and save exit code of command
- Postfix: Retrieve and save new path
- Postfix: Retrieve and save new environment variables
Output of commands gets separated by outputting a highly random string between, which should not result in false positives. If a false positive occurs for some reason, TrainSH will fail and output an error.
As providing target URLs to connect to can be tedious, TrainSH will detect targets to connect to via plugins. In these cases, trainsh connect
does not need any parameters.
This will check the TARGET
environment variable for a URL and use it to connect. Only one target is allowed.
This will detect if the current directory has a Test Kitchen configuration and a created machine. If so, it will connect to the machine by parsing information in .kitchen/
and the kitchen configuration file.