0 purchases
remoteexecapi 1.13.3
Remote
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.
System Requirements
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.
Getting Started
Installing on Mac OS X
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
Installing on other systems
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.
Configuring the remote host
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.
First run
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
Distribution
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 host
remote-ignore: set up directories/files to ignore while pushing
remote-push: explicitly push local changes remote
remote-pull: pull a directory from remote to local
remote: execute a command remotely, after first syncing the local tree with the remote tree
remote-explain: explain your remote setup, explain what command actually will get run
remote-quick: execute a command remotely without syncing the trees
remote-add: add another remote host to the mirror list
mremote: 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.
Configuration
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.
The remote CLI cannot execute any commands remotely until this file is present, or the global config
overwrites this with allow_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
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 the remote CLI.
Reference:
allow_uninitiated_workspaces (optional, defaults to false) - If this flag is set to true 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 the directory for a host, this setting will be ignored.
use_relative_remote_paths (optional, defaults to false)
If set to false, all the workspaces will be stored in the remote_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 the remote_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
[[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 by default = 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 to 22) - a port used by the ssh daemon on the host.
supports_gssapi_auth (optional, defaults to true) - true if the remote host supports gssapi-* auth
methods. We recommend disabling it if the ssh connection to the host hangs for some time during establishing.
default (optional, defaults to false) - true if this host should be used by default
label (optional) - a text label that later can be used to identify the host when running the remote CLI.
cmd_prefix (optional) - a string which is prefixed to all commands executed via the remote 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 in include.
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 the exclude.
include_vcs_ignore_patterns (optional, defaults to false) - if true and .gitignore is present,
all its patterns will be included in the exclude list.
Workspace Configuration File
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.
Workspace Files Sync Configuration File
.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.
.remoteenv file
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 run source .remoteenv on the destination host before running the actual command. For example,
here is how you can run remote'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.
Development & Contribution
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.
Changelog
1.13.3
Add cmd-prefix support in [[hosts]]
Avoid gitignoring remoteignor.toml
1.13.2
Fix the shiv creation script
1.13.1
Move the pypi repo to remote-exec-api
1.13.0
Allow providing extra arguments for the SSH command
1.12.0
Allow providing paths for remote-push
Support spaces in paths and improve shell escaping
1.11.0
Change order of application of API-provided env and user-provided env
1.10.1
Enable tunelling in remote-quick
1.10.0
Allow providing multiple ports for tunneling
1.9.1
fixed 1.9.0 release
1.9.0
improve sync include/exclude API
1.8.0
Introduce .remoteignore.toml file
Fix typo in include_vcs_ignore_patterns option name
Start using shlex to escape the shell command
1.7.0
improve the internal port forwarding API
1.6.1
add an API to set environment variables for the remote execution
1.6.0
support source changes streaming with watchdog
add remote-explain CLI entry point
add --multi and --log flags to remote entry point
add --push and --pull flags to remote-ignore entyr point
1.5.0
use TOML configs by default
new CLI option to select a host by label/index
new config option to provide custom port for SSH
.remoteenv usage fixes
1.4.5
Select host by their label
Configure ssh connection options
1.4.4
Add the host labeling in toml config files
1.4.3
Fixed rsync bugs
1.4.2
Add the possibility to start an ssh tunnel during the remote execution
1.4.1
Fail ssh and rsync when password is requested by adding -o BatchMode=yes
add -K to rsync ssh command to make it homogenous with ssh
Support rsync's --include-from to make exclude overrides possible
1.4.0
TOML config support
.gitignore files can be used as a source for sync ignore
1.3.8
Unit tests
Releases with good versions
1.3.7
Moving to pypi releases
1.3.6
Fix reverse-sync breakage for remote-dirs without trailing /
Reverse-sync always, even when remote command fails
1.3.5
Support for relative and absolute paths (including leading ~ and /) in remote directory
Support for global variables RSHELLandRSHELL_OPTS to configure remote shell invocation
1.3.4
Support for choice of login shell and shell options for remote host in .remote file
Better initialization for mremote screen
1.3.3
Fixing a bug in command execution for zsh based shells
1.3.2
Fixing signal handling, cleanup of temp files
Fixing a bug in inferring verbose mode
1.3.1
More comprehensive ignore rule specifications (push, pull, both) for .remoteignore files
Command-line argument improvements to remote, remote-push, remote-pull and remote-explain
1.3.0
Adding command line flags for remote, remote-push
Dropping remote-push-dry since remote-push -n is an equivalent alternative.
Improving remote-explain output to include diff status
1.2.7
remote-explain supports -c flag to test connection to the remote host
1.2.6
Support for remote-explain, a tool to explain your remote setup
1.2.5
Support for adding .remote related files to .gitignore
1.2.4
More robust parameter parsing
1.2.3
Support for initializing environment per remoted directory through .remoteenv file
1.2.2
Fail fast when local directory is not setup for remote
Support kerberos auth (through ssh -K)
1.2.1
New executables: remote-set, remote-host
mremote now uses screen
1.2.0
Support for multiple-hosts
New executables: remote-add and mremote
1.1.3
remote-pull: now pulls without syncing the trees
remote-quick: when you want to execute a command without syncing the trees
1.1.2
More fixes to md5 generation (verified to work on mac and linux)
Silent ssh connections for reduced verbosity
1.1.1
Robust in finding appropriate md5 utility
1.1
Added remote-pull
Moved remote-sync to remote-push
1.0
First drop
For personal and professional use. You cannot resell or redistribute these repositories in their original state.
There are no reviews.