mirror of
https://github.com/vicliu624/trail-mate.git
synced 2026-06-25 06:31:42 +00:00
71f10ae6d0
* fix(tdeck): improve display startup and brightness handling * fix(energy-sweep): use instant RSSI and only lock LoRa while scanning * feat(cardputer-zero): add linux shells and M5 SDK baseline * feat(cardputer-zero): add linux runtime baseline and shell ui simulator * feat(cardputer-zero): unify linux shell boot and polish simulator * docs(cardputer-zero): define final-shape adaptation spec * feat(cardputer-zero): integrate shared linux runtimes and pages * fix(linux-sim): mount repo root in dev container * Fix GPS runtime semantics and transport init Add a GPS specification and align platform runtimes around explicit GPS enable, power, receiver configuration, and external NMEA export semantics. Keep internal NMEA parsing independent from external export settings, stop treating gps_mode as an enable flag, and update phone/UI config paths to use gps_enabled. Decouple board-level GPS transport readiness from UBX receiver probing on T-Deck, T-Deck Pro, and T-LoRa Pager, and let boards own UART teardown. Verified with pio run -e tdeck, pio run -e tlora_pager_sx1262, pio run -e gat562_mesh_evb_pro, and pio run -e tdeck_pro_a7682e. * Add Russian localization pack Add an installable European Cyrillic Extended locale bundle with Russian translations, Cyrillic font metadata, and package catalog entry. Credit polarikus for the Russian translations based on the polarikus/trail-mate localization PR. * Prepare 0.1.23-alpha release * Fix T-Watch Morse release build * Format CI-checked sources * Fix Cardputer Linux CI dependencies * Fix WSL validation smoke target build * Prepare 0.1.24-alpha release * fix: unblock Cardputer Zero Linux CI
163 lines
4.7 KiB
Python
163 lines
4.7 KiB
Python
#!/usr/bin/env python3
|
|
"""Guard shared/platform UI boundaries from regressing."""
|
|
|
|
from __future__ import annotations
|
|
|
|
from dataclasses import dataclass
|
|
import os
|
|
from pathlib import Path
|
|
import re
|
|
import sys
|
|
|
|
|
|
REPO_ROOT = Path(__file__).resolve().parent.parent
|
|
|
|
|
|
@dataclass(frozen=True)
|
|
class Rule:
|
|
name: str
|
|
roots: tuple[str, ...]
|
|
includes: tuple[re.Pattern[str], ...]
|
|
excludes: tuple[str, ...] = ()
|
|
|
|
|
|
@dataclass(frozen=True)
|
|
class Violation:
|
|
path: Path
|
|
line_number: int
|
|
line: str
|
|
rule_name: str
|
|
|
|
|
|
SOURCE_SUFFIXES = {".c", ".cc", ".cpp", ".cxx", ".h", ".hpp", ".hh"}
|
|
EXCLUDED_DIR_NAMES = {
|
|
".git",
|
|
".pio",
|
|
".pytest_cache",
|
|
".tmp",
|
|
".venv",
|
|
"__pycache__",
|
|
"build",
|
|
"dist",
|
|
}
|
|
|
|
|
|
def compile_patterns(patterns: tuple[str, ...]) -> tuple[re.Pattern[str], ...]:
|
|
return tuple(re.compile(pattern) for pattern in patterns)
|
|
|
|
|
|
RULES = (
|
|
Rule(
|
|
name="legacy-platform-ui-shim-include",
|
|
roots=(".",),
|
|
includes=compile_patterns(
|
|
(
|
|
r'#include\s*[<"]ui/runtime/pack_repository\.h[>"]',
|
|
r'#include\s*[<"]ui/screens/team/team_ui_store\.h[>"]',
|
|
)
|
|
),
|
|
excludes=(
|
|
"modules/ui_shared/src/ui/runtime/pack_repository.cpp",
|
|
"modules/ui_shared/src/ui/screens/team/team_ui_store.cpp",
|
|
),
|
|
),
|
|
Rule(
|
|
name="ui_shared-esp-only-include",
|
|
roots=("modules/ui_shared",),
|
|
includes=compile_patterns(
|
|
(
|
|
r'#include\s*[<"]Arduino\.h[>"]',
|
|
r'#include\s*[<"]Preferences[>"]',
|
|
r'#include\s*[<"]SD\.h[>"]',
|
|
r'#include\s*[<"]FS\.h[>"]',
|
|
r'#include\s*[<"]platform/esp/[^>"]+[>"]',
|
|
)
|
|
),
|
|
),
|
|
Rule(
|
|
name="core-modules-platform-tail",
|
|
roots=(
|
|
"modules/core_chat",
|
|
"modules/core_gps",
|
|
"modules/core_hostlink",
|
|
"modules/core_sys",
|
|
"modules/core_team",
|
|
),
|
|
includes=compile_patterns(
|
|
(
|
|
r'#include\s*[<"]Arduino\.h[>"]',
|
|
r'#include\s*[<"]Preferences[>"]',
|
|
r'#include\s*[<"]SD\.h[>"]',
|
|
r'#include\s*[<"]FS\.h[>"]',
|
|
r'#include\s*[<"]freertos/[^>"]+[>"]',
|
|
r'#include\s*[<"]rtos\.h[>"]',
|
|
r'#include\s*[<"]platform/esp/[^>"]+[>"]',
|
|
)
|
|
),
|
|
excludes=("modules/core_chat/generated",),
|
|
),
|
|
)
|
|
|
|
|
|
def iter_source_files(root: Path):
|
|
for current_root, dir_names, file_names in os.walk(root):
|
|
dir_names[:] = [name for name in dir_names if name not in EXCLUDED_DIR_NAMES]
|
|
current_root_path = Path(current_root)
|
|
for file_name in file_names:
|
|
path = current_root_path / file_name
|
|
if path.suffix.lower() in SOURCE_SUFFIXES:
|
|
yield path
|
|
|
|
|
|
def is_excluded(path: Path, excludes: tuple[str, ...]) -> bool:
|
|
relative = path.relative_to(REPO_ROOT).as_posix()
|
|
return any(relative == prefix or relative.startswith(prefix.rstrip("/") + "/") for prefix in excludes)
|
|
|
|
|
|
def collect_violations() -> list[Violation]:
|
|
violations: list[Violation] = []
|
|
seen_files: set[Path] = set()
|
|
|
|
for rule in RULES:
|
|
for root_name in rule.roots:
|
|
root_path = REPO_ROOT / root_name
|
|
if not root_path.exists():
|
|
continue
|
|
for path in iter_source_files(root_path):
|
|
if is_excluded(path, rule.excludes):
|
|
continue
|
|
# Avoid rescanning repo root duplicates for the same rule set.
|
|
key = (rule.name, path)
|
|
if key in seen_files:
|
|
continue
|
|
seen_files.add(key)
|
|
for line_number, line in enumerate(path.read_text(encoding="utf-8", errors="ignore").splitlines(), start=1):
|
|
if any(pattern.search(line) for pattern in rule.includes):
|
|
violations.append(
|
|
Violation(
|
|
path=path,
|
|
line_number=line_number,
|
|
line=line.strip(),
|
|
rule_name=rule.name,
|
|
)
|
|
)
|
|
return violations
|
|
|
|
|
|
def main() -> int:
|
|
violations = collect_violations()
|
|
if not violations:
|
|
print("Boundary check passed.")
|
|
return 0
|
|
|
|
print("Boundary check failed:")
|
|
for violation in violations:
|
|
relative = violation.path.relative_to(REPO_ROOT).as_posix()
|
|
print(f"- [{violation.rule_name}] {relative}:{violation.line_number}")
|
|
print(f" {violation.line}")
|
|
return 1
|
|
|
|
|
|
if __name__ == "__main__":
|
|
sys.exit(main())
|