# Session handoff — 2026-05-21 (30)
Thirtieth handover. This session did two things:
1. **Started ADR-0033 Phase 3 (SQL DML)** — wrote the Phase-3
implementation plan, then shipped sub-phases **3a** (DSL/SQL
dispatch mechanism) and **3b** (SQL `INSERT` grammar +
minimal execution).
2. **Ran an advanced-mode completion/hint bug-fixing interlude**
triggered by manual testing — fixed the dead advanced-mode
ambient assistance plus two completion bugs (F1, F2), and
triaged several other reports.
The next session should **finish the interlude** (one bug left,
**F5**, which needs the user's OK because it's a core walker
change) and then **resume Phase 3 at sub-phase 3c**
(`INSERT … SELECT`).
## §1. State at handoff
**Branch:** `main`. **Tests: 1473 passing, 0 failing, 1 ignored.**
**Clippy:** clean (`cargo clippy --all-targets -- -D warnings`).
**Commits this session** (newest first; the last three are
local-only, `origin/main` is at `4e16d97`):
```
1c8cbc1 completion+hint: F1/F2 advanced-mode completion fixes (local)
ed40445 ui: re-enable advanced-mode ambient assistance (ADR-0022 Amendment 1) (local)
c873631 grammar+db: 3b — SQL INSERT grammar + minimal execution (ADR-0033 §1) (local)
4e16d97 walker: 3a — category-grouped mode-aware dispatch (ADR-0033 Amendment 1)
a37a0b7 docs: ADR-0033 Phase 3 — implementation plan + cross-cut matrix
```
(Unpushed commits are a normal working state; pushing is the
user's step — do not prompt about it.)
## §2. Phase 3 — the main job (ADR-0033, SQL DML)
**Controlling docs (read both):**
- `docs/adr/0033-sql-dml-grammar.md` — the decision. **Read
Amendment 1** especially (it supersedes §2's dispatch
mechanism — see below).
- `docs/plans/20260520-adr-0033-phase-3.md` — the build order
(sub-phases 3a–3k), per-sub-phase exit gates + DA gates, the
cross-cut verification matrix (filled in during 3k), and the
**standing authorizations** (commits pre-authorized within
the plan — but see §5 below: the user has since asked to
approve each commit and watch context, so confirm commits).
### 2.1. Done
**Sub-phase 3a — DSL/SQL dispatch mechanism (commit `4e16d97`).**
The ADR originally specified `Node::Guard` + `Choice(SQL, DSL)`.
**That was found unworkable during 3a** (any guard-in-`Choice`
forces a `walk_choice` change; `walk_choice` only falls through
on `NoMatch`, and `walk_seq` treats a `NoMatch` past `idx 0` as a
hard `Failed`). It was replaced — **with the user's explicit
approval** — by **category-grouped, mode-aware dispatch**,
recorded in **ADR-0033 Amendment 1**:
- Each `REGISTRY` entry is tagged `CommandCategory::{Simple,
Advanced}` (in `src/dsl/grammar/mod.rs`), generalising the old
whole-command `is_advanced_only` gate.
- `walker::walk` is now a thin dispatcher over `decide` +
`walk_one_command` + scratch helpers (`src/dsl/walker/mod.rs`).
Simple mode commits the DSL node (or emits "this is SQL");
Advanced tries Advanced candidate(s) first, DSL as fallback.
- A *shared* entry word carries a node in both groups; the
dispatcher selects by mode. **No `Node::Guard`, no
`walk_choice`/`walk_seq` change.**
**Sub-phase 3b — SQL `INSERT` grammar + minimal execution
(commit `c873631`).**
- `src/dsl/grammar/sql_insert.rs::SQL_INSERT_SHAPE` — `INTO
[ (cols) ] VALUES tuple(s)`, `__rdbms_*` target
rejection (reuses `sql_select::reject_internal_table`, now
`pub(crate)`).
- `Command::SqlInsert { sql, target_table }`;
`Request::RunSqlInsert` + `do_sql_insert` worker (tx-guarded:
execute → `finalize_persistence` for CSV + history → commit, so
failures roll back and don't re-persist). Auto-show is
best-effort via `last_insert_rowid` range.
- **Isolated behind a temporary dev entry word `sqlinsert`**
(Advanced category) so the SQL path is testable WITHOUT making
`insert` a shared word yet. Tests use `sqlinsert into …`.
- **Deviations from the plan's literal `Command::SqlInsert` field
list** (to avoid dead-code under `-D warnings`):
`listed_columns` is deferred to **3d** (shortid auto-fill) and
`returning` to **3g** — the sub-phases that actually read them.
- Tests: 6 grammar accept/reject (`sql_insert.rs`) + 8
integration (`tests/sql_insert.rs`).
### 2.2. Remaining sub-phases (resume here after the interlude)
Per the plan, in order: **3c** `INSERT … SELECT` → **3d**
`shortid` auto-fill (adds `listed_columns`) → **3e** `UPDATE` →
**3f** `DELETE` + cascade summary → **3g** `RETURNING` (adds
`returning` flag) → **3h** UPSERT (`ON CONFLICT`) → **3i**
diagnostics (`insert_arity_mismatch`, `auto_column_overridden`,
`not_null_missing`) → **3j** dispatch wiring → **3k**
verification sweep.
**Things to remember for later sub-phases:**
- **3j** is where the dev `sqlinsert` word is removed and
`insert` becomes a real shared word (register the SQL INSERT
node as `Advanced` alongside the DSL `insert` `Simple` node).
With the grouping mechanism this is mostly registry wiring.
**3j must also add de-duplication** to the completion
entry-word lists (`completion_probe_in_mode` /
`expected_at_input_in_mode` in `src/dsl/walker/mod.rs`): once
`insert` appears twice in `REGISTRY`, the entry-word candidate
lists would show it twice. (Flagged in ADR-0033 Amendment 1.)
- **3j** must also confirm advanced-mode DSL inserts don't
regress through the (now-shared) SQL path — which is why 3j
comes after 3d's shortid-auto-fill parity.
## §3. The advanced-mode completion/hint interlude
Triggered by the user manually testing advanced mode. **Paused
Phase 3 to harden it** because broken ambient assistance poisons
every advanced-mode test we'd write.
### 3.1. Fixed and committed
**Ambient assistance re-enabled in advanced mode (ADR-0022
Amendment 1, commit `ed40445`).** `render_hint_panel` hard-
returned `None` for advanced mode (a stale ADR-0022 §12 gate
predating the SQL grammar), and the hint resolver defaulted to
`Mode::Simple`. Now: `ambient_hint_in_mode` +
`hint_resolution_at_input_in_mode` + `expected_for_hint_snapshot`
thread `Mode`; `render_hint_panel` calls ambient for all modes;
the one-shot `:` sigil is stripped (`strip_one_shot_prefix`)
before the ambient walk. App-level render test
(`ui::advanced_mode_hint_panel_surfaces_sql_candidates`) +
ambient-layer locks. (Phase 2 had marked SQL hint/completion
"green" but only at the *engine* layer, never the UI — the gap
this fix closes.)
**F1 + F2 (commit `1c8cbc1`).**
- **F1** — a premature "no such table/column" ERROR shadowed the
completion for the token being typed (the hint panel *is* the
completion UI; there is no separate popup). Fix in
`input_render.rs::ambient_hint_in_mode`: suppress an
under-cursor ERROR diagnostic when a completion exists for the
(non-empty) partial it overlaps; fall through to candidates.
Genuinely-unknown names still error; WARNINGs unaffected; both
modes.
- **F2** — `select from T` (after deleting `*`) offered
the global column list instead of `T`'s columns, because the
§10.6 look-ahead's full-input walk can't reach `FROM` through
an empty projection. Fix in
`completion.rs::candidates_at_cursor_with_in_mode`: when the
look-ahead finds no scope, retry with a neutral `1 ` placeholder
inserted at the cursor so the trailing `FROM`/CTE scope is
recovered for narrowing (only the repaired walk's
`from_scope`/`cte_bindings` are consumed).
### 3.2. Triaged as NOT bugs (no work needed)
- **F3** ("`order by Name`" flagged red / no suggestions) — was a
**downstream symptom of the old-project migration gap** (see
§4), not an ORDER BY grammar bug. On a fresh db it does not
occur (`order by ` parses clean, narrows
correctly). Confirmed by reproduction.
- **F4** ("only one error span highlights") — **multi-span
highlighting works** (two simultaneous unknown columns both get
flagged). The "des red while typing" the user saw is as-you-type
harshness on a *partial keyword* (`desc` mid-typed reads as
trailing junk → parse error), not a single-span limit.
- **Case-sensitivity** (table-name case vs. column narrowing) —
**not a bug**. `columns_for_table` (`completion.rs:90`) uses
`eq_ignore_ascii_case`; verified `from orders` / `from ORDERS`
/ `from Orders` all narrow correctly. An earlier DA-gate "flag"
about this was an unverified guess and was **retracted**.
### 3.3. REMAINING — F5 (needs the user's OK before coding)
**F5 — the `order by ` (and projection) candidate list is padded
with ~13 invalid clause keywords** (`as`, `left`, `right`,
`full`, `cross`, `inner`, `join`, `where`, `group`, `having`,
`union`, `intersect`, `except`). The columns *are* present and
correctly filtered, but the noise shoves them off-screen. The
user is fine with keywords-before-columns ordering **as long as
the keyword list is reasonably filtered** — i.e. fix the spurious
keywords.
**Confirmed mechanism** (probed): these are stale
*skipped-optional* clause keywords from the clauses positioned
**between the last-consumed token and `order by`** (the FROM's
JOIN options, WHERE/GROUP BY/HAVING, set-ops). They accumulate in
`walk_seq`'s `pending_skipped` and get merged into ORDER BY's
`Incomplete` expected set — even though ORDER BY consumed bytes
and committed *past* those optionals. Control inputs proved it:
`… where id = 1 order by ` drops `where`+JOIN keywords (consumed),
leaving `group/having/set-ops`; `select 1 order by ` (no FROM)
*adds* `from`.
**Proposed fix (CORE walker change — get user OK):** in
`src/dsl/walker/driver.rs` `walk_seq`, the **`Incomplete` arm**
should NOT merge the accumulated `pending_skipped` when the child
that produced the `Incomplete` consumed bytes (i.e. committed past
those optionals). The `NoMatch` arm (child consumed nothing →
optionals are genuine alternatives at the cursor) keeps merging.
**Blast radius: all parsing (1469+ tests).** This is exactly the
kind of core change the project is careful about and the user
asked to approve it first. Do it test-first: a failing test that
asserts the `order by ` expected set excludes
`where`/`group`/`join`/`union`/etc., then the fix, then full
suite + clippy.
## §4. Deferred — old-project migration (do NOT fix now)
The user's first repro used an **old project**
(`~/.local/share/rdbms-playground/projects/20260521-repro-1/`,
`created_at: 2026-05-09`) whose `__rdbms_playground_columns` is
the **pre-`check_expr` 3-column schema**. Current code's
`read_schema` (`db.rs:4197`) hard-references `m.check_expr`, so:
- `show table` crashes (`no such column: m.check_expr`);
- `describe_table` fails for every table → `build_schema_cache`
(`runtime.rs:992`) **silently swallows the error** →
`table_columns` ends up empty → degraded completion (this was
the real cause of the "F3" symptoms above).
**The user explicitly deferred this** — "having a migration story
at the end is perfectly fine." So: **no migration work now.** When
the migration framework lands (ADR-0015 Iter 6, pending), the
targeted fix is to ensure-current-metadata-columns on open
(idempotent `ALTER TABLE … ADD COLUMN check_expr TEXT`), make
`read_schema` not hard-crash on a missing internal column, and
stop `build_schema_cache` from silently swallowing the describe
error (it masked this for a long time). ADR-0015 would get an
amendment. **Recorded here so it isn't lost.**
## §5. Process pins (important for the next session)
- **Confirm every commit.** The ADR-0033 plan pre-authorized
commits, but mid-session the user asked to **approve each commit
and watch context use** — so propose the commit message and wait
for the go-ahead. No auto-commit at sub-phase gates.
- **Push is the user's step.** Never push; never prompt about it.
- **No AI attribution** in commits (global rule).
- **Test-first discipline** held throughout this session and
should continue: reproduce a bug with a FAILING test first, then
fix. For new features, exit-gate tests before "done."
- **Don't trust unverified diagnoses** — this session twice
produced wrong guesses (the case-sensitivity "bug"; F3 as an
ORDER BY grammar bug) that empirical probes disproved. Probe
before fixing.
- **DA discipline:** specific critiques first, verdict last;
a clean PASS with no critiques is a process failure.
- **Core walker changes (`walk_seq`/`walk_choice`) need explicit
user OK** before coding (the F5 fix is the pending example).
## §6. How to take over
1. **Read, in order:** this file → `docs/adr/0033-sql-dml-grammar.md`
(esp. **Amendment 1**) → `docs/plans/20260520-adr-0033-phase-3.md`
→ `docs/adr/0022-ambient-typing-assistance.md` Amendment 1 →
`CLAUDE.md`.
2. **Baseline:**
```
cargo test # expect 1473 passing / 0 failing / 1 ignored
cargo clippy --all-targets -- -D warnings # clean
```
3. **First: finish the interlude — F5.** Confirm with the user
that the core `walk_seq` change is OK (it's the only blocker),
then implement test-first per §3.3.
4. **Then: resume Phase 3 at sub-phase 3c** (`INSERT … SELECT`)
per the plan, continuing 3c → 3k. Remember the 3j notes in §2.2
(remove `sqlinsert` dev word, make `insert` shared, add
completion-list de-dup) and the deferred `listed_columns`/
`returning` Command fields (3d/3g).
5. **Escalate** anything not settled in ADR-0033 / the plan;
don't classify work "out of scope" without user confirmation.