mirror of
https://github.com/kuhyx/diet-guard.git
synced 2026-07-04 15:23:16 +02:00
Rewrites python_pkg.diet_guard imports to diet_guard, vendors the shared as_float coercion helper, drops the monorepo PYTHONPATH from install.sh and the systemd unit (package is now pip-installed), and scaffolds standalone lint/test config matching testsAndMisc's real enforced bar (pylint --fail-under=10 with tests excluded and the use-implicit-booleaness/consider-using-with disables, mypy's actual disabled-error-code set, ruff ALL, bandit, 100% branch coverage).
112 lines
3.5 KiB
Python
112 lines
3.5 KiB
Python
"""Pure meal-slot arithmetic for the diet_guard gate.
|
|
|
|
This module is deliberately I/O-free and clock-free: every function is a total
|
|
function of its ``now`` argument and the configured slot constants, so the
|
|
fiddly time-of-day edges (07:59 vs 08:00, the 20:00->22:00 tail, the midnight
|
|
reset) are exhaustively unit-testable without mocking the filesystem or the
|
|
wall clock. The stateful "which slots have I actually logged?" question lives
|
|
in :mod:`diet_guard._state`; the two are composed in
|
|
:mod:`diet_guard._gate`.
|
|
|
|
A "slot" is simply the integer hour at which a meal checkpoint opens (08, 12,
|
|
16, 20). A slot is *elapsed* once its hour has arrived and we are still inside
|
|
the daily enforcement window; an elapsed slot with no logged meal is what makes
|
|
the gate fire.
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
from typing import TYPE_CHECKING
|
|
|
|
from diet_guard._constants import (
|
|
GATE_DAY_START_HOUR,
|
|
GATE_EATING_END_HOUR,
|
|
GATE_SLOT_INTERVAL_HOURS,
|
|
)
|
|
|
|
if TYPE_CHECKING:
|
|
from datetime import datetime
|
|
|
|
_HOURS_PER_DAY = 24
|
|
|
|
|
|
def day_slots() -> tuple[int, ...]:
|
|
"""Return the fixed meal-slot hours for a day, e.g. ``(8, 12, 16, 20)``.
|
|
|
|
Slots run from the day-start hour, every interval, up to (but not past) the
|
|
overnight cutoff. Derived from the constants so changing the cadence in one
|
|
place reshapes the whole schedule.
|
|
|
|
Returns:
|
|
The slot hours in ascending order.
|
|
"""
|
|
return tuple(
|
|
range(GATE_DAY_START_HOUR, GATE_EATING_END_HOUR, GATE_SLOT_INTERVAL_HOURS)
|
|
)
|
|
|
|
|
|
def within_enforcement_window(now: datetime) -> bool:
|
|
"""Return True if ``now`` is inside the daily slot-enforcement window.
|
|
|
|
Outside ``[day_start, eating_end)`` the gate never fires, so unlogged slots
|
|
lapse overnight instead of trapping you at 03:00.
|
|
|
|
Args:
|
|
now: Reference local time.
|
|
|
|
Returns:
|
|
True if slot enforcement is active at ``now``.
|
|
"""
|
|
return GATE_DAY_START_HOUR <= now.hour < GATE_EATING_END_HOUR
|
|
|
|
|
|
def elapsed_slots(now: datetime) -> tuple[int, ...]:
|
|
"""Return today's slots whose hour has arrived as of ``now``.
|
|
|
|
Empty outside the enforcement window (before the first slot, or after the
|
|
overnight cutoff), so the caller never has to special-case the night.
|
|
|
|
Args:
|
|
now: Reference local time.
|
|
|
|
Returns:
|
|
The elapsed slot hours, ascending (possibly empty).
|
|
"""
|
|
if not within_enforcement_window(now):
|
|
return ()
|
|
return tuple(slot for slot in day_slots() if slot <= now.hour)
|
|
|
|
|
|
def missing_slots(now: datetime, logged: set[int]) -> tuple[int, ...]:
|
|
"""Return elapsed slots that have not been satisfied by a logged meal.
|
|
|
|
Args:
|
|
now: Reference local time.
|
|
logged: The set of slot hours already covered by today's log.
|
|
|
|
Returns:
|
|
The unsatisfied elapsed slot hours, ascending (empty == nothing due).
|
|
"""
|
|
return tuple(slot for slot in elapsed_slots(now) if slot not in logged)
|
|
|
|
|
|
def current_slot(now: datetime) -> int | None:
|
|
"""Return the most recent elapsed slot as of ``now``, or None.
|
|
|
|
Used to tag a meal logged through the plain ``ate`` CLI with the slot it
|
|
belongs to, so it counts toward that checkpoint.
|
|
|
|
Args:
|
|
now: Reference local time.
|
|
|
|
Returns:
|
|
The latest elapsed slot hour, or None when none have elapsed yet.
|
|
"""
|
|
elapsed = elapsed_slots(now)
|
|
return elapsed[-1] if elapsed else None
|
|
|
|
|
|
def slot_label(slot: int) -> str:
|
|
"""Return a human ``HH:00`` label for a slot hour, e.g. ``"08:00"``."""
|
|
return f"{slot % _HOURS_PER_DAY:02d}:00"
|