ghr
ghr is a fast terminal workspace for staying on top of GitHub. It brings your GitHub inbox, pull requests, issues, repo tabs, conversations, checks, and diffs into one stateful TUI, so you can triage, review, comment, approve, merge, and jump back in without waiting on a fresh GitHub fetch.
Conversation and triage view

Pull request diff review view

Features
- Inbox, pull request, and issue views.
- Snapshot-first startup: cached data is shown immediately, the active view refreshes first, and non-active PR/issue sections are kept warm by a quiet idle sweep.
- Configurable sections and repo tabs, including multi-query sections such as
Needs Attention. - Automatic current-repo tab persistence when launched inside a Git checkout with a GitHub remote.
- Paged PR and issue lists with configurable page size.
- Persistent ignored PRs and issues, stored in UI state and hidden from all lists.
- Recent Items command for jumping back to the last 20 PRs and issues where you spent at least 10 seconds in Details.
- Fuzzy filtering in every loaded list with
/, quick PR/issue section filters withf, plus repo-scoped GitHub search withS. - Details pane with rendered Markdown, clickable links and image attachments, fenced code blocks with lightweight Rust and plain/log highlighting, descriptions, comments, review comments, PR commit activity, labels, milestones, action hints, and check summaries.
- Inbox notifications lazily load linked PR or issue details when opened, so descriptions and recent PR commit activity appear without making the initial inbox fetch heavier.
- PR diff mode with a changed-file list, per-file diff rendering, inline review comments, and review ranges.
- Comment, reply, edit, milestone, merge, close/reopen, update-branch, rerun-failed-checks, local PR checkout, draft / ready-for-review, and full PR review submit flows from inside the TUI.
- Inbox read/done/mute handling with explicit commands, local cache updates, GitHub sync, and dimmed read items.
- Mouse support for tabs, lists, links, descriptions, comments, editor cursor placement, details drag-copy, scrolling, text selection mode, and split resizing.
- UI state persistence under
~/.ghr, including focus, selected item, scroll position, split ratio, and diff mode. - Local state under
~/.ghr: config, SQLite snapshot database, logs, and UI state. - Uses the GitHub CLI for authentication, API access, and browser opening behavior.
Requirements
- GitHub CLI gh
- An authenticated GitHub CLI session:
Installation
You can install ghr with Cargo:
Or install the latest release binary on macOS or Linux:
|
Install the latest release binary on Windows PowerShell:
irm https://raw.githubusercontent.com/chenyukang/ghr/main/install.ps1 | iex
Then authenticate GitHub CLI and run ghr:
The installers detect your OS and CPU architecture, download the matching GitHub release asset with visible progress, verify its .sha256, and install ghr into ~/.local/bin by default. Use GHR_INSTALL_DIR to choose another install directory, or GHR_VERSION to pin a release tag:
| GHR_VERSION=v0.7.0 GHR_INSTALL_DIR=/usr/local/bin
Usage
Run ghr from any terminal after gh auth login. When started inside a Git checkout with a GitHub remote, ghr adds that repository as a local project tab automatically.
Keybindings
Press ? in the TUI for the live shortcut reference. The top-right status shows the current app state; the footer stays focused on the most useful shortcuts for the active area.
| Key | Action |
|---|---|
: |
Open the command palette and fuzzy search every shortcut; recently run commands appear first |
Project Switch |
Filter configured repos by prefix and activate a project tab |
Project Add |
Add a repo project to the menu and save it to config.toml |
Project Remove |
Select a configured repo project, confirm, and remove it from config.toml |
Recent Items |
Fuzzy search recently viewed PRs/issues and jump back to the selected item |
Copy GitHub Link |
Copy the selected comment link, or the current PR/issue link, to the clipboard |
Copy Content |
Copy the selected comment content, or the current PR/issue description, to the clipboard |
Clear Cache |
Choose a local cache layer to clear: current section, current view, all list snapshots, suggestions, loaded details/diffs, or all cache |
Mark Done |
Move the selected GitHub inbox notification out of inbox lists; future activity can still notify unless the thread is muted |
Mark All Read |
Mark every GitHub inbox notification as read |
Mute Thread |
Ignore future notifications for the selected inbox thread |
Subscribe Thread / Unsubscribe Thread |
Change the selected inbox thread subscription |
Info |
Show version, config/db/log paths, ghr process memory usage, ignored/recent item counts, and current UI state |
1 / 2 / 3 / 4 |
Focus ghr / Sections / list / Details |
Tab / Shift+Tab |
Switch list/details focus; when ghr or Sections is focused, move within that tab group |
h / l |
Move within the focused ghr or Sections tab group, wrapping at the ends |
Enter |
Focus the details pane from the list |
Esc |
Return from details to list, clear search, or leave diff mode |
j / k |
Move list selection, choose diff files, select diff lines, or scroll details |
n / p in ghr, Sections, or List |
Mirror next/previous movement for focus or list selection; p at the first list item returns to Sections |
[ / ] in List |
Load previous/next GitHub result page |
PgDown / PgUp or d / u |
Page current list/details movement |
n / p in Details |
Focus next/previous comment in conversation or diff details; p from the first conversation comment returns to PR/issue details |
h / l in diff Details |
Page down/up through the file diff; at file edges, jump to the next/previous file |
g / G |
Jump to top/bottom in list, details, or diff; in conversation details, top clears comment focus and bottom focuses the last comment |
[ / ] in diff mode |
Switch diff files |
/ |
Fuzzy filter the current list |
f |
Filter the current PR/issue section with qualifiers such as state:closed label:bug author:alice; empty input or clear resets |
S |
Search matching PRs and issues in the current repo |
i |
Ignore the selected PR or issue and hide it from future lists |
v |
Open PR diff mode |
q in diff mode |
Return to the state before opening diff |
o |
Open the selected item in the browser; in diff mode, open the PR changes page |
a |
Add a normal issue or PR comment |
L |
Add a label to the selected issue or PR; type a prefix and choose from repo labels |
N |
In a PR list, create a PR from the repo local_dir current branch; in an issue list, create an issue |
+ |
Add a reaction to the selected issue, PR, or focused comment |
@ / - |
Assign or unassign assignees on the selected issue or PR |
c in Details |
Add a normal comment in conversation mode, or an inline review comment in diff mode |
R |
Reply to the focused comment |
e |
Edit the focused comment when it is yours; in diff mode, end a review range |
T |
Edit the selected issue or PR title/body |
m |
Toggle terminal text selection mode; in diff details, begin a review range |
M |
Open a merge confirmation for the selected PR, defaulting to merge commits |
C |
Open a close or reopen confirmation for the selected issue or PR |
X |
Open a confirmation to run gh pr checkout <number> --repo <owner/repo> from the matching local checkout |
F |
Rerun failed checks for the selected PR |
U |
Open an update-branch confirmation for the selected PR |
m / s / r in merge confirmation |
Choose merge, squash, or rebase before confirming |
Tab in merge confirmation |
Cycle merge method |
s |
Submit a PR review summary as comment, request changes, or approve |
A |
Approve through the PR review summary flow |
Ctrl+D |
Discard a pending PR review created in this session |
E |
Open an enable auto-merge confirmation for the selected PR |
O |
Open a disable auto-merge confirmation for the selected PR |
D |
Toggle the selected open PR between draft and ready for review |
P |
Request or re-request PR reviewers with comma-separated logins |
Y |
Remove pending PR review requests with comma-separated logins |
t |
Change or clear the selected issue/PR milestone |
y / Enter |
Confirm the current action in the confirmation dialog |
Enter in Reviewer Action |
Submit the reviewer login list |
Ctrl+Enter |
Send or update a comment/title/body from the editor dialog |
Ctrl+Enter in issue dialog |
Create the issue |
← / → / ↑ / ↓ in editor dialogs |
Move the cursor by character or rendered line |
Home / End in editor dialogs |
Jump to the start or end of the current line |
Ctrl+W / Alt+Backspace in editor dialogs |
Delete the previous word |
Alt+B / Alt+F / Alt+D in editor dialogs |
Jump previous/next word, or delete the next word |
Ctrl+U / Ctrl+K in editor dialogs |
Delete to the start or end of the current line |
Ctrl+X in editor dialogs |
Delete the current line |
Ctrl+Z / Cmd+Z in editor dialogs |
Undo text edits |
Ctrl+R / Cmd+Shift+Z in editor dialogs |
Redo text edits |
r |
Refresh from GitHub |
q / Ctrl+C |
Save UI state and quit |
Diff review ranges:
- Press
mon a diff line to begin a range, move the highlight, then presseto end it. - Press
cafter ending a range to post an inline review comment for the selected range. - With the mouse, single click selects one diff line. Double click begins a range, then single click the end line to complete it.
- Press
sto open the review summary editor, useTabor1/2/3to choose comment, request changes, or approve, then pressCtrl+Enterto submit. - Press
Ctrl+Pin the review summary editor to create a pending review draft, then pressslater to submit it orDto discard it.
Local PR checkout:
- Press
Xon a pull request in the list or Details pane, then confirm withyorEnter. - Pull request Details show the remote branch as a clickable link when GitHub provides it.
- Checkout runs from the matching local repository directory. Set
local_diron a repo entry to make the target explicit:
[[]]
= "Rust"
= "rust-lang/rust"
= "~/code/rust"
= true
= true
- If
local_diris not set,ghrtries the directory where it was launched when that directory has a GitHub remote for the pull request repository. If neither path matches,ghrshows a hint instead of running checkout.
Mouse behavior:
- Click ghr or Sections tabs to switch views or sections.
- Click list rows to select them and focus Details. Mouse hover and mouse wheel do not change PR/issue selection.
- Click descriptions, comments, links, and visible actions in Details to focus or open them.
- Click inside comment, review, issue, or PR editor dialogs to place the edit cursor.
- Drag text inside Details to copy that rendered selection to the clipboard without leaving mouse mode.
- Scroll Details with the mouse wheel.
- Drag the split between list and Details to resize panes; the ratio is saved.
- Press
moutside diff mode to temporarily disable TUI mouse capture for terminal text selection; pressmagain to restore mouse controls.
Default Sections
Inbox:
AllReview RequestedAssignedMentionedSubscribedOthers
Pull Requests:
Needs Attention: open PRs where you were requested for review, assigned, or mentioned.My Pull Requests: open PRs authored by you.Reviewed: open PRs reviewed by you and authored by someone else.
Issues:
Assigned to MeMentionedInvolved
Configuration
The config file is created at:
~/.ghr/config.toml
Example:
[[]]
= "Rust"
= "rust-lang/rust"
= true
= true
= ["T-compiler"]
= ["S-waiting-on-review"]
= ["E-easy"]
[]
= "pull_requests"
= ":"
= "info"
= 50
= 50
= 50
= 60
= true
[[]]
= "Needs Attention"
= [
"is:open review-requested:@me archived:false sort:updated-desc",
"is:open assignee:@me archived:false sort:updated-desc",
"is:open mentions:@me archived:false sort:updated-desc",
]
[[]]
= "My Pull Requests"
= "is:open author:@me archived:false sort:updated-desc"
[[]]
= "Reviewed"
= "is:open reviewed-by:@me -author:@me archived:false sort:updated-desc"
= ["some-org/archive-*"]
Use filters for a single GitHub search query. Use queries when a section should merge several GitHub searches into one deduplicated list. Label filters can be written directly in either form, for example filters = "is:open label:bug archived:false sort:updated-desc" or label:"good first issue" for labels with spaces.
Use [[repos]] to add repository tabs to the top bar. Each configured repo shows its name as a top-level tab; inside that tab, show_issues and show_prs control whether the sections are shown as Issues and Pull Requests. Repo tabs default to open issues and open PRs, with Issues shown first. Set labels to filter both repo issue and PR lists, or use issue_labels / pr_labels for kind-specific filters.
When ghr starts inside a Git checkout with a GitHub remote, it adds that repository as a repo tab if it is not already configured and saves it back to config.toml with local_dir set to the launch directory. If the repo already exists in the config but has no local_dir, ghr fills that field without overwriting an existing value.
Set command_palette_key to change the command palette shortcut. Printable keys such as ":" are treated as text while typing in search, filter, and editor dialogs; use a modified key such as "Ctrl+L" if you want the palette to open from those text inputs.
Set log_level to trace, debug, info, warn, or error. In debug mode, gh / gh api requests plus UI focus/view changes and mouse clicks are written to ~/.ghr/ghr.log. RUST_LOG still overrides this config value when it is set.
pr_per_page and issue_per_page control the page size used for PR and issue search sections. Use [ and ] in the list to load adjacent GitHub result pages.
Local Data
ghr keeps all local files in ~/.ghr:
config.toml: user configurationghr.db: SQLite snapshot cacheghr.log: log filestate.toml: persisted UI state, including ignored PRs/issues
The snapshot cache is intentionally local and disposable. Use the Clear Cache command to clear one cache layer, or delete ~/.ghr/ghr.db if you want to rebuild every persisted cache row from GitHub.
Contributing
Contributions are very welcome! Please open an issue or a pull request with any bug fixes, improvements, or new features.