2021-05-15 16:16:31 +00:00
|
|
|
# Jujutsu
|
2020-12-12 07:37:25 +00:00
|
|
|
|
|
|
|
|
2020-12-12 08:12:04 +00:00
|
|
|
## Disclaimer
|
2020-12-12 07:37:25 +00:00
|
|
|
|
2020-12-12 08:12:04 +00:00
|
|
|
This is not a Google product. It is an experimental version-control system
|
2021-09-08 17:00:47 +00:00
|
|
|
(VCS). It was written by me, Martin von Zweigbergk (martinvonz@google.com). It
|
2022-02-08 17:36:20 +00:00
|
|
|
is my personal hobby project and my 20% project at Google. It does not indicate
|
|
|
|
any commitment or direction from Google.
|
2020-12-12 07:37:25 +00:00
|
|
|
|
|
|
|
|
2020-12-12 08:12:04 +00:00
|
|
|
## Introduction
|
2020-12-12 07:37:25 +00:00
|
|
|
|
2022-01-04 05:38:34 +00:00
|
|
|
Jujutsu is a [Git-compatible](docs/git-compatibility.md)
|
2021-10-28 04:45:37 +00:00
|
|
|
[DVCS](https://en.wikipedia.org/wiki/Distributed_version_control). It combines
|
2022-01-04 05:31:46 +00:00
|
|
|
features from Git (data model,
|
|
|
|
[speed](https://github.com/martinvonz/jj/discussions/49)), Mercurial (anonymous
|
|
|
|
branching, simple CLI [free from "the index"](docs/git-comparison.md#the-index),
|
2022-01-29 06:47:18 +00:00
|
|
|
[revsets](docs/revsets.md), powerful history-rewriting), and Pijul/Darcs
|
2021-12-12 08:19:39 +00:00
|
|
|
([first-class conflicts](docs/conflicts.md)), with features not found in either
|
|
|
|
of them ([working-copy-as-a-commit](docs/working-copy.md),
|
2022-01-10 15:59:16 +00:00
|
|
|
[undo functionality](docs/operation-log.md), automatic rebase,
|
|
|
|
[safe replication via `rsync`, Dropbox, or distributed file
|
|
|
|
system](docs/technical/concurrency.md)).
|
2020-12-12 07:37:25 +00:00
|
|
|
|
2020-12-12 08:12:04 +00:00
|
|
|
The command-line tool is called `jj` for now because it's easy to type and easy
|
2021-05-15 16:16:31 +00:00
|
|
|
to replace (rare in English). The project is called "Jujutsu" because it matches
|
2021-10-28 04:45:37 +00:00
|
|
|
"jj".
|
2020-12-12 07:37:25 +00:00
|
|
|
|
2022-01-29 06:36:56 +00:00
|
|
|
## Features
|
2021-05-23 21:42:20 +00:00
|
|
|
|
2022-01-29 06:36:56 +00:00
|
|
|
### Compatible with Git
|
2022-02-27 22:00:46 +00:00
|
|
|
|
2022-01-29 06:36:56 +00:00
|
|
|
Jujutsu has two backends. One of them is a Git backend (the other is a native
|
2022-03-05 16:42:35 +00:00
|
|
|
one [^native-backend]). This lets you use Jujutsu as an alternative interface to Git. The commits
|
2022-01-29 06:36:56 +00:00
|
|
|
you create will look like regular Git commits. You can always switch back to
|
2022-02-20 06:13:49 +00:00
|
|
|
Git. The Git support uses the [libgit2](https://libgit2.org/) C library.
|
2022-01-29 06:36:56 +00:00
|
|
|
|
2022-03-05 16:42:35 +00:00
|
|
|
[^native-backend]: There's currently practically no reason to use the native
|
|
|
|
backend. It exists to make sure that it's possible to eventually add
|
|
|
|
functionality that cannot easily be added the Git backend.
|
|
|
|
|
2022-01-29 06:36:56 +00:00
|
|
|
<a href="https://asciinema.org/a/DRCzktCyEAxH6j788ZDT6aSjS" target="_blank">
|
|
|
|
<img src="https://asciinema.org/a/DRCzktCyEAxH6j788ZDT6aSjS.svg" />
|
|
|
|
</a>
|
|
|
|
|
|
|
|
### The working copy is automatically committed
|
|
|
|
|
|
|
|
Most Jujutsu commands automatically commit the working copy. This leads to a
|
|
|
|
simpler and more powerful interface, since all commands work the same way on the
|
|
|
|
working copy or any other commit. It also means that you can always check out a
|
|
|
|
different commit without first explicitly committing the working copy changes
|
|
|
|
(you can even check out a different commit while resolving merge conflicts).
|
|
|
|
|
|
|
|
<a href="https://asciinema.org/a/zWMv4ffmoXykBtrxvDY6ohEaZ" target="_blank">
|
|
|
|
<img src="https://asciinema.org/a/zWMv4ffmoXykBtrxvDY6ohEaZ.svg" />
|
|
|
|
</a>
|
|
|
|
|
|
|
|
### Operations update the repo first, then possibly the working copy
|
|
|
|
|
|
|
|
The working copy is only updated at the end of an operation, after all other
|
2022-02-27 22:00:46 +00:00
|
|
|
changes have already been recorded. This means that you can run any command
|
2022-01-29 06:36:56 +00:00
|
|
|
(such as `jj rebase`) even if the working copy is dirty.
|
|
|
|
|
|
|
|
### Entire repo is under version control
|
|
|
|
|
|
|
|
All operations you perform in the repo are recorded, along with a snapshot of
|
|
|
|
the repo state after the operation. This means that you can easily revert to an
|
|
|
|
earlier repo state, or to simply undo a particular operation (which does not
|
|
|
|
necessarily have to be the most recent operation).
|
|
|
|
|
|
|
|
<a href="https://asciinema.org/a/OFOTcm2XlZ09LLEI5bHYM8Alw" target="_blank">
|
|
|
|
<img src="https://asciinema.org/a/OFOTcm2XlZ09LLEI5bHYM8Alw.svg" />
|
|
|
|
</a>
|
|
|
|
|
|
|
|
### Conflicts can be recorded in commits
|
|
|
|
|
|
|
|
If an operation results in conflicts, information about those conflicts will be
|
|
|
|
recorded in the commit(s). The operation will succeed. You can then resolve the
|
|
|
|
conflicts later. One consequence of this design is that there's no need to
|
|
|
|
continue interrupted operations. Instead, you get a single workflow for
|
|
|
|
resolving conflicts, regardless of which command caused them. This design also
|
|
|
|
lets Jujutsu rebase merge commits correctly (unlike both Git and Mercurial).
|
|
|
|
|
|
|
|
Basic conflict resolution:
|
|
|
|
<a href="https://asciinema.org/a/MWQz2nAprRXevQEYtaHScN2tJ" target="_blank">
|
|
|
|
<img src="https://asciinema.org/a/MWQz2nAprRXevQEYtaHScN2tJ.svg" />
|
|
|
|
</a>
|
|
|
|
|
|
|
|
Juggling conflicts:
|
|
|
|
<a href="https://asciinema.org/a/HqYA9SL2tzarPAErpYs684GGR" target="_blank">
|
|
|
|
<img src="https://asciinema.org/a/HqYA9SL2tzarPAErpYs684GGR.svg" />
|
|
|
|
</a>
|
|
|
|
|
|
|
|
### Automatic rebase
|
|
|
|
|
|
|
|
Whenever you modify a commit, any descendants of the old commit will be rebased
|
|
|
|
onto the new commit. Thanks to the conflict design described above, that can be
|
|
|
|
done even if there are conflicts. Branches pointing to rebased commits will be
|
|
|
|
updated. So will the working copy if it points to a rebased commit.
|
2021-05-23 21:42:20 +00:00
|
|
|
|
2022-01-29 06:47:18 +00:00
|
|
|
### Comprehensive support for rewriting history
|
2021-05-23 21:42:20 +00:00
|
|
|
|
2022-01-29 06:47:18 +00:00
|
|
|
Besides the usual rebase command, there's `jj describe` for editing the
|
|
|
|
description (commit message) of an arbitrary commit. There's also `jj edit`,
|
|
|
|
which lets you edit the changes in a commit without checking it out. To split
|
|
|
|
a commit into two, use `jj split`. You can even move part of the changes in a
|
2022-02-27 22:00:46 +00:00
|
|
|
commit to any other commit using `jj move`.
|
2022-01-29 06:47:18 +00:00
|
|
|
|
|
|
|
|
|
|
|
## Status
|
2021-09-08 17:00:47 +00:00
|
|
|
|
2021-10-28 04:45:37 +00:00
|
|
|
The tool is quite feature-complete, but some important features like (the
|
|
|
|
equivalent of) `git blame` and `git log <paths>` are not yet supported. There
|
|
|
|
are also several performance bugs. It's also likely that workflows and setups
|
|
|
|
different from what I personally use are not well supported. For example,
|
|
|
|
pull-request workflows currently require too many manual steps.
|
|
|
|
|
|
|
|
I have almost exclusively used `jj` to develop the project itself since early
|
|
|
|
January 2021. I haven't had to re-clone from source (I don't think I've even had
|
|
|
|
to restore from backup).
|
|
|
|
|
|
|
|
There *will* be changes to workflows and backward-incompatible changes to the
|
|
|
|
on-disk formats before version 1.0.0. Even the binary's name may change (i.e.
|
|
|
|
away from `jj`). For any format changes, I'll try to implement transparent
|
|
|
|
upgrades (as I've done with recent changes), or provide upgrade commands or
|
|
|
|
scripts if requested.
|
2021-09-08 17:00:47 +00:00
|
|
|
|
|
|
|
|
2021-10-13 15:42:29 +00:00
|
|
|
## Installation
|
|
|
|
|
2022-02-28 00:10:23 +00:00
|
|
|
### Linux
|
|
|
|
|
|
|
|
On most distributions, you'll need to install from source.
|
|
|
|
|
|
|
|
#### From source
|
|
|
|
|
2022-03-02 06:11:36 +00:00
|
|
|
First make sure that you have the `libssl-dev` and `openssl` packages installed
|
|
|
|
by running something like this:
|
|
|
|
```shell script
|
|
|
|
sudo apt-get install libssl-dev openssl
|
|
|
|
```
|
|
|
|
|
|
|
|
Now run:
|
2022-02-28 00:10:23 +00:00
|
|
|
```shell script
|
|
|
|
cargo install --git https://github.com/martinvonz/jj.git
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
|
|
#### Nix OS
|
|
|
|
|
|
|
|
If you're on Nix OS you can use the flake for this repository.
|
|
|
|
For example, if you want to run `jj` loaded from the flake, use:
|
|
|
|
|
|
|
|
```shell script
|
|
|
|
nix run 'github:martinvonz/jj'
|
|
|
|
```
|
|
|
|
|
|
|
|
You can also add this flake url to your system input flakes. Or you can
|
|
|
|
install the flake to your user profile:
|
|
|
|
|
|
|
|
```shell script
|
|
|
|
nix profile install 'github:martinvonz/jj'
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
|
|
### Mac
|
|
|
|
|
|
|
|
You may need to run some or all of these:
|
2022-02-27 22:00:46 +00:00
|
|
|
```shell script
|
|
|
|
xcode-select --install
|
|
|
|
brew install openssl
|
|
|
|
brew install pkg-config
|
|
|
|
export PKG_CONFIG_PATH="/opt/homebrew/opt/openssl@3/lib/pkgconfig"
|
2022-02-20 06:56:31 +00:00
|
|
|
```
|
|
|
|
|
2022-02-28 00:10:23 +00:00
|
|
|
Now run:
|
2021-10-13 15:42:29 +00:00
|
|
|
```shell script
|
2022-02-27 22:00:46 +00:00
|
|
|
cargo install --git https://github.com/martinvonz/jj.git
|
2021-10-13 15:42:29 +00:00
|
|
|
```
|
|
|
|
|
2022-02-28 00:10:23 +00:00
|
|
|
|
|
|
|
### Windows
|
|
|
|
|
|
|
|
Run:
|
|
|
|
```shell script
|
|
|
|
cargo install --git https://github.com/martinvonz/jj.git
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
|
|
## Initial configuration
|
|
|
|
|
|
|
|
You may want to configure your name and email so commits are made in your name.
|
|
|
|
Create a file at `<config dir>/jj/config.toml` (where `<config dir>` is
|
2022-02-27 02:10:08 +00:00
|
|
|
`${XDG_CONFIG_HOME}` or `~/.config/` on Linux, `~/Library/Application Support/`
|
|
|
|
on macOS, and `~\AppData\Roaming\` on Windows) or `~/.jjconfig` and make it
|
|
|
|
look something like this:
|
2021-10-14 03:33:17 +00:00
|
|
|
```shell script
|
|
|
|
$ cat ~/.jjconfig
|
|
|
|
[user]
|
|
|
|
name = "Martin von Zweigbergk"
|
|
|
|
email = "martinvonz@google.com"
|
|
|
|
```
|
|
|
|
|
2022-02-28 00:10:23 +00:00
|
|
|
|
|
|
|
## Command-line completion
|
2022-02-27 21:53:41 +00:00
|
|
|
|
|
|
|
To set up command-line completion, source the output of
|
|
|
|
`jj debug completion --bash/--zsh/--fish`. Exactly how to source it depends on
|
|
|
|
your shell.
|
|
|
|
|
2022-02-28 00:10:23 +00:00
|
|
|
### Bash
|
2022-02-27 21:53:41 +00:00
|
|
|
```shell script
|
|
|
|
source <(jj debug completion) # --bash is the default
|
|
|
|
```
|
|
|
|
|
2022-02-28 00:10:23 +00:00
|
|
|
### Zsh
|
2022-02-27 21:53:41 +00:00
|
|
|
```shell script
|
|
|
|
autoload -U compinit
|
|
|
|
compinit
|
|
|
|
source <(jj debug completion --zsh | sed '$d') # remove the last line
|
|
|
|
compdef _jj jj
|
|
|
|
```
|
|
|
|
|
2022-02-28 00:10:23 +00:00
|
|
|
### Fish
|
2022-02-27 21:53:41 +00:00
|
|
|
```shell script
|
|
|
|
jj debug completion --fish | source
|
|
|
|
```
|
|
|
|
|
|
|
|
|
2021-09-09 17:56:22 +00:00
|
|
|
## Getting started
|
2021-05-23 21:42:20 +00:00
|
|
|
|
2021-09-09 17:56:22 +00:00
|
|
|
The best way to get started is probably to go through
|
2021-11-19 04:05:43 +00:00
|
|
|
[the tutorial](docs/tutorial.md). Also see the
|
|
|
|
[Git comparison](docs/git-comparison.md), which includes a table of
|
|
|
|
`jj` vs. `git` commands.
|
2021-10-20 22:27:33 +00:00
|
|
|
|
|
|
|
|
|
|
|
## Related work
|
|
|
|
|
|
|
|
There are several tools trying to solve similar problems as Jujutsu. See
|
2021-12-18 15:54:46 +00:00
|
|
|
[related work](docs/related-work.md) for details.
|