Rewrite the percentage-split test and fix two pre-existing stale tests that were masked behind it (the file is mode:serial, so the first failure hides the rest): - Percentage split: reorder so no whole-form operation runs between typing and the save (add rows + toggle to % first, then pick accounts and type 50/50, then save). The old order typed an amount then added a row, and apply-new-account rebuilds rows from the stale snapshot -- dropping the typed value (66.67/33.33 instead of 50/50). Same observed behavior verified, just an ordering that doesn't trip the snapshot round-trip. - Pre-populate-default-account: read the actual transaction total from the grid instead of hard-coding $400 for "row index 3" (same-date seed rows have no pinned order). - openEditModalForTransaction: drop the removed multi-step "Transaction Actions" wizard navigation; the modal is single-page, action tabs are immediately available. skill: gotchas.md records the snapshot-operations-drop-live-values bug (heuristic-2 work, deferred to the wizard->plain-form rewrite) and the two stale-test traps; test-recipes.md updates the baseline to 38 pass / 1 fail / 0 skip (transaction-edit 8/8, swap 6/6; the one failure is the unrelated navigation date-range test).
135 lines
7.7 KiB
Markdown
135 lines
7.7 KiB
Markdown
# Gotchas
|
|
|
|
GROWS every migration. One entry per surprise. Also the home for any **written exception**
|
|
to the scorecard ratchet (a metric that regressed for a documented reason).
|
|
|
|
---
|
|
|
|
## Stale `$refs` / `tippy` after a swap
|
|
|
|
A whole-form swap can run an Alpine event handler *before* the component re-initialises,
|
|
so a handler that dereferences `$refs.input.__x_tippy` or calls `tippy.show()` throws.
|
|
**Always null-guard:** `$refs.input?.__x_tippy?.hide()`, `tippy?.show()`. The
|
|
`transaction-edit-swap.spec.ts` `trackErrors()` helper fails the test on any `pageerror`
|
|
or `console.error`, which is exactly how a stale-ref throw surfaces.
|
|
|
|
## Let the server value win — don't preserve Alpine state across a server-driven change
|
|
|
|
When a server change should update a component (e.g. choosing a vendor sets its default
|
|
account), rebuild that section fresh on the swap so the server-provided value lands
|
|
without keying tricks. The bug this prevents: "changing the vendor a *second* time doesn't
|
|
update the account" because preserved Alpine state shadowed the new server value. If you
|
|
*must* preserve a component, key it by value so a change forces re-init:
|
|
`(assoc attrs :key (str id "--" current-value))`.
|
|
|
|
## Focus dies if the typed input is inside its own swapped region
|
|
|
|
The single most important invariant. Amount field → swap a sibling tbody, not the row.
|
|
Memo → swap nothing. If a caret test (`sameNode`) fails, the input is in its own swap
|
|
region — re-target to a sibling/ancestor that excludes it.
|
|
|
|
## Faked cursors breed duplicate render fns
|
|
|
|
A `with-cursor`/`MapCursor` re-root to fake a deep start forces a `*-no-cursor*` twin.
|
|
Removing the fake lets you delete the twin. Don't "fix" a faked cursor in place — top-root
|
|
it and collapse to one render fn. (See `render-functions.md`.)
|
|
|
|
## Edit Clojure with clojure-mcp tools, not the file editor
|
|
|
|
`clojure_edit` / `clojure_edit_replace_sexp`. If a file won't compile: `clj-paren-repair`
|
|
the file, then retry; if still broken, `lein cljfmt check`. Run tests via `clojure-eval` /
|
|
`clj-nrepl-eval -p PORT`, never `lein test` (slow, last resort).
|
|
|
|
## Solr/typeahead in tests
|
|
|
|
Account/vendor search is backed by Solr, unavailable in tests. To drive a typeahead in
|
|
e2e: type under the 3-char threshold, then inject a result into Alpine state
|
|
(`Alpine.$data(el).elements = [{value, label}]`) and click it — the real click handler,
|
|
`tippy.hide()`, Alpine reactivity, and the HTMX swap all run as in production. Entity ids
|
|
come from `GET /test-info`.
|
|
|
|
---
|
|
|
|
## UI-only control fields must be stripped before a Datomic upsert
|
|
|
|
The wizard snapshot/step-params carry UI control fields that are **not** schema
|
|
attributes — `:action`, `:amount-mode`, and (added by the simple/advanced work) `:mode`.
|
|
The `:manual` save handler stripped `:action`/`:amount-mode` but not `:mode`, so every
|
|
*advanced* manual save passed `:mode "advanced"` into `:upsert-transaction` and 500'd with
|
|
`:db.error/not-an-entity :mode`. Lesson: when a save derives its tx-data from the form
|
|
snapshot, **strip every non-schema control key** before transacting. The session-backed
|
|
wizard engine (Phase 6) avoids this class of bug by storing per-step *validated* data
|
|
only — UI control fields never enter the combined data. This was a real production bug
|
|
surfaced by the e2e gate, not a test artifact.
|
|
|
|
## E2E helpers must use the Alpine **v3** API, not the v2 `__x` internal
|
|
|
|
The app loads Alpine v3 (`cdn.jsdelivr.net/npm/alpinejs@3.x.x`). The v2 internal
|
|
`el.__x.$data` is **gone** — `el.__x` is `undefined`, so any helper that pokes it silently
|
|
no-ops. A stale `selectAccountFromTypeahead` did this and left the posted account empty
|
|
(account-controlled by `x-model`, so the raw DOM `.value` you set is overwritten from
|
|
Alpine's empty state). Drive components the real way instead: `window.Alpine.$data(el)`,
|
|
open the tippy dropdown, inject `elements`, click the result — exactly as
|
|
`transaction-edit-swap.spec.ts` does. Probe with
|
|
`{ hasLegacy__x: !!el.__x, hasAlpineData: !!window.Alpine.$data(el) }`.
|
|
|
|
## Diagnosing a "modal won't close after save"
|
|
|
|
The edit modal closes on an `hx-trigger: modalclose` from a *successful* save; a
|
|
validation failure re-renders the `#wizard-form` (200), and a server exception returns 500
|
|
(caught by `wrap-error`). To find which: capture POST responses in Playwright
|
|
(`page.on('response', …)`), read the `edit-submit` body — a `<form id="wizard-form">` means
|
|
validation re-render; a `#error {…}` stack means a 500. Then serialize the form right
|
|
before save (`new FormData(document.querySelector('#wizard-form'))`) to see exactly what
|
|
posts. This is how the `:mode` 500 and the empty-account bugs above were isolated.
|
|
|
|
## De-faking a cursor is not a drop-in — `with-field-default` mutates
|
|
|
|
Tempting fix for a faked deep cursor (`with-cursor` + synthetic `MapCursor` at index 0):
|
|
replace it with `(fc/with-field-default 0 {})` to advance naturally. **It broke the
|
|
simple-mode swap** (`transaction-edit-swap` test 1 threw). `with-field-default` calls
|
|
`cursor/transact!` — it *mutates the form cursor* (assoc-ing the default row) as a render
|
|
side effect, which changes simple-mode behavior. The read-only synthetic `MapCursor` did
|
|
not. Lesson: removing a faked cursor on these modals is **not** a one-liner — it's part of
|
|
the larger render-fn extraction (render the row from explicit data, construct field names
|
|
directly, look up errors explicitly), done when the simple/advanced rows are reworked into
|
|
pure render fns / Selmer. Don't swap one cursor primitive for another and assume parity;
|
|
verify against the swap spec, and expect the de-fake to come with the render-fn rewrite.
|
|
|
|
## Snapshot operations read stale state and drop live form values (heuristic 2)
|
|
|
|
The whole-form operation handlers (`apply-new-account`, `apply-remove-account`,
|
|
`apply-toggle-amount-mode`) rebuild the account rows from the **decoded `:snapshot`** (the
|
|
hidden EDN field), not from the live posted `:step-params`. So any value the user has typed
|
|
but that hasn't been re-serialised into the snapshot yet — e.g. an amount typed right
|
|
before clicking "New account" — is **silently lost** when the operation re-renders. This is
|
|
the snapshot round-trip fragility the migration removes (heuristic 2: → 0 merges; state
|
|
should ride in the form, not a parallel snapshot). It bit the percentage-split e2e: typing
|
|
50% then adding a row reverted the first row to its snapshot value, yielding a 66.67/33.33
|
|
split. Two ways it shows up and how to handle until the snapshot is gone:
|
|
|
|
- **In tests:** order interactions so no whole-form operation runs between typing and the
|
|
save (toggle/add/remove *first*, then pick accounts and type, then save). The
|
|
account→location and amount→totals swaps are *targeted* (don't rebuild rows), so they're
|
|
safe between typing and save.
|
|
- **The real fix** (deferred to the wizard→plain-form rewrite): operations read the live
|
|
`:step-params` rows (coercing string amounts/ids), or there is no snapshot at all and the
|
|
posted form *is* the state.
|
|
|
|
## Characterization tests rot against table order and removed wizard chrome
|
|
|
|
Two stale-test traps surfaced once the masking failure was fixed (a `mode: 'serial'` file
|
|
hides every test after the first failure, so fixing one unmasks the next):
|
|
|
|
- **Hard-coded amounts per table row index** (`openEditModal(page, 3)` then
|
|
`expect(amount).toBeCloseTo(400)`) break because same-date seed transactions have no
|
|
pinned row order. Read the actual value (e.g. the grid's `.account-grand-total-row`)
|
|
instead of hard-coding.
|
|
- **Helpers that navigate the old multi-step wizard** (`click('button:has-text("Transaction
|
|
Actions")')`) hang once the modal is single-page. Drop the navigation; the action tabs
|
|
are present immediately.
|
|
|
|
## Scorecard exceptions (ratchet violations with a reason)
|
|
|
|
_None yet._ Append here if a migration must let a metric regress for a documented reason.
|