Files
gps-denied-onboard/tests/test_recovery.py
T
Yuzviak 094895b21b feat(phases 2-7): implement full GPS-denied navigation pipeline
Phase 2 — Visual Odometry:
  - ORBVisualOdometry (dev/CI), CuVSLAMVisualOdometry (Jetson)
  - TRTInferenceEngine (TensorRT FP16, conditional import)
  - create_vo_backend() factory

Phase 3 — Satellite Matching + GPR:
  - SatelliteDataManager: local z/x/y tiles, ESKF ±3σ tile selection
  - GSD normalization (SAT-03), RANSAC inlier-ratio confidence (SAT-04)
  - GlobalPlaceRecognition: Faiss index + numpy fallback

Phase 4 — MAVLink I/O:
  - MAVLinkBridge: GPS_INPUT 15+ fields, IMU callback, 1Hz telemetry
  - 3-consecutive-failure reloc request
  - MockMAVConnection for CI

Phase 5 — Pipeline Wiring:
  - ESKF wired into process_frame: VO update → satellite update
  - CoordinateTransformer + SatelliteDataManager via DI
  - MAVLink state push per frame (PIPE-07)
  - Real pixel_to_gps via ray-ground projection (PIPE-06)
  - GTSAM ISAM2 update when available (PIPE-03)

Phase 6 — Docker + CI:
  - Multi-stage Dockerfile (python:3.11-slim)
  - docker-compose.yml (dev), docker-compose.sitl.yml (ArduPilot SITL)
  - GitHub Actions: ci.yml (lint+pytest+docker smoke), sitl.yml (nightly)
  - tests/test_sitl_integration.py (8 tests, skip without SITL)

Phase 7 — Accuracy Validation:
  - AccuracyBenchmark + SyntheticTrajectory
  - AC-PERF-1: 80% within 50m 
  - AC-PERF-2: 60% within 20m 
  - AC-PERF-3: p95 latency < 400ms 
  - AC-PERF-4: VO drift 1km < 100m  (actual ~11m)
  - scripts/benchmark_accuracy.py CLI

Tests: 195 passed / 8 skipped

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-02 17:00:41 +03:00

63 lines
2.2 KiB
Python

"""Tests for Failure Recovery Coordinator (F11)."""
import pytest
import numpy as np
from gps_denied.core.recovery import FailureRecoveryCoordinator
from gps_denied.core.chunk_manager import RouteChunkManager
from gps_denied.core.graph import FactorGraphOptimizer
from gps_denied.core.gpr import GlobalPlaceRecognition
from gps_denied.core.metric import MetricRefinement
from gps_denied.core.models import ModelManager
from gps_denied.schemas.graph import FactorGraphConfig
@pytest.fixture
def recovery():
opt = FactorGraphOptimizer(FactorGraphConfig())
cm = RouteChunkManager(opt)
mm = ModelManager()
gpr = GlobalPlaceRecognition(mm)
gpr.load_index("test", "test")
metric = MetricRefinement(mm)
return FailureRecoveryCoordinator(cm, gpr, metric)
def test_handle_tracking_lost(recovery):
flight = "recovery_fl"
res = recovery.handle_tracking_lost(flight, 404)
assert res is True
# active chunk should be 404
active = recovery.chunk_manager.get_active_chunk(flight)
assert active.start_frame_id == 404
def test_process_chunk_recovery_success(recovery, monkeypatch):
# Mock LitSAM to guarantee match
def mock_align(*args, **kwargs):
from gps_denied.schemas.metric import ChunkAlignmentResult, Sim3Transform
from gps_denied.schemas import GPSPoint
return ChunkAlignmentResult(
matched=True, chunk_id="x", chunk_center_gps=GPSPoint(lat=49, lon=30),
rotation_angle=0, confidence=0.9, inlier_count=50,
transform=Sim3Transform(translation=np.zeros(3), rotation=np.eye(3), scale=1),
reprojection_error=1.0
)
monkeypatch.setattr(recovery.metric_refinement, "align_chunk_to_satellite", mock_align)
flight = "rec2"
recovery.handle_tracking_lost(flight, 10)
chunk_id = recovery.chunk_manager.get_active_chunk(flight).chunk_id
images = [np.zeros((256, 256, 3)) for _ in range(3)]
res = recovery.process_chunk_recovery(flight, chunk_id, images)
assert res is True
from gps_denied.schemas.chunk import ChunkStatus
active = recovery.chunk_manager.get_active_chunk(flight)
assert active.has_anchor is True
assert active.matching_status == ChunkStatus.ANCHORED