git-ibundle

git-ibundle is a tool for incremental offline mirroring of a Git repository.

Incremental repository data is transferred from a source network to a destination network via a sequence of "ibundle" (incremental bundle) files. No interactive connection is needed between the two networks; only a reliable one-way file transfer capability is required.

Typical transfer process

Consider mirroring repo.git from a source network to a disconnected destination network.

First, perform one-time setup:

Next, repeat the following steps as often as desired to keep source and destination repositories synchronized:

Change history

See CHANGES for a list of changes.

License

git-ibundle is licensed under the terms of the MIT license; see LICENSE.

Requirements

Development and most testing is done on Linux; this is the best-supported platform. Limited testing is done on Windows. No testing is done on Macos.

Installation

Options for installation include:

Invocation as git ibundle

git-ibundle is named with a git- prefix so that it can integrate into Git as the command ibundle. If the executable git-ibundle is found on the PATH, then the Git command git ibundle will delegate to git-ibundle. These invocations are then equivalent:

git-ibundle <ibundle-arguments>
git ibundle <ibundle-arguments>

This allows git-ibundle to inherit some generic Git's functionality, the most useful of which is:

git -C path/to/repository <command>

This causes Git to to change the directory to path/to/repository before running <command>. For example:

# Create directory and initialize as a bare Git repo:
mkdir repo.git
git -C repo.git init --bare

This is useful for git-ibundle as well. Consider having a repository and an ibundle file in the same directory:

./
    repo.git/
    repo.ibundle

To fetch from this ibundle into the repository, you could change into the repository directory and fetch like this:

cd repo.git
git ibundle fetch ../repo.ibundle
cd ..

Or you could use -C repo.git to do this in one step:

git -C repo.git ibundle fetch ../repo.ibundle

Note that changing the directory occurs before git-ibundle examines its arguments, so if you use a relative path to repo.ibundle, you must make that path relative to the repository's location (which is why the example above uses ../repo.ibundle).

Model

git-ibundle synchronizes two repositories at discrete synchronization points in time. Each time an ibundle is created via git-ibundle create, a new synchronization point is defined, and the current repository state is recorded. Repository state includes HEAD and all branches, tags, and associated commit IDs. An automatically incrementing sequence number provides a way to identify the synchronization point and to label the associated ibundle file and current repository state.

An ibundle file contains the source repository changes occurring between a previous (basis) state and the current state. At the destination, git-ibundle fetch will apply these changes to the destination, synchronizing that repository with the source. git-ibundle verifies that the destination repository has already applied the changes for the ibundle's basis.

By default, an ibundle is created using the immediately preceding sequence number as a basis; it's possible to choose a different basis via git-ibundle create --basis <seq_num>. This is useful if any previous ibundle files have been lost before fetching them into the destination repository.

For a repository repo.git, git-ibundle uses the directory repo.git/ibundle/ to hold its metadata. This directory is transparent to Git and does not interfere or overlap with normal Git operations.

Mirroring a subset

git-ibundle itself always makes a complete mirror of the source repository. This includes all references in the repository, including anything found below refs/remotes/<REMOTE>. The source repository should be cloned to a local repo.git directory using git clone --mirror to prevent creation of refs/remotes/<REMOTE> and ensure accurate mirroring.

It's possible to mirror a subset of the origin repository by setting up a negative refspec. For example, to avoid mirroring Github pull requests (which have refspecs of the form refs/pull/*), the following negative refspec can be used:

