Majjit LSP

An out-there suggestion for the nascent jj ecosystem!

I was skimming Martins post, and the VS Code section of it kicked off this particular train of thought. Summarizing, VSCode has some built-in integrations with git, and they arent necessarily directly applicable to jj. The direct solution is, of course, to add first class support for jj to VSCode.

But I want better. Specifically, my problem is that I dont actually like VS Code VCS UI. I use @kaholes awesome implementation of Magit for VSCode, edamagit. Magit is objectively the correct way to implement a VCS user interface (more on this in just a moment).

So I need to wait for someone to code the Magit/jj/VSCode triple. Whats worse, I actually want to switch to Zed at some point, and the absence of Magit is one of the bigger things that prevents this. But Zed doesnt at the moment have a rich enough plugin API to express that.

At the same time, although two separate code-bases, Magit in VSCode and Magit in Emacs are basically the same thing from the perspective of the user. We are clearly missing a narrow waist here which would allow us to implement Magit only once.

Or rather, theres a barely adequate enough narrow waist shell/terminal combo. You could implement magit-tui which would be re-usable between, say, Vim and Helix. But this particular thin waist sucks. It suffers from Byzantine complexity (why the hell the kernel needs to be directly involved), unfixable design bugs (terminfo database, necessity to set both your terminal emulators and your shell color themes separately), and missing features (window management, for example). I think an absolutely massive improvement is possible here. I have a separate set of notes about that, but thats a huge reinvent the wheel by starting from atoms kind of thing.

Luckily, for the present problem of getting a first-class UI for jj into any editor, I think a shortcut exists! We can implement magit for jj once, and have it working for almost free in any editor, GUI or terminal. To not spoil the answer right away, let me describe, first,

What is Magit

The core idea is text file is the user interface. Thats it! This is the paradigm of Emacs (and Acme), with Magit being the most successful application. Text being the interface means two things:

  • When we need to present information to the user, we present it in textual format.
  • When we need input from the user, it is also text driven. In the simplest form, the user canclick on a particular word in the text.

Lets apply this to various UI tasks of a VCS. The basic one is of course informing the user about changes. Here, we can generate a textual diff a file with each line marked as + or -. On top of this, theres a number of progressive enhancements:

  • Of course + lines are colored green and - red.
  • Consecutive changes are grouped into hunks, and each hunk is individually foldable.
  • A diff can span multiple files, and files form another level in the folding hierarchy. By folding the entire diff recursively, you get a list of changed files, from which you can drill down into an individual file.

This is the presentation side of the interface. And now, interaction:

Clicking on any line in the diff opens the corresponding file in the working copy. This composes well with split view (a feature missing from the terminals narrow waist): on the right you can have your hierarchical diff, on the left, your code. Moving in the diff file automatically moves the working copy view.

This is a very productive interface to make sense of larger changes. You open the change as this hierarchical diff. First, everything is folded, so you only see the list of changed files, which gives you the context about a change. You then unfold the most interesting changed file and start perusing the diff. If, at some point, you realize that you need more context to understand a particular hunk, you click on it, and view, at the other half of the screen, the entire function surrounding the changed line.

The rest of the VCS builds naturally on top. Above hunks and files you add commits as another level of hierarchy. So, everything folded is now a list of commits. The list of commits view suggest a natural way to model history manipulation. If you want to reorder two commits, you can reorder two lines in the list of commits text document.

If you want to split an individual commit, you can open a text document which contains two diffs. One is the commit, the other is initially empty. Clicking on a hunk moves is from one diff to another.

And of course, you can have a dashboard text document, which shows the diff for the latest commit (jj) or staging area (git), a list of recent commits, a list of branches, and has hyper-linksfor all VCS operations.

Magit For All Editors

Finally, the idea I wrote this article for:

You can materialize all the above files on disk

That is, for the dashboard, we dont write a custom editor extension that materializes a virtual text buffer, but rather directly create a .jj/status.jj file on disk and open it in the editor. The editor watches the file for changes, so we can update the on-disk buffer to update the user.

And I think most of the interactions you need to implement Magit-like experience are expressible in LSP:

  • Semantic Tokens gives syntax highlighting for diffs&dashboard.
  • Folding Ranges gives folding for diffs.
  • Goto Definition gives click hunk to jump to working copy behavior for diffs. It can also be used by the dashboard for materializing additional files. The dashboard probably shouldnt show diffs for any commits but the last. Rather, goto definition on a commit zxf should materialize a .jj/zxf.diff file and then jump there.
  • And of course, Code Actions can provide an interface for arbitrary commands. For example, if you want to reword a commit, you could place your cursor on it in the dashboard, and pick reword from the code-actions context menu.

What you wouldnt get out of LSP is Magits sleek one-letter mnemonics for actions. In Magit, the status buffer is read-only, so rewording a commit is a simple as placing a cursor on it and typing r. This level of integration I think will require some tighter interaction with the editor, but this should be a relatively thin layer.

Bigger Picture

Its useful to take a birds eye view here. You can think of a VCS as a system for storing your source code files. But that is the illusion, the reality is the opposite. What VCS actually does is it stores a complex graph of objects linked through content hashes. You use VCS to mutate this graph. Now, mutating the graph manually is a pain, so VCS resorts to using your working copy as a convenient way to author graph changes! To view a node in the graph, it is checked out into the working copy. To insert a new node, you save the working copy as this node. See how bidirectional human-computer interaction is mediated through a mutable file system!

In other words, Git, and jj, already use the file system as the user interface paradigm at the macro level (authoring new commits) and at the micro level (authoring commit messages). The middle-end is missing though! git status is a CLI utility, but it could be just a file on disk!