Pearl Edits and Reflects Linear
Features | Installation | Quick Start | Commands | Configuration | Development & Testing | FAQ | History | License
" We like lists because we don't want to die. " — Umberto Eco
Pearl (backronym: Pearl Edits and Reflects Linear) brings Linear issues into Emacs as a working Org file. Fetch your open issues, a project, a Linear Custom View, an ad-hoc filter, or a saved local query; Pearl renders each issue as an Org heading with the description and comments in the body and Linear metadata in a namespaced property drawer. Edit what you need, then push it back explicitly with conflict-aware commands.
Features
- Fetch open issues, project issues, server-side Linear Custom Views, ad-hoc filters, or named local saved queries
- Read issues as an Org outline: title as heading, description in the body, comments as a chronological subtree
- Keep structured fields in
LINEAR-*properties: id, URL, team, state, priority, assignee, labels, timestamps, and sync hashes - Edit descriptions, titles, and your own comments in place, then push them with a three-way conflict check
- Set priority, state, assignee, and labels by command, using Linear ids behind display-name completion
- Add comments, create issues, delete issues, and open the current issue or view in Linear
- Refresh the active view from the source recorded in the file, or refresh one issue at point
- Optionally sync Org TODO keyword changes back to Linear workflow states
- Use one transient dispatcher,
M-x pearl-menu, for the whole command surface - Well-tested with isolated ERT files, request fixtures, and coverage support
Installation
Pearl requires Emacs 27.1+, Org, a Linear API key, and the request, dash, s, and transient packages. transient ships with Emacs 28+, and package managers handle the rest.
MELPA
Pearl is not on MELPA yet.
package-vc-install (Emacs 29+)
(unless (package-installed-p 'pearl)
(package-vc-install "https://git.cjennings.net/pearl.git"))use-package with :vc (Emacs 29+)
(use-package pearl
:vc (:url "https://git.cjennings.net/pearl.git" :rev :newest)
:commands (pearl-menu pearl-list-issues pearl-run-view pearl-new-issue)
:bind ("C-c L" . pearl-menu)
:custom
(pearl-org-file-path (expand-file-name "gtd/linear.org" org-directory))
:config
(pearl-load-api-key-from-env))Straight
(straight-use-package
'(pearl :type git :repo "https://git.cjennings.net/pearl.git"))Doom Emacs
In packages.el:
(package! pearl
:recipe (:type git :repo "https://git.cjennings.net/pearl.git" :files ("*.el")))In config.el:
(use-package! pearl
:commands (pearl-menu pearl-list-issues pearl-new-issue
pearl-run-view pearl-enable-org-sync)
:init
(setq pearl-org-file-path (expand-file-name "gtd/linear.org" org-directory))
:config
(pearl-load-api-key-from-env))Manual
git clone https://git.cjennings.net/pearl.git(add-to-list 'load-path "/path/to/pearl")
(require 'pearl)Install request, dash, and s first if your package manager did not already pull them in.
Quick Start
Create a Linear API key from Linear's Settings -> Account -> API -> Personal API Keys.
Put it somewhere Pearl can read it. The simplest development setup is:
export LINEAR_API_KEY=lin_api_...Then load it from Emacs:
(pearl-load-api-key-from-env)For normal use, keep the key in
auth-sourceinstead:machine api.linear.app login apikey password YOUR_API_KEY(setq pearl-api-key (auth-source-pick-first-password :host "api.linear.app"))Choose the Org file Pearl owns:
(setq pearl-org-file-path (expand-file-name "gtd/linear.org" org-directory))Run
M-x pearl-menu, or start withM-x pearl-list-issues.
Pearl writes one active Org file. Running a different query or view replaces that file's contents after checking for dirty buffers. Refresh commands reuse the source stored in the file header.
Commands
Command menu
M-x pearl-menu opens a transient dispatcher with commands grouped by fetch, view, issue-at-point, creation, Org sync, and setup. Bind that command if you use Pearl regularly:
(global-set-key (kbd "C-c L") #'pearl-menu)Every command below is also available directly through M-x.
Fetching and refreshing
| Command | What it does |
|---|---|
pearl-list-issues |
Fetch your open issues |
pearl-list-issues-by-project |
Fetch open issues in a chosen project |
pearl-list-issues-filtered |
Build an ad-hoc issue filter interactively |
pearl-run-view |
Run a Linear Custom View server-side |
pearl-run-saved-query |
Run a named local query from pearl-saved-queries |
pearl-refresh-current-view |
Re-run the source recorded in the active file |
pearl-refresh-current-issue |
Re-fetch the issue at point |
Ad-hoc filtering starts with a team, then completes states, projects, and labels from that team's actual Linear values. Saved queries are local Lisp data:
(setq pearl-saved-queries
'(("My open work" :filter (:assignee :me :open t) :sort updated :order desc)
("Open bugs" :filter (:labels ("bug") :open t) :sort priority :order asc)))Sorting is local and deterministic. Query filters are AND-only; use a Linear Custom View for OR-heavy logic.
Editing issues
All issue commands work from anywhere inside an issue subtree.
| Command | What it changes |
|---|---|
pearl-sync-current-issue |
Push the edited description body |
pearl-sync-current-issue-title |
Push the edited heading title |
pearl-set-priority |
Set None, Urgent, High, Medium, or Low |
pearl-set-state |
Set a workflow state from the issue's team |
pearl-set-assignee |
Set a team member as assignee |
pearl-set-labels |
Replace labels; empty selection clears them |
pearl-add-comment |
Add a new Linear comment |
pearl-edit-current-comment |
Push edits to one of your own comments |
pearl-delete-current-issue |
Soft-delete the current issue after confirmation |
pearl-open-current-issue |
Open the issue URL in a browser |
pearl-open-current-view-in-linear |
Open the active view in Linear |
Description, title, and comment pushes use the same conflict gate: unchanged local text sends nothing; a local edit against an unchanged remote pushes; if both local and remote changed since fetch, Pearl refuses to clobber either side and reports the conflict. Destructive conflict choices stash local text in *pearl-conflict-backup* first.
Only comments you authored are editable. Pearl refuses edits to comments from another person, a bot, or an integration before making an API call.
Org TODO sync
pearl-state-to-todo-mapping maps Linear state names to Org TODO keywords for rendering and optional state sync:
(setq pearl-state-to-todo-mapping
'(("Todo" . "TODO")
("In Progress" . "IN-PROGRESS")
("In Review" . "IN-REVIEW")
("Backlog" . "BACKLOG")
("Blocked" . "BLOCKED")
("Done" . "DONE")))Make sure your org-todo-keywords include every keyword you map to. Enable and disable the save/TODO-change hooks with:
(pearl-enable-org-sync)
(pearl-disable-org-sync)The Org File
A fetched Pearl file is intentionally readable. The header records the source, run time, filter summary, count, and whether pagination truncated the result. Issues sit under one top-level view heading:
#+title: Linear - My open issues
#+STARTUP: show3levels
#+TODO: TODO IN-PROGRESS IN-REVIEW BACKLOG BLOCKED | DONE
#+LINEAR-SOURCE: (:type filter :name "My open issues" :filter (:assignee :me :open t))
#+LINEAR-RUN-AT: 2026-05-23 19:30
#+LINEAR-COUNT: 12
#+LINEAR-TRUNCATED: no
* My open issues
** TODO [#B] ENG-123 Issue title
:PROPERTIES:
:LINEAR-ID: <uuid>
:LINEAR-IDENTIFIER: ENG-123
:LINEAR-URL: https://linear.app/.../ENG-123
:LINEAR-STATE-NAME: In Progress
:LINEAR-ASSIGNEE-NAME: Craig
:LINEAR-LABELS: [bug, p1]
:LINEAR-DESC-SHA256: <last-fetched markdown hash>
:LINEAR-DESC-ORG-SHA256: <rendered Org body hash>
:END:
The issue description renders here as Org and can be edited in place.
*** Comments
**** Author Name - 2026-05-23T10:00:00.000Z
A comment, oldest first.The LINEAR-* properties store both ids and display names so common commands do not need a network lookup just to render. The hash properties are provenance for conflict-aware sync.
Configuration
Most users only need an API key and an output path. The rest are knobs for teams with large issue sets or stronger preferences about window behavior.
| Variable | Purpose |
|---|---|
pearl-api-key |
Linear API key |
pearl-org-file-path |
Active Org output file |
pearl-default-team-id |
Default team for issue creation |
pearl-saved-queries |
Named local issue queries |
pearl-state-to-todo-mapping |
Linear state to Org keyword mapping |
pearl-max-issue-pages |
Pagination cap, 100 issues per page |
pearl-request-timeout |
Synchronous request timeout in seconds |
pearl-fold-after-update |
Re-fold the active page after fetch/refresh |
pearl-surface-buffer |
Show the active buffer after a command updates it |
pearl-surface-select-window |
Move focus to the surfaced buffer |
pearl-debug |
Log request/response details to *Messages* |
If a fetch stops at the pagination cap, Pearl writes #+LINEAR-TRUNCATED: yes in the file header. Raise pearl-max-issue-pages if your result set is larger than the default 1000 issues.
Development & Testing
Clone the repo and install the Eask-managed dependencies:
git clone https://git.cjennings.net/pearl.git
cd pearl
make setupUseful development targets:
| Target | What it does |
|---|---|
make test |
Run unit and integration tests, excluding :slow tests |
make test-all |
Run every test, including :slow tests |
make test-unit |
Run unit tests only |
make test-integration |
Run integration tests only |
make test-file FILE=mapping |
Run one test file by fuzzy match |
make test-one TEST=priority |
Run one test by fuzzy match |
make test-name TEST'test-pearl-map-*'= |
Run tests matching an ERT selector |
make coverage |
Generate undercover/simplecov coverage data |
make compile |
Byte-compile pearl.el |
make lint |
Run the lint/checkdoc path from the test harness |
make validate |
Check parentheses across source and tests |
make clean |
Remove test artifacts and coverage output |
Each test file runs in its own Emacs batch process for isolation. See TESTING.org for the full test guide, naming conventions, fixture helpers, and coverage notes.
FAQ
I edited the heading title and synced the description. Why did the title stay the same?
Titles and descriptions push through separate commands. Use pearl-sync-current-issue for the body and pearl-sync-current-issue-title for the heading title.
Why did square brackets disappear from a synced title?
Pearl strips [ and ] from titles before rendering so Org does not misparse them. A title like Fix [URGENT] bug round-trips as Fix URGENT bug.
Why did a Markdown heading or single-asterisk italic change after syncing?
The Markdown-to-Org round trip is intentionally lossy for a few constructs. Markdown # heading renders as a bold line, and single-asterisk *italic* is read as bold. Use these constructs carefully in descriptions you plan to edit from Org.
Why did Pearl refuse my comment edit?
Linear only lets you edit comments you authored. Pearl checks that before pushing and refuses edits to other people's comments, bot comments, and integration comments.
Why did Pearl refuse a description, title, or comment sync as a conflict?
The local text and remote text both changed since the last fetch. Refresh to reconcile, or use the conflict prompt. Pearl stashes local text before any destructive resolution.
Why did a hand-edited drawer field get overwritten?
Drawer fields are generated metadata. Change priority, state, assignee, and labels through Pearl commands so display names can resolve to Linear ids correctly.
Why are renamed teams, states, labels, or assignees stale?
Pearl caches Linear lookup tables. Run M-x pearl-clear-cache.
Can I keep the file expanded after refresh?
Yes. Set pearl-fold-after-update to nil.
Troubleshooting
M-x pearl-check-setupchecks whether the API key is loaded.M-x pearl-test-connectionchecks API connectivity.M-x pearl-toggle-debugenables request/response logging in*Messages*.M-x pearl-clear-cacherefreshes cached names after Linear-side changes.
History
Pearl is based on and inspired by Gael Blanchemain's linear-emacs. The package has since grown into a separately maintained Linear workflow for Org, including broader fetch modes, editable descriptions and comments, conflict-aware sync, view refresh, command menus, and a dedicated test suite.
Bug reports, feature requests, and pull requests are welcome.
License
GPL-3.0-or-later. See LICENSE.