remote.origin.fetch=^refs/pull/*

This can't be configured via git clone --mirror --config because the negative refspec doesn't take effect soon enough; instead, manually setup the source repo.git via:

mkdir repo.git
cd repo.git
git init --bare
git remote add origin --mirror=push https://github.com/user/repo.git
git config remote.origin.fetch '+refs/*:refs/*'
git config --add remote.origin.fetch '^refs/pull/*'

You may then fetch and verify that the refs are as expected:

git fetch
git show-ref

Command invocation details

Create an ibundle

```text Usage: git-ibundle create [OPTIONS]

Arguments: ibundle file to create

Options: --standalone force ibundle to be standalone --basis choose alternate basis sequence number -q, --quiet run quietly --allow-empty allow creation of an empty ibundle -h, --help Print help information -V, --version Print version information ```

On the first ibundle creation, the repository is assigned a random repoid. This is used to help prevent accidental application of an ibundle file to the wrong destination Git repository. The repoid will be checked during git-ibundle fetch operations.

The basis sequence number defaults to one less than the ibundle's sequence number; for the first ibundle (which will have sequence number 1), the basis sequence number will be 0.

With --basis 0, the created ibundle will assume no prerequisite commits are present at the destination; it will contain everything needed to create a mirror repository via git-ibundle fetch. Note that --basis 0 implies --standalone.

Without --standalone, the ibundle will be created with the assumption that the destination has been synchronized to the --basis sequence number and thus contains all prerequisite commits and references; as a result, the created ibundle file contains only the changed references for compactness, along with a Git "PACK" containing updated Git objects.

With --standalone, the ibundle will instead contain the full set of named references and a full enumeration of prerequisite commit IDs. Commit data will still be incremental and based on the commits implied by --basis. This may be used for cases where the destination repository is known to have the prerequisite commits but lacks the actual basis sequence number (e.g., when using a pre-existing repository mirror on the destination network).

Normally, git-ibundle create will refuse to create an ibundle when there have been no changes since the last ibundle was created. In this case, an exit status of 3 is provided (whereas most failures result in an exit status of 1). To allow creation of an empty ibundle, use --allow-empty.

Fetch from an ibundle

```text Usage: git-ibundle fetch [OPTIONS]

Arguments: ibundle file to fetch

Options: --dry-run perform a trial fetch without making changes to the repository -q, --quiet run quietly --force force fetch operation -h, --help Print help information -V, --version Print version information ```

With --dry-run, a fetch operation is simulated but no changes will be made to the repository. This is useful for checking the validity of an ibundle file and for testing.

git-ibundle is cautious about fetching from an unexpected bundle. Use --force to override this caution. --force may be used in these cases:

Convert an ibundle into a bundle

```text Usage: git-ibundle to-bundle [OPTIONS]

Arguments: ibundle file to convert bundle file to create from ibundle

Options: -q, --quiet run quietly --force force fetch operation -h, --help Print help information -V, --version Print version information ```

This converts an ibundle into a standard Git bundle. The Git bundle will contain all references and prerequisite commit IDs needed to apply the bundle via:

git fetch <BUNDLE_FILE> --force --prune "*:*"

Note that Git bundles cannot represent information about symbolic changes to HEAD. git-ibundle to-bundle indicates which Git operations are necessary to apply the bundle to a repository. For example:

console $ git-ibundle to-bundle ../repo1.ibundle ../repo1.bundle) read '../repo1.ibundle', seq_num=13, 0 refs wrote '../repo1.bundle', 11 refs, 5 prereqs To apply this bundle file in destination repository: git fetch .../file.bundle --force --prune "*:*" git symbolic-ref HEAD refs/heads/main

An ibundle created with git-ibundle create --standalone can be converted without reference to a Git repository; otherwise, a repository with the proper repo_id and basis must be used.

The --force switch is required in the same cases as for git-ibundle fetch.

Report status

```text Usage: git-ibundle status [OPTIONS]

Options: --long Provide longer status -h, --help Print help information -V, --version Print version information ```

This provides git-ibundle status for a given repository. For example:

console $ git-ibundle status repo_id: 18450f13-4003-474a-a69e-22782ef3848f max_seq_num: 13 next_seq_num: 14

The next_seq_num field indicates the sequence number that will be used for the next git-ibundle create operation.

The max_seq_num field indicates the sequence number used by the most recent git-ibundle create operation.

With --long, more detail is provided:

console $ git-ibundle status --long repo_id: 18450f13-4003-474a-a69e-22782ef3848f max_seq_num: 13 next_seq_num: 14 long_details: seq_num num_refs HEAD 1 0 refs/heads/main 2 0 refs/heads/main 3 5 refs/heads/main 4 5 refs/heads/main 5 6 refs/heads/main 6 7 refs/heads/fix1 7 7 refs/heads/main2 8 7 refs/heads/main 9 7 343f8d34eb565c0e97194604fa2c6c3ff8ba4931 (detached) 10 7 refs/heads/main 11 7 refs/heads/main 12 7 refs/heads/main 13 11 refs/heads/main

Cleanup old sequence numbers

```text Usage: git-ibundle clean [OPTIONS]

Options: --keep Number of sequence numbers to retain [default: 20] -h, --help Print help information -V, --version Print version information ```

By default, git-ibundle retains the metadata for all sequence numbers. Use git-ibundle clean to cleanup older sequence numbers.