diet-guard/diet_guard/tests/test_cli_sync.py

44 lines
1.4 KiB
Python
Raw Normal View History

Add cross-device log sync (Python half of Milestone 3) Pulls every other device's pushed log from GitHub-backed dumb storage, merges it with the local log, and pushes this device's own merged copy back -- the PC half of the diet-guard-app sync plan. - _sync_merge.py: pure union-by-id merge, tombstone always wins, legacy (time, desc) dedup for pre-id entries. Commutative and idempotent. - _sync_github.py: minimal GitHub Contents API client (list/get/put), distinguishing a 404 on an unused path from the repo itself being unreachable. - _sync.py: orchestration -- pull, merge, re-sign every persisted entry regardless of origin, write, rebuild the food bank, push. Re-signing unconditionally is load-bearing: an unsigned phone-origin entry would otherwise be silently dropped on the very next read once a machine holds the shared HMAC key. - _foodbank.rebuild_food_bank(): the "replay a full log into a fresh bank" entrypoint the Python side was missing (the Dart port already had its equivalent). Backs sync's bank-rebuild step. - New diet-guard-sync.service/.timer (15-minute cadence, headless, a separate unit from the gate so a held lock can't stall sync) and a new install.sh step to install them. - Created the private kuhyx/diet-guard-sync GitHub repo for storage. Incidental to this feature: adding the `sync` subcommand pushed _cli.py past the repo's 500-line cap, so `gate`'s CLI glue moved out alongside sync's into _cli_gate.py/_cli_sync.py -- same split pattern already used for the gate window logic itself, not a sync-specific design choice. 338 tests, 100% branch coverage. Verified importing and running cleanly under /usr/bin/python (the production interpreter), not just the dev venv -- the gap that caused the earlier 3-day outage. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> Claude-Session: https://claude.ai/code/session_01FU3f5KQ1GHXsbbSecfVEyF
2026-06-22 19:36:27 +02:00
"""Tests for the sync subcommand's handler, split out of test_cli.py
alongside its source module (see _cli_sync.py's module docstring).
"""
from __future__ import annotations
from unittest.mock import patch
from diet_guard import _cli_sync
from diet_guard._sync import SyncError
from diet_guard._sync_github import GitHubSyncError
class TestCmdSync:
def test_reports_synced_entry_count(self) -> None:
merged = {
"2026-06-22": [{"id": "a"}, {"id": "b"}],
"2026-06-21": [{"id": "c"}],
}
lines: list[str] = []
with patch.object(_cli_sync, "run_sync", return_value=merged):
assert _cli_sync.cmd_sync(lines.append) == 0
assert lines == ["synced: 3 entries across 2 day(s)."]
def test_reports_sync_error_as_not_configured(self) -> None:
lines: list[str] = []
with patch.object(
_cli_sync,
"run_sync",
side_effect=SyncError("no token"),
):
assert _cli_sync.cmd_sync(lines.append) == 1
assert lines == ["sync not configured: no token"]
def test_reports_github_sync_error_as_failed(self) -> None:
lines: list[str] = []
with patch.object(
_cli_sync,
"run_sync",
side_effect=GitHubSyncError("network down"),
):
assert _cli_sync.cmd_sync(lines.append) == 1
assert lines == ["sync failed: network down"]