"""CLI handler for the ``gate`` subcommand. Split out from :mod:`diet_guard._cli` to keep that module under the repo's 500-line cap (see ``CLAUDE.md``'s "feat: split oversized modules" history). The gate's actual window logic already lives in ``_gatelock*.py``; this is just the thin CLI glue, same as ``_cli_sync.py`` is for ``sync``. """ from __future__ import annotations from typing import TYPE_CHECKING from diet_guard._gate import gate_is_due from diet_guard._gatelock import MealGate, acquire_gate_lock, release_gate_lock from diet_guard._gatelock_support import wait_for_display if TYPE_CHECKING: from collections.abc import Callable def cmd_gate(emit: Callable[[str], None], *, check: bool, demo: bool) -> int: """Run the log-to-unlock gate. Three modes: ``--check`` is a headless decision (no window) whose exit code a timer reads; ``--demo`` always shows a safe demo window; bare ``gate`` shows the real lock only when one is due. A flock guard stops a second window from stacking on top of the first, and a window-opening mode first waits for the X display so a session-start launch never crashes unshown. Args: emit: A one-line output sink (``_cli._emit``, passed in rather than imported -- see ``_cli_sync.cmd_sync`` for why). check: Headless mode -- print and return an exit code, open no window. demo: Use safe demo mode (local grab + close button) for the window. Returns: For ``--check``: 0 if not due, 1 if a lock is due. Otherwise 0. """ if check: due = gate_is_due() emit("due (a lock is warranted)" if due else "ok (no lock needed)") return 1 if due else 0 if not demo and not gate_is_due(): emit("ok - no lock needed right now.") return 0 handle = acquire_gate_lock() if handle is None: emit("the gate is already running.") return 0 try: # At session start the timer can fire before the X display/auth cookie # is ready; wait it out so the window opens instead of crashing on a # "couldn't connect to display" TclError (see _gatelock.wait_for_display). if not wait_for_display(): emit("display not ready yet; will retry on the next timer tick.") return 0 MealGate(demo_mode=demo).run() finally: release_gate_lock(handle) return 0