../upgit
git pull, parallel
Clone with
git clone git://git.awl.red/neallred/upgit

upgit📎

Command line tool for updating git repos quickly. Download the latest version for Linux or Mac.

Motivation📎

When you have lots of repos cloned, they tend to get out of date. Stale repos mean stale code searches and increased merge conflicts when developing on a shared branch (or branching off a stale one).

A quick to implement solution is a bash script that lists your git projects folder and loop through them, running git pull. That works, but has some issues:

  1. Given the embarassingly parallel nature of updating many repos, going one at a time is slow. You could use GNU parallel, but that hampers portability.
  2. The script may have hard-coded assumptions about directory structure or location, limiting reuse and sharing.
  3. The script may require a lot of password entry.
  4. Does the script guard against files and folders that are not git repos?
  5. What should be done if the repo is dirty? You might cause merge conflicts you would rather have avoided.
  6. Scrolling back through hundreds of lines of output to see what happened is time consuming and error prone.

upgit solves these by:

  1. updating repos in parallel if possible.
  2. allowing flexible patterns of git folders, and shortening command line typing with env vars.
  3. allowing flexible patterns of password reuse to fit your workflow.
  4. checking whether a folder is a git repo.
  5. leaving dirty repos alone, and aborting merge conflicts.
  6. presenting a consise summary of what happened to each repo, grouped by outcome.

Usage📎

Examples📎

Update all the repos contained in the github folder, and all the repos contained in the bitbucket folder:

upgit ~/github ~/bitbucket

Supposing you work for a company in which you are part of multiple teams, and you organize your repos according to the teams you are on, you can update them all like this:

upgit ~/megacorp/team-a ~/megacorp/team-b

Update all repos in the github folder, being prompted immediately for the password to an assumed ssh key in $HOME/.ssh/id_rsa:

upgit --default-ssh ~/github

Or, use a different key than $HOME/.ssh/id_rsa:

upgit --default-ssh $HOME/.ssh/my_other_key ~/github

A way of only entering a password once per domain when all your orgs/repos passwords for a given user are the same:

upgit --share domain --plain https://neallred@bitbucket.org/ --plain https://neallred@github.com/ ~/github

Help📎

Run upgit --help for the following usage help.

upgit 0.1.0
Nathaniel Allred <neallred@gmail.com>
Pulls all repos within a folder containing git projects, in parallel. Supports
configuration via command line flags and params, or via ENV vars. Command line
takes precedence. If no option is set but is needed (i.e. repos requiring auth),
user will be prompted if a TTY is available, or skip that repo if it is not
available.

USAGE:
    upgit [FLAGS] [OPTIONS] [--] [git-dirs]...

FLAGS:
        --default-plain
            Default password to attempt for https cloned repos. User will be
            prompted for the password. Env var is UPGIT_DEFAULT_PLAIN set to any
            value.
        --default-ssh
            Default password to use for ssh keys. User will be prompted for the
            password. Env var is UPGIT_DEFAULT_SSH set to any value.
    -h, --help
            Prints help information

    -V, --version
            Prints version information


OPTIONS:
        --plain <plain>...
            Git repo https url with username. For example, `--plain
            https://neallred@bitbucket.org/neallred/allredlib-data-backup.git`.
            For each time this option is passed, user will be prompted for a
            password. Env var is comma separated UPGIT_PLAIN.
        --share <share>
            Degree to which credentials may reused between repos needing auth.
            Each level is additive. `none` means no credential reuse between
            repos, and defaults are ignored. `default` means default provided
            credentials may be reused. `duplicate` means deefaults, plus
            multiple copies of a repo can reuse each other's credential. `org`
            means duplicate, plus upgit will infer a matching org by looking at
            the second to last url path segment (e.g. `neallred` in
            https://github.com/neallred/upgit`). `domain` means reusing when
            user and url domain match. Env var is UPGIT_SHARE. [default:
            default]  [possible values: none, default, duplicate, org, domain]
        --ssh <ssh>...
            Paths to ssh keys to preverify. User will be prompted for password
            for each key given. Can enter "blank" if ssh key is not password
            protected. Env var is comma separated UPGIT_SSH.

ARGS:
    <git-dirs>...
            Paths (relative or absolute) to folders that contain git repos. Env
            var is comma separated UPGIT_GIT_DIRS.

Authentication📎

upgit supports ssh and user/pass auth methods. It is work in progress and subject to change. The user is prompted when needed, or when the user signals preprovision of credentials via flags or env vars.

Ssh supports multiple keys. If none is provided, and a repo requests ssh authentication, upgit assumes the private key is at $HOME/.ssh/id_rsa. It allows entering a blank ssh passphrase if there is no passphrase on the key.

The plain text method assumes the last password entered is the one that should be used for unseen URLs. Because of how threading is currently implemented, entering a wrong password means a LOT of password re-entry. To mitigate this, plain text password entry prompts for password confirmation.

Local developement / building📎

Prerequisites📎

A working cargo install. It (and rustup) can be installed by following the instructions on rustup.rs.

Install📎

To build and install locally, from the repo root, run

cargo build --release

Move the target/release/upgit executable to a folder in your $PATH.

Running📎

To build and run, from the repo root, run

cargo run <relative path(s) to folder holding git project(s)>

For example, if I stored all my repos in ~/github, cargo run ~/github

Cross compiling📎

Currently, cross compiling from (Debian) Linux to MacOS 10.7 and higher is supported. This is done via osxcross, which is included as a git submodule. osxcross has a number of system dependencies. On Debian, you can run ./ensure-cross-compile-setup-linux.bash && ./compile-mac-on-linux.bash. The build output is created at target/x86_64-apple-darwin/release/upgit.

Tests📎

Run unit tests with cargo test.

Run integration tests with ./integration.bash. Automated assertions are still TODO, but you can manually check if STDOUT matches expectations.

Run end-to-end (e2e) tests with ./e2e.bash. Depends on docker and docker-compose. The tests build a common, reused git server, and various git clients. Each git client is an end to end test. Current tests are:

  1. Pulling and reporting updates to one repo. The test script:
  • makes a repo
  • pushes it to the git server
  • clones it to another folder
  • updates the original repo and the git server origin repo
  • runs upgit, checking that the clone is updated
.cargo
config 4 years ago
.gitignore 4 years ago
.gitmodules 4 years ago
Cargo.lock 1 year ago
Cargo.toml 1 year ago
LICENSE 4 years ago
README.md 4 years ago
e2e.bash 4 years ago
integration.bash 4 years ago
src
config.rs 1 year ago
creds.rs 1 year ago
end.rs 4 years ago
main.rs 4 years ago
string_ops.rs 4 years ago
test
e2e
docker-compose.yml 4 years ago
git-server.Dockerfile 4 years ago
git_server_keys 4 years ago
test-pull.Dockerfile 4 years ago
test-pull.sh 4 years ago
upgit_rsa 4 years ago
upgit_rsa.pub 4 years ago