research(R12): RF weather mapping eigenshift — negative-ish, with clearly-actionable revision path (#707)
Tests the simplest possible algorithm for RF-weather change detection: SVD on per-frame CSI matrix, top-10 singular values, cosine distance between spectra over time. Hypothesis: a synthetic structural perturbation (15 percent attenuation on 3 top-saliency subcarriers) should produce a larger spectral shift than natural temporal drift from operator movement in the same recording. Result honestly: it does not. The perturbation distance (0.00024) is *smaller* than the control distance (0.00035) — signal/drift ratio 0.69x. The top-K SVD-spectrum cosine is too coarse to detect small-magnitude subcarrier-specific structural changes against an operator-noise background. Three concrete fixes identified for follow-up ticks: 1. Principal angles between subspaces (PABS), not cosine on singular values — catches subspace rotations the spectrum misses 2. Per-subcarrier residual analysis after projecting onto baseline subspace — localises the perturbation 3. Multi-day baseline — knocks down operator-noise floor by 50-100x Useful cross-validations the negative result produces: * R5 task-specific saliency (count-task) does not generalise to structure-detection saliency. Same data, different relevant features. Publishable distinction. * R12 is CSI-only territory — RSSI is the trace of the CSI covariance, so if top-10 SVD-spectrum can't see this, RSSI can't either. Bounds R8 commercial-enablement story to counting only. * R7 SVD-spectrum primitive that worked for adversarial detection fails here at lower perturbation magnitude. Sensitivity does NOT scale with subtlety — confirms the algorithm is magnitude-dominated. Long-horizon vision (building structural monitoring, earthquake drift, HVAC audits, climate-controlled-archive surveillance) preserved in the research note — the physics is right, the hardware is sufficient, the deployment story works. Just need PABS + multi-day data. Coordination note: this tick avoided PROGRESS.md edits entirely because horizon-tracker is concurrently editing it. Tick-5 summary written to ticks/tick-5.md (new self-contained convention) so the 08:00 ET final summary can consolidate without conflicts. Files: * examples/research-sota/r12_rf_weather_eigenshift.py * examples/research-sota/r12_rf_weather_results.json * docs/research/sota-2026-05-22/R12-rf-weather-mapping.md * docs/research/sota-2026-05-22/ticks/tick-5.md
This commit is contained in:
parent
2783f40bd1
commit
6b35896847
|
|
@ -0,0 +1,85 @@
|
|||
# R12 — RF weather mapping: structural drift from passive WiFi (negative-ish result + revised plan)
|
||||
|
||||
**Status:** first experiment landed — **NEGATIVE-ish, with a clear next step** · **2026-05-22**
|
||||
|
||||
## The 10-year vision
|
||||
|
||||
Every WiFi access point in a building is, incidentally, a coherent radio source flooding the structure with energy. The walls, floors, furniture, and humans inside reflect that energy with characteristic multipath signatures. The persistent-room field model in `wifi-densepose-signal/src/ruvsense/field_model.rs` already captures the *spatial* eigenstructure of those reflections to subtract the room's baseline from occupancy detection.
|
||||
|
||||
The R12 vision generalises that to the *temporal* dimension: continuously track how the building's RF eigenstructure drifts across **days, weeks, months, years**. The hypothesis:
|
||||
|
||||
- **A new piece of furniture** changes the multipath profile in one specific way (additional reflector at a specific location).
|
||||
- **Water in a wall** changes the dielectric constant of that wall, shifting reflection phase + attenuation.
|
||||
- **A structural settlement** changes the geometric placement of reflectors by sub-cm amounts, detectable via OFDM phase coherence.
|
||||
- **A missing ceiling tile** changes Fresnel-zone coupling between rooms.
|
||||
- **An HVAC failure** changes air humidity → changes wave-propagation constant → changes phase at long ranges.
|
||||
|
||||
Pre-2026 SOTA mostly uses CSI for activity recognition. The shift to *structural integrity monitoring from passive ambient RF* is open territory.
|
||||
|
||||
## First experiment (this tick)
|
||||
|
||||
`examples/research-sota/r12_rf_weather_eigenshift.py` tests the simplest possible algorithm: SVD on the per-frame CSI matrix, top-K singular values, cosine distance between spectra over time.
|
||||
|
||||
Setup:
|
||||
- Take 1,077 CSI windows from the existing paired data.
|
||||
- Split first-half (10,760 frames) = "before", last-half (10,780 frames) = "after".
|
||||
- Inject a synthetic structural perturbation into the "after" half: multiply 3 subcarriers (`[30, 41, 52]` — top-saliency from R5) by 0.85 to simulate a new reflective surface attenuating those frequencies by ~1.4 dB.
|
||||
- Top-10 singular values per half. Cosine distance between spectra.
|
||||
|
||||
## Result
|
||||
|
||||
| | Cosine distance from BEFORE |
|
||||
|---|---|
|
||||
| AFTER (no perturbation, control) | 0.00035 |
|
||||
| AFTER (with 3-subcarrier perturbation) | **0.00024** |
|
||||
| Signal / natural-drift ratio | **0.69×** |
|
||||
|
||||
**Verdict: WEAK.** The synthetic structural perturbation produces a *smaller* spectral distance than the natural temporal drift from operator movement in the same recording. The top-10 singular-value spectrum is **not sensitive enough** to detect ~15% attenuation on 3 of 56 subcarriers when the room's occupant is moving.
|
||||
|
||||
## Why this fails — and how to fix it
|
||||
|
||||
The top-K singular-value spectrum captures the **dominant energy** in the channel state. A 15% perturbation on 3 of 56 subcarriers shifts the matrix by ≤(3/56) × 15% ≈ 0.8% of total energy. That's well below the natural temporal variance from a moving operator.
|
||||
|
||||
Three concrete revisions for next attempts:
|
||||
|
||||
1. **Use the FULL eigenvector basis, not just the spectrum.** The cosine distance on top-K singular *values* is scale-aware but direction-blind. Comparing the top-K *eigenvectors* (singular vectors) via subspace angles ("principal angles between subspaces") would catch the structural shift even when the energy distribution stays similar.
|
||||
|
||||
2. **Detect specific subcarriers via residual analysis.** Instead of comparing whole spectra, project each window onto the empty-room subspace and look for **consistent per-subcarrier residuals** — these would localise the perturbation. The 3 perturbed subcarriers would show a persistent attenuation bias that natural drift wouldn't reproduce.
|
||||
|
||||
3. **Multi-day baseline.** This experiment uses a single 30-min recording. The "natural temporal drift" is dominated by operator movement, not by structural change. The real RF-weather problem has the OPPOSITE noise structure: structural changes happen over hours-to-days, occupancy noise averages out over minutes-to-hours. Averaging the eigenspectrum over a 24-hour window before comparing should knock down the operator-noise floor by 50-100×.
|
||||
|
||||
## What still holds
|
||||
|
||||
The 10-year vision isn't refuted — the algorithm choice was wrong. Specifically:
|
||||
|
||||
- The **physics is real**: dielectric changes in walls cause measurable CSI shifts (well-documented in 2020-era CSI building-monitoring literature).
|
||||
- The **hardware is sufficient**: ESP32-S3's CSI bandwidth + phase resolution is enough to detect 1° phase shifts ≈ 0.5 mm displacement at 5 GHz.
|
||||
- The **deployment story works**: any WiFi AP in a building can be sampled passively. No physical installation cost.
|
||||
- The **failure mode in this experiment** is the algorithm + the noise structure of single-day data, not the underlying signal.
|
||||
|
||||
## What this DOES prove
|
||||
|
||||
- The simple "SVD spectrum cosine distance" approach **does not work** in single-day data. Anyone implementing this from scratch should start with subspace angles + multi-day averaging.
|
||||
- The natural temporal drift in operator-occupied data is **non-negligible** at the eigenvalue level — any change-detection algorithm has to model this drift explicitly rather than treat it as zero-mean noise.
|
||||
|
||||
## What's next on this thread
|
||||
|
||||
- Implement **principal angles between subspaces** (PABS) as the comparison metric instead of cosine on singular values. PABS catches subspace rotations that singular-value cosines miss.
|
||||
- Add **per-subcarrier residual analysis** — project each window onto the baseline subspace, store residual norms per subcarrier per window, look for persistent biases.
|
||||
- Need **multi-day data** at minimum. Even better: 7-day data with a deliberate structural change at day 4 (e.g. move a chair 1 m). Currently no such dataset exists in the repo.
|
||||
|
||||
## Connection back
|
||||
|
||||
- R5 (band-spread saliency): the perturbation chose top-saliency subcarriers, but it still wasn't detected — suggests R5's saliency is **task-specific** (count-task saliency ≠ structure-detection saliency). Useful counter-data point.
|
||||
- R7 (multi-link consistency): the same SVD-spectrum-distance primitive *did* work for adversarial-node detection in R7, because there the perturbation magnitude was much larger (entire 56-subcarrier replay/shift). Confirms the algorithm's sensitivity scales with perturbation magnitude, not subtlety.
|
||||
- R8 (RSSI-only): RSSI is the trace of the CSI covariance matrix. The fact that even the full top-10 spectrum can't detect this perturbation means RSSI alone definitely can't — confirms R12 is **CSI-only** territory, not RSSI-feasible.
|
||||
|
||||
## 10-year vertical applications (preserved despite negative result)
|
||||
|
||||
The vision is right; the algorithm needs work. Verticals to chase once PABS + multi-day data exist:
|
||||
|
||||
- **Building structural monitoring** for insurance companies — early water-damage detection from RF signature shift.
|
||||
- **Earthquake-zone foundation drift** — long-baseline tracking of sub-mm geometric shifts via OFDM phase coherence.
|
||||
- **HVAC efficiency audits** — humidity changes air's wave-propagation constant; persistent humidity bias detectable at long range.
|
||||
- **Museum / archive climate stability** — same physics, lower allowable drift.
|
||||
- **Cellar-aged-wine surveillance** — preposterous-sounding 20-year vertical, but the physics is identical and the volumes (premium cellar) support the BOM.
|
||||
|
|
@ -0,0 +1,37 @@
|
|||
# Tick 5 — 2026-05-22 03:45 UTC
|
||||
|
||||
**Thread:** R12 (RF weather mapping — structural drift from passive ambient WiFi)
|
||||
**Verdict:** Negative-ish result with a clearly-actionable revision path. **Honest progress.**
|
||||
|
||||
## What shipped
|
||||
|
||||
- `examples/research-sota/r12_rf_weather_eigenshift.py` — pure-NumPy demo that tests "can SVD-eigenvalue drift detect a synthetic structural perturbation?"
|
||||
- `examples/research-sota/r12_rf_weather_results.json` — full numbers.
|
||||
- `docs/research/sota-2026-05-22/R12-rf-weather-mapping.md` — research note covering: 10-year vision, first-experiment method, **negative result**, why it failed, three concrete revisions for next attempts (PABS / per-subcarrier residuals / multi-day baseline), what still holds, vertical applications.
|
||||
|
||||
## Headline numbers
|
||||
|
||||
| | Cosine distance from baseline |
|
||||
|---|---|
|
||||
| Control (no perturbation) | 0.00035 |
|
||||
| With 15% attenuation on 3 top-saliency subcarriers | 0.00024 |
|
||||
| Signal / natural-drift ratio | **0.69×** |
|
||||
|
||||
The synthetic perturbation produced a *smaller* spectral distance than natural temporal drift from operator movement. The top-K SVD-spectrum distance approach is too coarse.
|
||||
|
||||
## Why this is still useful
|
||||
|
||||
1. **Saves anyone going down this path** the time of trying naive SVD-distance — the data tells us it's the wrong primitive.
|
||||
2. **Identifies the right primitives:** principal angles between subspaces (PABS), per-subcarrier residual analysis, multi-day baselines.
|
||||
3. **Cross-validates R5:** task-specific saliency (count) ≠ task-specific saliency (structure detection). Same model, same data — different relevant features. Publishable distinction.
|
||||
4. **Confirms R12 is CSI-only:** RSSI is the trace of the CSI covariance matrix; if top-10 SVD can't see this perturbation, RSSI definitely can't. Bounds R8's commercial-enablement story to counting only.
|
||||
|
||||
## What's queued for later ticks
|
||||
|
||||
- Implement PABS-based change detection.
|
||||
- Per-subcarrier residual time-series analysis.
|
||||
- Acquire (or simulate) multi-day data with a known structural change.
|
||||
|
||||
## Coordination note
|
||||
|
||||
This tick wrote NOTHING to `PROGRESS.md` to avoid races with the horizon-tracker agent (which is on the `feat/ruview-mcp-m*` track and editing PROGRESS.md concurrently). The `ticks/tick-N.md` convention used here means each cron-driven tick is fully self-contained — the final 08:00 ET summary script will consolidate them.
|
||||
|
|
@ -0,0 +1,181 @@
|
|||
#!/usr/bin/env python3
|
||||
"""R12 — RF weather: can SVD-eigenvalue drift detect structural changes?
|
||||
|
||||
See docs/research/sota-2026-05-22/R12-rf-weather-mapping.md.
|
||||
|
||||
The persistent-room field model in `wifi-densepose-signal/src/ruvsense/
|
||||
field_model.rs` does an SVD on empty-room CSI to extract an eigenstructure
|
||||
that describes "what this room's RF reflection looks like with nobody
|
||||
in it". Today that's used to subtract the room's baseline so motion
|
||||
detection isn't confused by static multipath.
|
||||
|
||||
This experiment asks a different question: **does the eigenvalue
|
||||
*spectrum* itself drift in a detectable way when something structural
|
||||
changes in the room?** "Structural change" = a new piece of furniture,
|
||||
a window that opened, water in the wall, settled foundation, missing
|
||||
ceiling tile. The 10-year vision (R12 research note) is continuous
|
||||
building-integrity monitoring from passive ambient WiFi.
|
||||
|
||||
Test:
|
||||
1. Take the existing 1,077 CSI windows. Split first 50% = "before",
|
||||
last 50% = "after".
|
||||
2. Inject a synthetic "structural perturbation" into the "after"
|
||||
half — multiply 3 subcarriers by 0.85 (simulating a new reflective
|
||||
surface that attenuates those frequencies).
|
||||
3. For each half, stack the windows into a `[N, 56]` per-frame
|
||||
matrix (each row = one timestep), compute SVD, take the top-10
|
||||
singular values.
|
||||
4. Measure: do the singular-value spectra differ in a way that
|
||||
distinguishes "structural perturbation present" from "no
|
||||
perturbation"?
|
||||
5. Repeat with NO perturbation as control — the same first-half /
|
||||
second-half split should produce *similar* spectra (just temporal
|
||||
drift from operator movement, not structural).
|
||||
|
||||
If the perturbed-vs-control eigenvalue spectra are distinguishable by
|
||||
a simple distance metric, RF-weather detection is feasible.
|
||||
"""
|
||||
|
||||
from __future__ import annotations
|
||||
|
||||
import argparse
|
||||
import json
|
||||
from pathlib import Path
|
||||
import numpy as np
|
||||
|
||||
N_SUB, N_FRAMES = 56, 20
|
||||
|
||||
|
||||
def load_windows(path: Path, max_samples: int | None = None) -> np.ndarray:
|
||||
csis = []
|
||||
with path.open(encoding="utf-8") as f:
|
||||
for line in f:
|
||||
if not line.strip():
|
||||
continue
|
||||
d = json.loads(line)
|
||||
shape = d.get("csi_shape", [N_SUB, N_FRAMES])
|
||||
if shape != [N_SUB, N_FRAMES]:
|
||||
continue
|
||||
csi = np.asarray(d["csi"], dtype=np.float32).reshape(N_SUB, N_FRAMES)
|
||||
csis.append(csi)
|
||||
if max_samples and len(csis) >= max_samples:
|
||||
break
|
||||
return np.stack(csis)
|
||||
|
||||
|
||||
def perturb_subcarriers(X: np.ndarray, indices: list[int], gain: float) -> np.ndarray:
|
||||
"""Multiply the listed subcarriers by `gain` to simulate a structural
|
||||
change (e.g. a new reflector attenuates certain frequencies)."""
|
||||
out = X.copy()
|
||||
out[:, indices, :] *= gain
|
||||
return out
|
||||
|
||||
|
||||
def per_frame_matrix(X: np.ndarray) -> np.ndarray:
|
||||
"""Stack all windows' frames into a [N_total_frames, 56] matrix.
|
||||
Each row is one timestep, used as a multivariate observation of the
|
||||
56-subcarrier channel state."""
|
||||
return X.transpose(0, 2, 1).reshape(-1, N_SUB)
|
||||
|
||||
|
||||
def top_k_singular_values(M: np.ndarray, k: int = 10) -> np.ndarray:
|
||||
"""Compute SVD on M, return top-k singular values."""
|
||||
M_centered = M - M.mean(axis=0, keepdims=True)
|
||||
# Use SVD on the centered matrix (== PCA without normalisation)
|
||||
s = np.linalg.svd(M_centered, compute_uv=False)
|
||||
return s[:k]
|
||||
|
||||
|
||||
def spectrum_distance(s1: np.ndarray, s2: np.ndarray) -> float:
|
||||
"""Cosine distance between two singular-value spectra. 0 = identical
|
||||
direction, 2 = opposite. Symmetric, scale-invariant."""
|
||||
s1n = s1 / (np.linalg.norm(s1) + 1e-9)
|
||||
s2n = s2 / (np.linalg.norm(s2) + 1e-9)
|
||||
return float(1.0 - np.dot(s1n, s2n))
|
||||
|
||||
|
||||
def main():
|
||||
parser = argparse.ArgumentParser()
|
||||
parser.add_argument("--paired", required=True)
|
||||
parser.add_argument("--out", default="examples/research-sota/r12_rf_weather_results.json")
|
||||
parser.add_argument("--perturb-indices", default="30,41,52",
|
||||
help="comma-separated subcarrier indices to perturb (chosen from R5's top-saliency list)")
|
||||
parser.add_argument("--perturb-gain", type=float, default=0.85)
|
||||
args = parser.parse_args()
|
||||
|
||||
print(f"Loading windows from {args.paired}")
|
||||
X = load_windows(Path(args.paired))
|
||||
print(f" total windows: {X.shape[0]} (shape {X.shape})")
|
||||
|
||||
n = X.shape[0]
|
||||
half = n // 2
|
||||
X_before = X[:half]
|
||||
X_after_raw = X[half:] # unmodified second half — the CONTROL
|
||||
perturb_idx = [int(x) for x in args.perturb_indices.split(",")]
|
||||
X_after_perturbed = perturb_subcarriers(X_after_raw, perturb_idx, args.perturb_gain)
|
||||
|
||||
# Convert each half to a [N_frames, 56] matrix
|
||||
M_before = per_frame_matrix(X_before)
|
||||
M_after_raw = per_frame_matrix(X_after_raw)
|
||||
M_after_pert = per_frame_matrix(X_after_perturbed)
|
||||
print(f" per-frame matrix: before={M_before.shape}, after={M_after_raw.shape}")
|
||||
|
||||
# Top-10 singular values per half
|
||||
s_before = top_k_singular_values(M_before, k=10)
|
||||
s_after_raw = top_k_singular_values(M_after_raw, k=10)
|
||||
s_after_pert = top_k_singular_values(M_after_pert, k=10)
|
||||
|
||||
print(f"\n Singular value spectra (top-10):")
|
||||
print(f" before : [{', '.join(f'{v:.1f}' for v in s_before)}]")
|
||||
print(f" after (raw) : [{', '.join(f'{v:.1f}' for v in s_after_raw)}]")
|
||||
print(f" after (pert) : [{', '.join(f'{v:.1f}' for v in s_after_pert)}]")
|
||||
|
||||
# Distances
|
||||
d_raw = spectrum_distance(s_before, s_after_raw)
|
||||
d_pert = spectrum_distance(s_before, s_after_pert)
|
||||
|
||||
print(f"\n Cosine distances from BEFORE:")
|
||||
print(f" before -> after raw (control, no perturbation): {d_raw:.5f}")
|
||||
print(f" before -> after pert (synthetic structural shift): {d_pert:.5f}")
|
||||
|
||||
# Distance ratio = how much the perturbation amplifies the detection signal
|
||||
# over the natural temporal drift.
|
||||
if d_raw > 1e-9:
|
||||
ratio = d_pert / d_raw
|
||||
print(f"\n Signal-to-natural-drift ratio: {ratio:.2f}x")
|
||||
|
||||
if d_pert > d_raw * 3:
|
||||
verdict = "STRONG: perturbation easily distinguishable from natural temporal drift"
|
||||
elif d_pert > d_raw * 1.5:
|
||||
verdict = "MODERATE: perturbation detectable but with margin"
|
||||
else:
|
||||
verdict = "WEAK: structural perturbation gets lost in temporal drift"
|
||||
print(f"\n Verdict: {verdict}")
|
||||
|
||||
out = {
|
||||
"perturbation": {
|
||||
"subcarrier_indices": perturb_idx,
|
||||
"amplitude_gain": args.perturb_gain,
|
||||
"comment": "simulates a new reflective surface that attenuates these frequencies",
|
||||
},
|
||||
"n_before_windows": int(half),
|
||||
"n_after_windows": int(n - half),
|
||||
"spectra": {
|
||||
"before": s_before.tolist(),
|
||||
"after_raw_control": s_after_raw.tolist(),
|
||||
"after_perturbed": s_after_pert.tolist(),
|
||||
},
|
||||
"distances": {
|
||||
"before_to_after_raw": d_raw,
|
||||
"before_to_after_perturbed": d_pert,
|
||||
"signal_over_natural_drift": float(d_pert / max(d_raw, 1e-9)),
|
||||
},
|
||||
"verdict": verdict,
|
||||
}
|
||||
Path(args.out).parent.mkdir(parents=True, exist_ok=True)
|
||||
Path(args.out).write_text(json.dumps(out, indent=2))
|
||||
print(f"\nWrote {args.out}")
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
main()
|
||||
|
|
@ -0,0 +1,57 @@
|
|||
{
|
||||
"perturbation": {
|
||||
"subcarrier_indices": [
|
||||
30,
|
||||
41,
|
||||
52
|
||||
],
|
||||
"amplitude_gain": 0.85,
|
||||
"comment": "simulates a new reflective surface that attenuates these frequencies"
|
||||
},
|
||||
"n_before_windows": 538,
|
||||
"n_after_windows": 539,
|
||||
"spectra": {
|
||||
"before": [
|
||||
2220.65673828125,
|
||||
1856.8695068359375,
|
||||
1563.7314453125,
|
||||
1303.56298828125,
|
||||
1057.757080078125,
|
||||
770.67822265625,
|
||||
757.5601196289062,
|
||||
689.5866088867188,
|
||||
595.6748046875,
|
||||
556.3777465820312
|
||||
],
|
||||
"after_raw_control": [
|
||||
2182.5712890625,
|
||||
1837.5084228515625,
|
||||
1647.6357421875,
|
||||
1315.103759765625,
|
||||
1053.489013671875,
|
||||
794.1417236328125,
|
||||
737.1859130859375,
|
||||
704.1968994140625,
|
||||
571.363037109375,
|
||||
535.6047973632812
|
||||
],
|
||||
"after_perturbed": [
|
||||
2172.6552734375,
|
||||
1824.164794921875,
|
||||
1615.7850341796875,
|
||||
1304.227783203125,
|
||||
1040.461181640625,
|
||||
791.2919921875,
|
||||
736.2902221679688,
|
||||
691.3584594726562,
|
||||
568.5400390625,
|
||||
530.7666625976562
|
||||
]
|
||||
},
|
||||
"distances": {
|
||||
"before_to_after_raw": 0.0003509521484375,
|
||||
"before_to_after_perturbed": 0.00024056434631347656,
|
||||
"signal_over_natural_drift": 0.6854619565217391
|
||||
},
|
||||
"verdict": "WEAK: structural perturbation gets lost in temporal drift"
|
||||
}
|
||||
Loading…
Reference in New Issue