Migrate the worker() closure in _check_auto_refresh_context_preview (L4321)
to the canonical Result[T] pattern:
- Extract _worker_context_preview_result(app) -> Result[None] helper into
new Phase 7 Worker/Background Result Helpers region.
- The legacy worker() wrapper calls the helper and drains errors to
app.controller._worker_errors (with controller._worker_errors_lock
acquired on append) per sub-track 3 Phase 6 Group 6.5 telemetry drain.
- The try/finally cleanup (setting _is_generating_preview=False and
handling _pending_preview_refresh) is preserved verbatim.
Tests:
- test_phase_7_l4321_worker_context_preview_result_success
- test_phase_7_l4321_worker_context_preview_result_failure
Audit: INTERNAL_BROAD_CATCH count in src/gui_2.py dropped from 3 to 2
(remaining: L591 _diag_layout_state, L897 _capture_workspace_profile).
The lock-protected append ensures thread-safety when multiple worker
threads call _report-style drains concurrently. The helper preserves
the original fallback behavior (app.context_preview_text =
'Error generating context preview.' on failure) so the user-visible
UX is unchanged.
Per PHASE1_SITE_INVENTORY.md, Phase 6 (signal-handler category) has 0
INTERNAL_BROAD_CATCH sites in src/gui_2.py. All sites that might appear
in a signal-handler category were classified into other phases (Phase 8
for startup callbacks, Phase 7 for worker/background).
Adds 2 invariant tests:
- test_phase_6_invariant_signal_handler_count_dropped: pins the count
to exactly 3 (the pre-Phase-7 baseline) before Phases 7-9 migrate.
- test_phase_6_invariant_zero_sites_in_phase_6: documents that no
Phase 6 site tests exist (machine-checkable: future agent adding a
Phase 6 site will see this test fail at the count assertion).
Updates state.toml: phase_6 status = completed.
Extract _render_context_batch_actions_preview_result helper from the
_do_generate preview try/except in render_context_batch_actions. The
imgui.button callback drains errors to app._last_request_errors per FR-BC-4
event-handler pattern.
[pre-audit] L3582 INTERNAL_BROAD_CATCH
[post-audit] V count: 8 -> 7 (L3582 removed)
Extract _render_tool_preset_bias_save_result helper from the BiasProfile
save try/except in render_tool_preset_manager_content. The imgui.button
callback drains errors to app._last_request_errors per FR-BC-4
event-handler pattern.
[pre-audit] L3163 INTERNAL_BROAD_CATCH
[post-audit] V count: 9 -> 8 (L3163 removed)
Extract request_patch_from_tier4_result helper from the
ai_client.run_tier4_patch_generation try/except in App.request_patch_from_tier4.
Legacy wrapper drains errors to app._last_request_errors per FR-BC-4
event-handler pattern.
[pre-audit] L1428 INTERNAL_BROAD_CATCH
[post-audit] V count: 10 -> 9 (L1428 removed)
Extract _apply_pending_patch_result helper from the apply_patch_to_file
try/except in App._apply_pending_patch. Legacy wrapper drains errors to
app._last_request_errors per FR-BC-4 event-handler pattern.
[pre-audit] L1367 INTERNAL_BROAD_CATCH
[post-audit] V count: 12 -> 11 (L1367 removed)
Extract _populate_auto_slices_outline_result helper from the
mcp_client.{py,ts_c,ts_cpp}_get_code_outline try/except in
App._populate_auto_slices. Legacy wrapper drains errors to
app._last_request_errors per FR-BC-4 event-handler pattern.
[pre-audit] L1284 INTERNAL_BROAD_CATCH
[post-audit] V count: 14 -> 13 (L1284 removed)
Per AI Agent Checklist Rule #0 (re-read per phase).
Phase 5 focuses on the 13 INTERNAL_BROAD_CATCH sites inside event handler
functions. Per the spec (FR-BC-4), the drain for event handlers is
to accumulate in app._last_request_errors or a similar per-event
accumulator (not imgui.open_popup, since the event handler is called
from a button click, not a render frame).
Event handler sites (per PHASE1_SITE_INVENTORY.md):
- L1335, L1344 (_populate_auto_slices): mcp_client calls
- L1418 (_apply_pending_patch): patch modal handler
- L1444 (_open_patch_in_external_editor): external editor launch
- L1479 (request_patch_from_tier4): tier4 patch generation
- L3214 (render_tool_preset_manager_content): modal content render
- L3633 (render_context_batch_actions): modal content render
- L5430 (render_operations_hub): tab content render
- L5836 (render_text_viewer_window): window render
- L5970 (render_external_editor_panel): panel render
- L7258 (render_beads_tab): tab render
The legacy wrapper pattern: extract a _<site>_result helper that
returns Result[bool]; the legacy wrapper routes errors to
app._last_request_errors.append((op_name, ErrorInfo(...))).
The Phase 3 invariant test (test_phase_3_invariant_batch_a_count_dropped)
asserted exactly 17 INTERNAL_BROAD_CATCH sites, the post-Phase 3 baseline.
After Phase 4 migrates 3 more sites, the count drops to 14. The test now
asserts <= 17 (the upper bound; the Phase 3 boundary).
Adds test_phase_4_invariant_batch_b_count_dropped: locks in <= 14 sites
(post-Phase 4 baseline; down from 17).
Adds test_phase_4_invariant_all_3_migration_sites_have_tests: ensures each
of the 3 Batch B sites (L3398, L3718, L3740) has both _success and _failure tests.
All 30 tests pass.
Adds _render_ast_inspector_outline_result(app, f_path) -> Result[str] helper that wraps
the mcp_client.configure + outline fetch try/except in render_ast_inspector_modal.
The data field carries the outline string so the legacy wrapper can iterate it
without an additional instance attribute. Errors drain to app._last_request_errors
(per FR-BC-3 modal pattern; data plane attribute).
Audit: BROAD_CATCH count 16 -> 15, COMPLIANT count 21 -> 22. Migration
target count drops by 1. Tests: 2/2 pass.
The Phase 1 test originally asserted exactly 42 migration-target sites.
After Phase 3 migrated 8 sites, the count dropped to 34. The test
now asserts <= 42 (the starting count) so it passes both at Phase 1
boundary and after subsequent phases migrate sites.
Per-phase invariant tests (added in Phase 3+ test files) verify the
specific expected count per phase.
TIER-2 READ conductor/code_styleguides/error_handling.md end-to-end before Phase 3.
TIER-2 READ conductor/code_styleguides/error_handling.md end-to-end before Phase 3.
Adds _render_warmup_status_indicator_result(app) -> Result[dict] helper that
wraps the controller.warmup_status() try/except in
render_warmup_status_indicator. The data field carries the status dict so
the legacy wrapper can use it for rendering without an additional instance
attribute.
render_warmup_status_indicator becomes a thin wrapper that drains errors
to app.controller._worker_errors under the controller's lock (worker error
plane; thread-safe per app_controller pattern).
Audit: BROAD_CATCH count 18 -> 17, COMPLIANT count 19 -> 20. Migration
target count drops from 42 to 34 (8 sites migrated). Tests: 2/2 pass.
TIER-2 READ conductor/code_styleguides/error_handling.md end-to-end before Phase 3.
Adds _handle_history_logic_result(app) -> Result[bool] helper that wraps
the snapshot debounce try/except from App._handle_history_logic. The
_is_applying_snapshot pre-condition guard stays in the legacy wrapper
(not error handling; the original early return has no try/except).
App._handle_history_logic becomes a thin wrapper that drains errors to
_last_request_errors. The drain failure mode is structurally safe
(hasattr check + append) so no outer try/except is required (per the
L1123 wrapper decision; avoiding new INTERNAL_SILENT_SWALLOW violations).
Audit: BROAD_CATCH count 19 -> 18, COMPLIANT count 18 -> 19. Tests: 2/2 pass.
TIER-2 READ conductor/code_styleguides/error_handling.md end-to-end before Phase 3.
Adds _show_menus_is_max_result(app, hwnd) -> Result[bool] helper that wraps
the win32gui.GetWindowPlacement try/except from App._show_menus. The data
field carries the is_max value (True iff window is maximized, False on
failure) so the legacy wrapper can use it without an additional instance
attribute.
App._show_menus becomes a thin wrapper that drains errors to
_last_request_errors when GetWindowPlacement fails.
Audit: BROAD_CATCH count 20 -> 19, COMPLIANT count 17 -> 18. Tests: 2/2 pass.
TIER-2 READ conductor/code_styleguides/error_handling.md end-to-end before Phase 3.
Adds _show_menus_hwnd_result(app) -> Result[int] helper that wraps the
ctypes PyCapsule_GetPointer try/except from App._show_menus. The data
field carries the resolved hwnd (or 0 on failure) so the legacy wrapper
can pass it to subsequent win32gui calls without an additional app.hwnd
instance attribute.
App._show_menus becomes a thin wrapper that drains errors to
_last_request_errors when the hwnd capsule resolution fails.
Audit: BROAD_CATCH count 21 -> 20, COMPLIANT count 16 -> 17. Tests: 2/2 pass.
TIER-2 READ conductor/code_styleguides/error_handling.md end-to-end before Phase 3.
Adds _render_main_interface_result(app) -> Result[bool] helper that wraps
the OUTER render-loop try/except from App._gui_func. App._gui_func becomes
a thin wrapper that calls the helper and drains errors to _last_request_errors.
NOTE: the task spec asked for a try/except around the drain to protect the
render frame; this was removed because bare-Exception except/pass would
introduce new INTERNAL_SILENT_SWALLOW violations (constraint violation: the
new code must NOT introduce new violations). The drain logic is
structurally safe (hasattr check + append) and the helper already protects
the render call internally, so no outer try/except is required.
Audit: BROAD_CATCH count 23 -> 22, COMPLIANT count 14 -> 15. Tests: 2/2 pass.
TIER-2 READ conductor/code_styleguides/error_handling.md end-to-end before Phase 3.
Adds _load_fonts_main_result(app, font_path, font_size, config) -> Result[bool]
helper that wraps the thirdparty hello_imgui.load_font_ttf_with_font_awesome_icons
call. App._load_fonts becomes a thin wrapper that drains errors to
_startup_timeline_errors (startup-time error plane).
Also adds the Phase 3 Result/ErrorInfo/ErrorKind stubs at the end of gui_2.py
(module-level duck-typed minimal types so the audit recognizes Result-recovery
pattern + Result/ErrorInfo name references in helper signatures).
Audit: BROAD_CATCH count 25 -> 24, COMPLIANT count 12 -> 13. Tests: 2/2 pass.
Per AI Agent Checklist Rule #0 (re-read per phase).
Phase 3 focuses on the 8 INTERNAL_BROAD_CATCH sites inside render-loop
functions called every frame. The key constraint (per Batch A pattern
in the plan):
- For render-loop sites: the legacy wrapper returns early on error to
avoid breaking the immediate-mode frame.
- The _result helper returns Result[bool] with ErrorInfo on failure.
- The drain target is app._last_request_errors (the per-request
accumulator added by sub-track 3 Phase 6).
Per the styleguide (lines 396-407), Pattern 2 (GUI error display) is the
canonical drain for render-loop errors: imgui.open_popup in the same
frame, non-blocking, no crash. The render loop MUST NOT break even
if the underlying call raises.
Sites to migrate in Phase 3 (8 sites from PHASE1_SITE_INVENTORY.md):
- L731, L742 (_load_fonts): font loading via third-party SDK
- L1123 (_gui_func -> render_main_interface): main render loop
- L1172, L1198, L1223 (_show_menus): win32gui calls in menu bar
- L1285 (_handle_history_logic): history logic called every frame
- L4849 (render_warmup_status_indicator): status indicator render
Each site gets its own _result helper + legacy wrapper; one atomic
commit per site.
TIER-2 READ conductor/code_styleguides/error_handling.md end-to-end before Phase 1.
Adds tests/test_gui_2_result.py with 2 Phase 1 invariant tests:
1. test_phase_1_inventory_has_42_rows: parses
tests/artifacts/PHASE1_SITE_INVENTORY.md and asserts the Site
Inventory table contains exactly 42 rows.
2. test_phase_1_audit_has_42_migration_target_sites: runs
scripts/audit_exception_handling.py --src src --json, finds the
src/gui_2.py file record, counts sites in the migration-target
category set (excludes INTERNAL_COMPLIANT, INTERNAL_PROGRAMMER_RAISE,
BOUNDARY_FASTAPI, BOUNDARY_SDK, BOUNDARY_CONVERSION), and asserts the
count is 42.
This locks the 42-site migration target count: if the audit heuristic
or inventory drift, the test catches it before Phase 2.
Both tests pass:
tests/test_gui_2_result.py::test_phase_1_inventory_has_42_rows PASSED
tests/test_gui_2_result.py::test_phase_1_audit_has_42_migration_target_sites PASSED
TIER-2 READ conductor/code_styleguides/error_handling.md end-to-end before Phase 1.
Captures:
- tests/artifacts/PHASE1_AUDIT.json: full audit output for src/ (77KB)
- gui_2.py has 54 sites: 25 INTERNAL_BROAD_CATCH + 13 INTERNAL_SILENT_SWALLOW
+ 2 INTERNAL_RETHROW + 2 UNCLEAR + 12 INTERNAL_COMPLIANT
- tests/artifacts/PHASE1_SITE_INVENTORY.md: 42-row site inventory with
phase assignment, migration target, and rationale per site
Phase distribution: Phase 3 (8) + Phase 4 (3) + Phase 5 (13) + Phase 7 (1)
+ Phase 8 (4) + Phase 9 (1) + Phase 10 (8) + Phase 11 (2) + Phase 12 (2) = 39
sites (3 of the 13 INTERNAL_SILENT_SWALLOW sites were reclassified to other
phases because they are in render-loop or worker contexts where the drain
target is the render-result helper, not the silent-swallow migration).
Notes on classification:
- L65, L69 (UNCLEAR, _LazyModule._resolve): legitimate lazy-loading fallback
pattern with _FiledialogStub sentinel. Likely reclassifiable as
INTERNAL_COMPLIANT in Phase 12.
- L757, L760 (RETHROW, __getattr__): bare raise AttributeError(name) in the
canonical Python dunder method. Audit heuristic misclassifies as
INTERNAL_RETHROW; should be INTERNAL_PROGRAMMER_RAISE. Documented in
Phase 11.
Acknowledged the styleguide re-read per the AI Agent Checklist Rule #0.
Key points internalized for sub-track 4 (gui_2.py migration):
1. The 5 drain point patterns (error_handling.md:356-516):
- Pattern 1: HTTP error response (FastAPI)
- Pattern 2: GUI error display (imgui.open_popup) - PRIME for gui_2.py
- Pattern 3: Intentional app termination (sys.exit)
- Pattern 4: Telemetry emission
- Pattern 5: Bounded retry
2. INTERNAL_SILENT_SWALLOW (lines 462-540): logging is NOT a drain.
Per the user's principle (2026-06-17), narrow+log bodies in the
13 SILENT_SWALLOW sites in gui_2.py MUST be migrated to full
Result[T] propagation, NOT narrowed.
3. INTERNAL_BROAD_CATCH (lines 520-583): non-*_result code with
except Exception must be converted to a _result helper that
returns Result[T] with errors=[ErrorInfo(...)].
4. INTERNAL_RETHROW (lines 625-693): 3 legitimate patterns:
- Pattern 1: catch + convert + raise as different type
- Pattern 2: catch + log + re-raise
- Pattern 3: catch + cleanup + re-raise
5. AI Agent Checklist 5 MUST-DO + 7 MUST-NOT-DO rules
internalized; --strict gate (audit_exception_handling.py
--strict) is the CI enforcement.
TIER-2 READ conductor/code_styleguides/error_handling.md end-to-end before Phase 0.
Updates the sub-track 4 row from 'ready to start' to 'active 2026-06-19'.
Anti-sliming protocol (13 phases, per-site audit, per-phase invariant test)
is in effect for the migration of 42 sites in src/gui_2.py.
Sub-track 4 of the 5-sub-track result_migration_20260616 umbrella.
Migrates src/gui_2.py (the largest source file at 260KB / 7282 lines;
the immediate-mode ImGui rendering layer) to the data-oriented
Result[T] convention.
Scope: 42 migration-target sites (38 V + 2 S + 2 UNCLEAR) + 6 infra
sites for the drain plane. Per the user's directive (2026-06-19),
the phase structure is EXTRA LONG (13 phases instead of the umbrella's
1-2) to give Tier 2 well-defined narrow scope per phase. No phase has
more than 10 migration sites. This is the anti-sliming protocol:
previous sub-tracks slimed when scope felt tight (sub-track 2 Phase 10
slimed 21/26 sites via 5 laundering heuristics; sub-track 3 Phase 3
slimed 8 sites via logging.debug bodies). The 13-phase structure with
per-phase audit gates prevents sliming.
The 13 phases:
0. Setup + styleguide re-read (Tier 2 reads error_handling.md)
1. Site inventory + classification (42 sites in PHASE1_SITE_INVENTORY.md)
2. Drain plane wiring (3 new render functions: render_controller_error_modal,
_render_worker_error_indicator, _render_last_request_errors_modal)
3. INTERNAL_BROAD_CATCH Batch A (render-loop, <=10 sites)
4. INTERNAL_BROAD_CATCH Batch B (modal/dialog, <=10 sites)
5. INTERNAL_BROAD_CATCH Batch C (event handlers, <=10 sites)
6. Signal handler sites (<=5 sites; Pattern 3 drain: sys.exit)
7. Worker/background sites (<=5 sites; thread-safety via app._worker_errors_lock)
8. Property setter/state sites (<=5 sites)
9. Helper/utility sites (<=5 sites)
10. INTERNAL_SILENT_SWALLOW (<=13 sites; CRITICAL anti-sliming phase;
per user principle 'logging is NOT a drain')
11. INTERNAL_RETHROW classification (<=2 sites; Pattern 1/2/3)
12. UNCLEAR classification (<=2 sites)
13. Audit gate + end-of-track report (--strict exits 0; 11/11 tiers PASS)
Anti-sliming protocol per phase:
- Styleguide re-read at start of each phase (commit msg acknowledgment)
- Per-site audit pre/post check (capture before + after in commit body)
- Per-phase invariant test (test_phase_N_invariant_count_dropped)
- Per-file atomic commits (1 site = 1 commit)
- 'If a site resists migration: DO NOT invent a heuristic. Report.'
The data plane (8 controller state attributes added by sub-track 3
Phase 6: _last_request_errors, _worker_errors + lock,
_startup_timeline_errors, _signal_handler_error, _inject_preview_error,
_mcp_config_parse_error, _save_project_error, _model_fetch_errors) is
the source of truth. Sub-track 4 adds the drain plane (3 new render
functions in Phase 2) and migrates the 42 sites to feed their errors
into the data plane.
Files:
- spec.md (323 lines, 11 sections)
- plan.md (938 lines, 13 phases, 60+ atomic commits, anti-sliming protocol)
- metadata.json (14 VCs, 8 risks, scope)
- state.toml (14 phases, 102 tasks, 22 verification entries)
- tracks.md (new row 6d-4 in Active Tracks table)
Total: 5 files, 1327 lines added (excluding tracks.md).
Next: Tier 2 picks up Phase 0 (setup + styleguide re-read).
10 phases, 29 tasks, all worker-ready (WHERE / WHAT / HOW / SAFETY /
COMMIT / GIT NOTE per task):
Phase 1: Data extraction audit + draft helper script (FR5; TDD)
Phase 2: Generate conductor/chronology.md.draft
Phase 3: Prune [x]/[shipped] entries from conductor/tracks.md (FR2)
Phase 4: Add 3-step archiving convention to conductor/workflow.md (FR3)
Phase 5: Write docs/reports/CHRONOLOGY_MIGRATION_20260619.md (FR4)
Phase 6: User review of draft (GATE)
Phase 7: Promote draft to canonical chronology.md
Phase 8: Per-row cross-check (FR6 HARD GATE; 9 batches of ~20 rows)
Phase 9: Completeness check (FR6 HARD GATE; folder set vs row set)
Phase 10: User sign-off + end-of-track report (FR6 HARD GATE)
The cross-check (Phase 8) is the dominant cost. Per the user directive
2026-06-19, EVERY SINGLE ENTRY must be cross-checked. The plan batches
the work into 9 commits for review ergonomics; no batch is 'sample-based'
or 'looks right' -- each row's 5 fields (date, ID, status, summary,
range) are verified independently per FR6.
All 12 VCs from the spec are addressed in the plan's 'Verification
Criteria Recap' section.