generated from eblume/project-template
Some checks failed
Build / validate (pull_request) Failing after 2s
Backend: enrich `list` to return titled RankedTask rows (title + canonical_context_id, via a shared ranked_from_row with `next`), so the Organizational view needs no N+1 node.get. TDD: query_surface test asserts list rows carry title + context id. Plugin: - view.lua: Tactical `next` + Organizational `list` rendered scratch buffers; <CR> opens the row's canonical-context doc. Narrowed the node autocmd to heph://node/* so view buffers (heph://next, heph://list) don't trip it. - task.lua: capture, set-attention, done/drop, skip, per-task log append, all resolving "the current task" from the buffer (a task node, or a context doc via its canonical-context backlink). - picker.lua: vim.ui.select with Telescope auto-upgrade (headless-safe). - command.lua: :Heph next/list/capture/attention/done/drop/skip/log/search. e2e: capture→next→open context→add/check checklist→done; recurring fresh-checklist (complete rolls forward in place, next occurrence all-unchecked — the §4.4 hard requirement). 6 specs green via `mise run test-nvim`. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
6.3 KiB
6.3 KiB
Begin the v1 prototype (Phase 1, tech-spec §11.1), built in TDD slices:
- Cargo workspace +
heph-corecrate; migration-run SQLite schema (§4.5); clock-injectedStoretrait +LocalStorenode create/get; single local-user bootstrap. - Markdown extraction (§5):
[[wiki-links]]and GFM- [ ]checkbox context-items derived purely and idempotently from a body, skipping code blocks. - Committed tasks (§4.3, §6):
task.createauto-creates the canonical contextdoc+canonical-contextlink; attention/do-date/late-on/state/recurrence columns; set-state/set-attention. Links CRUD (outgoing/backlinks). A body update reconcileswikilinks (diff-based, resolved by alias/title, idempotent). - "What is next?" ranking (§7): pure, clock-injected, two-stage engine — candidacy filter (do-date as a boolean gate only) then a reorderable list of named dimensions (past-late-on → overdue-amount → attention band → FIFO).
late_onis the sole urgency signal; blue hidden; red always shown. Proptest-checked total order.Store::nextsurfaces it over SQLite. - Recurrence — roll-forward in place (§4.4): completing a recurring task resets its checklist to all-unchecked, logs the occurrence, and advances the do-date to the next RRULE instance after now (skipping misses) — completion never carries forward (proptest-checked). Per-task append-only logs (
log-of) withlog.append/log.tail;skipadvances without logging. hephddaemon, local mode (§3, §6): exclusive file lock (handoff-ready), line-delimited JSON-RPC over a unix socket exposing the node/task/next/links/log methods, with DB work on tokio's blocking pool. Synchronous client for surfaces/CLI. Model types are serde-serializable.hephCLI (§1) — a thin client of the daemon:next,task,doc,get,export. Export materializes the store to a<kind>/<id>.mdtree with YAML frontmatter + body (§5), one-way, tombstones excluded.- Sync engine, local-only (§12): real hybrid logical clock + persistent device
origin; an append-only op-log per mutation; an idempotent, order-independent merge/apply engine — last-writer-wins task scalars (discards surfaced in aconflictsqueue), OR-set links, monotonic tombstones. Two-replica convergence proven. - Body text CRDT (§5, §12, slice 8d): node bodies now merge through the
yrstext CRDT (body_crdt) instead of last-writer-wins — whole-buffer writes are diffed into the doc and the yrs delta rides the op, so concurrent edits to different regions both survive and never enqueue a conflict. - Network sync over HTTP (§6.1, §12, slice 9a):
hephd --mode serverexposes a sync hub (POST /sync/push,GET /sync/pull?after=<hlc>, axum) over the same store;hephd --mode local --hub-url <url>becomes a spoke that background-syncs its op-log with that hub (and on demand via thesync.now/sync.statusRPC). Exchange is incremental by HLC cursor (sync_state) and idempotent. The merge engine isheph-core's, unchanged. Unauthenticated/single-owner for now (auth lands with OIDC).conflicts.list/conflicts.resolveare now reachable over the daemon socket. - Client mode (§3.1, slice 9b):
hephd --mode client --server-url <url>runs with no local replica, proxying every store call to a server'sPOST /rpcendpoint (the full daemon API over HTTP). The daemon is now backend-agnostic (local/serverfront aLocalStore,clientaRemoteStore), so surfaces see the same unix-socket API in every mode. - Hub authentication (§13, slice 10a): the sync hub now verifies an OIDC bearer token on
/sync/*and/rpc— RS256-pinned JWT validation with exact issuer/audience, expiry, and a required subject; JWKS discovered and cached, refetched on key rotation (jsonwebtoken). Enabled withhephd --mode server --oidc-issuer <url> --oidc-audience <client-id>(open when unset, for local dev). A single-tenant owner gate binds the hub to the first authenticated identity and rejects any other. Verification sits behind aTokenVerifiertrait, so it's tested entirely offline (stub middleware + an adversarial battery against an in-process mock IdP). - Client authentication (§13, slice 10b):
heph auth login --hub-url <url> --issuer <url> --client-id <id>runs the OAuth 2.0 device-code flow and caches the token in the OS keyring; spokes andclientmode attach it to hub requests, refreshing on expiry (--oidc-issuer/--oidc-client-id). Offline-tested against a mock OAuth server and a full spoke-to-authenticated-hub loop. (Auth/proxy HTTP uses the runtime-freeureq, sincereqwest::blockingis unsafe inside the async daemon.) - CI runs the Rust suite (fmt/clippy/test) via the project build hook.
heph.nvimslice 11a (§8) — the primary surface begins: a Neovim plugin that is a thin client of the localhephdover its unix socket. Avim.uvJSON-RPC client (blockingcallviavim.wait, id-demuxed, partial-line buffered, JSONnull→Luanil); buffer-backed nodes (heph://node/<id>withBufReadCmd→node.get/BufWriteCmd→node.update, whole-buffer body round-tripping exactly through the CRDT);[[wiki-link]]follow on<CR>via a new exactnode.resolve {title}RPC (alias-then-title, the same mapping that materializeswikilinks — unresolved links allowed); the daily journal (:Heph today); and the:Hephcommand surface. Headless e2e (§9) drives the plugin against a real daemon over a temp socket with a self-contained busted-style runner (no external plugins, no network): journal round-trip, follow-link, and link-two-docs/backlink.heph.nvimslice 11b (§8) — task views:listis enriched to return titled rows (the same shape asnext, with the canonical-context id) so the Organizational survey needs no per-rownode.get. The plugin gains the Tactical:Heph nextand Organizational:Heph listviews (<CR>opens a task's canonical-context doc), task capture, set-attention, done/drop, skip, and per-tasklogappend — each resolving "the current task" from the buffer (a task node, or a context doc via itscanonical-contextbacklink). Avim.ui.selectpicker (Telescope auto-upgrade when installed) backs:Heph search/capture/attention. Headless e2e adds the capture→next→context→checklist→done workflow and the recurring fresh-checklist workflow (completing a recurring task rolls it forward and the next occurrence presents an all-unchecked checklist).