The remote
CLI lets you execute long or computation-heavy tasks (e.g., compilation, integration tests, etc.)
on a powerful remote host while you work on the source code locally.
This process is known as remote execution and can enable remote build capabilities, among other things.
When you execute remote <cmd>
, it will first sync your local workspace to the remote host you selected using rsync
.
It will then execute the command <cmd>
on this host using ssh
and finally, bring all the created/modified files back to your local workspace.
remote
supports a host of configuration options to allow for complete customization of patterns for files and folders to include during the synchronization process in both directions.
The CLI supports Linux and Mac OS X operating systems with Python 3.6 or higher installed. You can also use it on Windows if you have WSL configured.
The remote host must also be running on Linux or Mac OS X. The local and remote hosts can be running different operating systems. The only requirement is that the remote host must be accessible using ssh
from the local host.
If you use Mac OS X, you can install remote
using Homebrew
from our custom tap:
brew install remote-cli/remote/remote
Then, you will always be able to update it to the latest version:
brew upgrade remote
remote
doesn't support any package managers other than brew
yet. However, it can be manually downloaded
and installed. To do it, visit https://github.com/remote-cli/remote/releases and download the latest released -shiv
archive, unpack it to some local directory (e.g., ~/.bin
) and add it to PATH:
mkdir -p ~/.bin
tar -C ~/.bin -xzf ~/Downloads/remote-1.4.5-shiv.tgz
echo 'export PATH=$PATH:/home/username/.bin/remote/bin' >> ~/.bash_profile
source ~/.bash_profile
Don't forget to replace the /home/username
above with the actual path to your home directory.
remote
CLI needs to be able to establish a passwordless SSH connection to the remote host.
Please run ssh -o BatchMode=yes <your-host> echo OK
to confirm that everything is ready for you.
If this command fails, please go through SSH guide to set up
SSH keys locally and remotely.
After you are done with the configuration, switch the working directory to the root of your workspace in
terminal and run remote-init
to create a configuration file:
cd ~/path/to/workspace
remote-init remote-host.example.com
This will create a config file named .remote.toml
in the workspace root
(~/path/to/workspace/.remote.toml
). This file controls the remote connection and synchronization options.
You can read more about this file in the Configuration section of this doc.
After it, you can start using remote:
# This will sync workspace and run './gradlew build' remotely
remote ./gradlew build
# This will forcefully push all local files to the remote machine
remote-push
# This will bring in ./build directory from the remote machine to local even if
# the CLI is configured to ignore it
remote-pull build
remote
's distribution comes with a set of executables:
remote-init
: set up a local directory to point to a remote directory on a target hostremote-ignore
: set up directories/files to ignore while pushingremote-push
: explicitly push local changes remoteremote-pull
: pull a directory from remote to localremote
: execute a command remotely, after first syncing the local tree with the remote treeremote-explain
: explain your remote setup, explain what command actually will get runremote-quick
: execute a command remotely without syncing the treesremote-add
: add another remote host to the mirror listmremote
: execute a remote command on all the hosts, after first syncing the local tree with the remote trees
You can run each of these commands with --help
flag to get a list of options and arguments they accept.
Three configuration files control the behavior of remote
:
~/.config/remote/defaults.toml
is a global config file. It sets options that affect all the workspaces unless they are overwritten by.remote.toml
file..remote.toml
is a workspace config that is expected to be placed in the root of every workspace. Theremote
CLI cannot execute any commands remotely until this file is present, or the global config overwrites this withallow_uninitiated_workspaces
option..remoteignore.toml
is a workspace config that controls only sync exclude and include patterns and has the highest priority. While the same settings can be specified in the.remote.toml
file, you can use this file to check in project-specific ignore settings in the VCS because it doesn't contain host-specific information in it.
Both configs use TOML format.
Workspace root is a root directory of the project you're working on.
It is identified by the .remote.toml
file. Each time you execute remote
from workspace root or any of its
subdirectories, remote
syncs everything under workspace root with the destination host before running the command.
Global configuration file should be placed in ~/.config/remote/defaults.toml
. This config file is optional
and the remote
CLI will work with the default values if it is absent. This is the example of how it looks like:
[general]
allow_uninitiated_workspaces = false
use_relative_remote_paths = false
remote_root = ".remotes"
[[hosts]]
host = "linux-host.example.com"
label = "linux"
[[hosts]]
host = "macos-host.example.com"
port = 2022
supports_gssapi_auth = false
default = true
label = "mac"
[push]
exclude = [".git"]
[pull]
exclude = ["src/generated"]
include = ["build/reports"]
[both]
include_vcs_ignore_patterns = true
-
[general]
block controls system-wide behavior for theremote
CLI.Reference:
-
allow_uninitiated_workspaces
(optional, defaults tofalse
) - If this flag is set totrue
and the global config contains at least one remote host,remote
will treat its current working directory as a workspace root even if it doesn't have.remote.toml
file in it.Warning: If this option is on and you run
remote
in the subdirectory of an already configured workspace,remote
will ignore workspaces configuration and treat subdirectory as a separate workspace root. -
remote_root
(optional, defaults to".remotes"
) - The default directory on the remote machine that will be used to store synced workspaces. The path is expected to be relative to the remote user's home directory, so.remotes
will resolve in/home/username/.remotes
. If the workspace-level configuration sets thedirectory
for a host, this setting will be ignored. -
use_relative_remote_paths
(optional, defaults tofalse
)- If set to
false
, all the workspaces will be stored in theremote_root
of the target host in a flat structure. Each directory will have a name like<workspace_name>_<workspace_path_hash>
. - If set to
true
, the remote path will be placed in theremote_root
tree like it was placed in the user's home directory tree locally. Some examples:- If the local path is
/home/username/projects/work/project_name
, the remote path will be/home/username/.remotes/projects/work/project_name
- If the local path is
/tmp/project_name
, the remote path will be/home/username/.remotes/tmp/project_name
- If the local path is
- If set to
-
-
[[hosts]]
block lists all the remote hosts available for the workspaces. Used when the workspace configuration doesn't overwrite it.You can provide multiple hosts in this block, but only one will be selected when you execute
remote
. It would be either the host that is marked bydefault = true
or the first one in the list if no default was set explicitly.You can run most of the commands with
--label label|number
or-l label|number
option to run a command on the non-default host.label
here is the text label you put in the config file,number
is a number of required host in the hosts' list, starting from 1.Reference:
host
- a hostname, IP address, or ssh alias of a remote machine that you want to use for remote execution.port
(optional, defaults to22
) - a port used by the ssh daemon on the host.supports_gssapi_auth
(optional, defaults totrue
) -true
if the remote host supportsgssapi-*
auth methods. We recommend disabling it if the ssh connection to the host hangs for some time during establishing.default
(optional, defaults tofalse
) -true
if this host should be used by defaultlabel
(optional) - a text label that later can be used to identify the host when running theremote
CLI.cmd_prefix
(optional) - a string which is prefixed to all commands executed via theremote
CLI. The prefix is not shell escaped.
-
[push]
,[pull]
, and[both]
blocks control what files are synced from local to a remote machine and back before and after the execution. These blocks are used when the workspace configuration doesn't overwrite them.push
block controls the files that are uploaded from local machine to the remote one.pull
block controls files that are downloaded from remote machine to local one.both
block extends previous two.Each one of these blocks supports the following options:
exclude
(optional, defaults to empty list) - a list of rsync-style patterns. Every file in the workspace that matches these patterns won't be synced unless it is explicitly specified ininclude
.include
(optional, defaults to empty list) - a list of rsync-style patterns. Every file in the workspace that matches these patterns will be synced even if it matches theexclude
.include_vcs_ignore_patterns
(optional, defaults tofalse
) - iftrue
and.gitignore
is present, all its patterns will be included in theexclude
list.
This is the example of how standalone workspace-level .remote.toml
configuration file looks like:
[[hosts]]
host = "linux-host.example.com"
directory = ".remotes/workspace"
label = "linux"
supports_gssapi_auth = true
cmd_prefix = "nice -n5"
[[hosts]]
host = "macos-host.example.com"
port = 2022
directory = ".remotes/other-workspace"
supports_gssapi_auth = false
default = true
label = "mac"
[push]
exclude = [".git"]
[pull]
exclude = ["src/generated"]
include = ["build/reports"]
[both]
include_vcs_ignore_patterns = true
All the used blocks here are similar to the ones in the global config file. However, you cannot put
[general]
block in this file. Also, you can provide one more option in [[hosts]]
block:
directory
(optional) - a path relative to remote user's home. It will be used to store the workspace's file on the remote machine.
Also, if you set at least one value for any of the blocks in the workspace-level config,
all the values from this block in the global config will be ignored.
There is a way to change this behavior. You can use [extends.*]
blocks to do it.
Here is an example. Imagine, you have a following global config:
[[hosts]]
host = "linux-host.example.com"
label = "linux"
default = true
[push]
exclude = [".git"]
[both]
include_vcs_ignore_patterns = true
If you want to be able to use the same Linux host in the workspace, but you want to add one more and modify some exclude patterns, you can create the following workspace config:
[[extends.hosts]]
host = "mac-host.example.com"
directory = ".remotes/mac-workspace"
label = "mac"
default = true
[extends.push]
exclude = ["workspace-specific-dir"]
include = [".git/hooks"]
[both]
include_vcs_ignore_patterns = false
As you can see, some block names start with extends.
. This name tells remote to merge the
workspace and global settings.
There are a few things to note:
- If both workspace-level and global configs define a default host, the workspace-level config wins
- Hosts ordering is preserved, and globally configured hosts always go first.
- If an option value is a list (e.g.
exclude
), it is extended. Otherwise, the value is overwritten.
.remoteignore.toml
files is similar to .remote.toml
, but only supports push
, pull
, both
,
extends.push
, extends.pull
and extends.both
blocks. It also cannot be used to identify
the workspace root.
Sometimes you will need to do some action each time before you execute some remote command.
A common example will be to execute pytest
in the virtual environment: you need to activate it
first, but the activation state won't be preserved between the remote
runs.
There are two ways of solving this problem:
-
Running both initiation logic and the command together:
remote 'source env/bin/activate && pytest'
-
Creating a file called
.remoteenv
in the workspace root. If this file is present,remote
will always runsource .remoteenv
on the destination host before running the actual command. For example, here is how you can runremote
's tests on the other hosts:git clone [email protected]:remote-cli/remote.git cd remote remote-init <remote-host-name> remote python3 -m venv env echo '. env/bin/activate' >> .remoteenv # starting from this point, all python commands will be executed in virtualenv remotely # This should print virtualenv's python path remote which python remote pip install -e . remote pip install -r test_requirements.txt remote pytest
The
.remoteenv
file is guaranteed to sync to remote machine even if it is excluded by the workspace's.gitignore
file or other rules.
To bootstrap the development run:
git clone [email protected]:remote-cli/remote.git
cd remote
python3 -m venv env
source env/bin/activate
pip install -e .
pip install -r test_requirements.txt
After it, you can open the code in any editor or IDE you like. If you prefer VSCode, the project contains the configuration file for it.
Before submitting your pull request, please check it by running:
flake8 src test && mypy -p remote && black --check -l 120 src test && isort --check-only src test && pytest
If black
or isort
fails, you can fix it using the following command:
black -l 120 src test && isort src test
Don't forget to add changed files to your commit after you do it.