**REQUIRED** - Always activate FIRST on any git/VCS operations (commit, status, branch, push, etc.), especially when HEAD is detached. If `.jj/` exists -> this is a Jujutsu (jj) repo - raw git commands can corrupt data. Essential git safety instructions inside. DO NOT IGNORE.
80
74%
Does it follow best practices?
Impact
88%
1.27xAverage score across 3 eval scenarios
Advisory
Suggest reviewing before use
Optimize this skill with Tessl
npx tessl skill review --optimize ./jujutsu/SKILL.mdThis skill helps you work with Jujutsu, a Git-compatible VCS with mutable commits and automatic rebasing.
Tested with jj v0.37.0 - Commands may differ in other versions.
When running as an agent:
-m flags to provide messages inline rather than relying on editor prompts:# Always use -m to avoid editor prompts
jj desc -m "message" # NOT: jj desc
jj squash -m "message" # NOT: jj squash (which opens editor)Editor-based commands will fail in non-interactive environments.
jj st after mutations (squash, abandon, rebase, restore) to confirm the operation succeeded.In jj, your working directory is always a commit (referenced as @). Changes are automatically snapshotted when you run any jj command. There is no staging area.
There is no need to run jj commit.
CRITICAL: Unlike git, jj commits can be freely modified after creation. You can update descriptions, squash changes, rebase, and absorb — all without creating new commits. See "Essential Workflow" below for the recommended working pattern.
tqpwlqmp) that persists when a commit is rewritten — prefer these when referencing commits3ccf7581) that changes when commit content changesjj uses a revset language to select commits in commands. Common revsets:
@ — the working copy commit@- — the parent of the working copy::@ — all ancestors of @@:: — all descendants of @trunk()..@ — commits between trunk and @ (your branch)bookmarks() — all commits with bookmarksUse revsets with -r flags: jj log -r 'trunk()..@'
Always create your commit message before writing code:
Validate that you're on a blank revision with jj st. If you are not, you should type:
jj new# First, describe what you intend to do
jj desc -m "Add user authentication to login endpoint"
# Then make your changes - they automatically become part of this commit
# ... edit files ...
# Check status
jj stEach commit should represent ONE logical change. Use this format for commit messages:
Examples:
- "Add validation to user input forms"
- "Fix null pointer in payment processor"
- "Remove deprecated API endpoints"
- "Update dependencies to latest versions"# View recent commits
jj log
# View with patches
jj log -p
# View specific commit
jj show <change-id>
# View diff of working copy (use --git for familiar +/- format)
jj diff --gitIMPORTANT: jj diff output format: The default jj diff output uses a side-by-side line number format (e.g. 26 26:) that looks very different from git's +/- prefix format. This is normal and correct — it is NOT corrupted or showing stale content. However, to avoid confusion, always use jj diff --git to get standard unified diff format with +/- lines.
# Create a new empty commit on top of current
jj new
# Create new commit with message
jj new -m "Commit message"
# Edit an existing commit (working copy becomes that commit)
jj edit <change-id>
# Edit the previous commit
jj prev -e
# Edit the next commit
jj next -eMove changes from current commit into its parent:
# Squash all changes into parent
jj squashNote: jj squash -i opens an interactive UI and will hang in agent environments. Avoid it.
Warning: jj split is interactive and will hang in agent environments. To divide a commit, use jj restore to move changes out, then create separate commits manually.
Automatically distribute changes to the commits that last modified those lines:
# Absorb working copy changes into appropriate ancestor commits
jj absorbRemove a commit entirely (descendants are rebased to its parent):
jj abandon <change-id>Reverse the last jj operation:
jj undoThis reverts the repository to its state before the previous command. Useful for recovering from mistakes like accidental abandon, squash, or rebase.
Move commits to a different parent:
# Rebase current branch onto a destination
jj rebase -d <destination>
# Rebase a specific revision (without descendants) onto a destination
jj rebase -r <change-id> -d <destination>
# Rebase a revision and all its descendants
jj rebase -s <change-id> -d <destination>
# Rebase onto trunk (common: update your branch to latest main)
jj rebase -d mainDiscard changes to specific files or restore files from another revision:
# Discard all uncommitted changes in working copy (restore from parent)
jj restore
# Discard changes to specific files
jj restore path/to/file.txt
# Restore files from a specific revision
jj restore --from <change-id> path/to/file.txtBookmarks are jj's equivalent to git branches:
# Create a bookmark at current commit
jj bookmark create my-feature -r@
# Move bookmark to a different commit
jj bookmark move my-feature --to <change-id>
# List bookmarks
jj bookmark list
# Delete a bookmark
jj bookmark delete my-feature# Clone a git repository
jj git clone <url>
# Initialize jj in an existing git repo
jj git init --colocate# Fetch all branches from the default remote
jj git fetch
# Fetch from a specific remote
jj git fetch --remote <remote-name>
# Fetch specific branches
jj git fetch -b <branch-name>After fetching, rebase your work onto the updated trunk: jj rebase -d main
This section only applies to colocated repos (where both .jj/ and .git/ exist). In non-colocated repos, do not use git commands — they will corrupt jj state.
In a colocated repository, you can use both jj and git commands with care:
Switching to git mode (e.g., for merge workflows):
# First, ensure your jj working copy is clean
jj st
# Then checkout a branch with git
git checkout <branch-name>Switching back to jj mode:
# Use jj edit to resume working with jj
jj edit <change-id>Important notes:
When the user asks you to push changes:
# Push a specific bookmark to the remote
jj git push -b <bookmark-name>
# Example: push the main bookmark
jj git push -b mainBefore pushing, ensure:
IMPORTANT: Unlike git branches, jj bookmarks do not automatically move when you create new commits. You must manually update them before pushing:
# Move an existing bookmark to the current commit
jj bookmark move my-feature --to @
# Then push it
jj git push -b my-featureIf no bookmark exists for your changes, create one first:
# Create a bookmark at the current commit
jj bookmark create my-feature
# Then push it
jj git push -b my-featurejj allows committing conflicts — you can resolve them later:
# View conflicts
jj stAgent conflict resolution: Do not use jj resolve (interactive). Instead, edit the conflicted files directly to remove conflict markers, then run jj st to verify resolution.
IMPORTANT: Because commits are mutable, always refine them before considering work done:
jj show @ or jj diff --gitjj restore to move changes out, then create separate commitsjj squash or jj absorb| Action | Command |
|---|---|
| Describe commit | jj desc -m "message" |
| View status | jj st |
| View log | jj log |
| View diff | jj diff --git |
| New commit | jj new -m "message" (use jj st first; skip if @ is empty) |
| Edit commit | jj edit <id> |
| Squash to parent | jj squash |
| Auto-distribute | jj absorb |
| Rebase | jj rebase -d <destination> |
| Abandon commit | jj abandon <id> |
| Undo last operation | jj undo |
| Restore files | jj restore [paths] |
| Create bookmark | jj bookmark create <name> |
| Fetch remote | jj git fetch |
| Push bookmark | jj git push -b <name> |
72207e4
If you maintain this skill, you can claim it as your own. Once claimed, you can manage eval scenarios, bundle related skills, attach documentation or rules, and ensure cross-agent compatibility.