feat(v0.7-fire-tracker-1): registry correlation + 2 new categories
Phase 1 of the FIRMS+WFIGS fusion design doc. v13.sql adds fire_pixels
table for per-fire pixel history + spread_radius_mi/current_centroid_*/
last_hotspot_at on fires. FIRMS handler now attributes incoming pixels
to fires via point-in-circle within configurable radius (default 5 mi),
updating per-fire centroid as median of recent pixels. Unattributed
pixels go through a cluster detector: 3+ pixels within 1 mi within 60
min triggers a single unattributed_hotspot_cluster broadcast (Possible
new fire). Two new ALERT_CATEGORIES: wildfire_declared (priority,
WFIGS first-sight) and unattributed_hotspot_cluster (priority, FIRMS
cluster). All thresholds GUI-editable via adapter_config.fires.* and
adapter_config.firms.*. Phases 2-4 (movement analysis, spotting, LLM
summaries) deferred to subsequent commits.
Schema (v13.sql):
- fire_pixels table (irwin_id FK CASCADE, acq_time, lat/lon, frp,
satellite, pass_id, attributed_at). Indexed on (irwin_id, acq_time)
for centroid queries + on (acq_time) for Phase 2.
- fires gains spread_radius_mi (nullable; NULL => use global default),
current_centroid_lat/lon (median of last 24h pixels, distinct from
the WFIGS-declared anchor lat/lon), last_hotspot_at (Phase 2 halt
detector).
- firms_pixels gains attributed_at + cluster_broadcast_at + compound
index on (attributed_at, cluster_broadcast_at, acq_time) for the
cluster query.
adapter_config (defaults.py REGISTRY + ADAPTER_META):
- fires.spread_radius_mi_default = 5.0 (float)
- firms.cluster_min_pixels = 3 (int)
- firms.cluster_max_radius_mi = 1.0 (float)
- firms.cluster_time_window_minutes = 60 (int)
- ADAPTER_META["fires"] meta block (display_name + description).
ALERT_CATEGORIES (notifications/categories.py):
- wildfire_declared: priority/fire. WFIGS handler tags data["category"]
on cases (i)+(ii) [INSERT or row-exists-but-never-broadcast]; case
(iii) Update keeps the existing wildfire_incident category.
- unattributed_hotspot_cluster: priority/fire. FIRMS handler tags
data["category"] + data["severity"] when emitting the cluster wire.
FIRMS handler (central/firms_handler.py):
- Unchanged storage path: filter, INSERT OR IGNORE into firms_pixels.
- New _attribute_or_cluster() runs on every newly-stored pixel (dedup
hits skip -- the original insert had its shot already).
- Attribution: bbox prefilter on fires.tombstoned_at IS NULL, then
exact Haversine to fires(current_centroid_lat ?? lat,
current_centroid_lon ?? lon) inside spread_radius_mi (per-fire ?? global
default). Multi-match resolves to nearest (design doc Q2). On match:
INSERT fire_pixels, UPDATE firms_pixels.attributed_at,
recompute centroid as median of last 24h pixels for this fire.
- Cluster: on attribution miss, query firms_pixels WHERE attributed_at
IS NULL AND cluster_broadcast_at IS NULL AND acq_time > NOW-window.
If count >= cluster_min_pixels, fire the cluster wire and stamp
cluster_broadcast_at on every member so a 4th arrival cannot re-fire.
WFIGS handler (central/wfigs_handler.py): the existing prefix=New
branches (i)+(ii) now set data["category"]="wildfire_declared".
Existing _render() unchanged.
Wire strings:
- wildfire_declared: re-uses _render(prefix="New") -- emoji + name +
type + anchor + acres + containment + coords.
- unattributed_hotspot_cluster: _render_cluster_wire() emits
"Possible new fire: <N> hotspots within <r> mi @ <lat>,<lon>
(combined <total_frp> MW)".
Tests (tests/test_fire_tracker_phase1.py, 10 cases all green):
- Pixel within radius -> attribution + centroid + last_hotspot_at.
- Centroid recomputes as median across multiple passes.
- Pixel outside radius -> NO attribution + stays unattributed.
- 3 unattributed within 1 mi within 60 min -> cluster broadcast fires
exactly once, all 3 stamped cluster_broadcast_at.
- 4th pixel in the same footprint -> NO second broadcast (existing
3 are stamped so SQL filter excludes them).
- 5th-7th pixels 2h later -> form a NEW cluster (window prune fires).
- WFIGS first-sight tags data["category"]="wildfire_declared".
- WFIGS Update branch does NOT retag wildfire_declared.
- New adapter_config rows seeded on init_db.
- ALERT_CATEGORIES contains both new entries with correct toggle/severity.
Live verification on CT108 after rebuild:
- v13 migration applied (schema_meta version=13, no Traceback).
- adapter_config.fires.spread_radius_mi_default = 5.0
- adapter_config.firms.cluster_min_pixels = 3
- adapter_config.firms.cluster_max_radius_mi = 1.0
- adapter_config.firms.cluster_time_window_minutes = 60
- fires gains 4 new columns; firms_pixels gains 2 new columns; fire_pixels
table created.
- Container healthy, FIRMS pixels continue arriving (126 pre-deploy).
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-06-06 05:34:22 +00:00
|
|
|
"""v0.7-fire-tracker-1 tests.
|
|
|
|
|
|
|
|
|
|
Coverage map (vs the user-provided scope item 7):
|
|
|
|
|
- Pixel within radius -> attribution + centroid + last_hotspot_at
|
|
|
|
|
- Pixel outside radius -> no attribution, stays unattributed
|
|
|
|
|
- 3 unattributed within 1 mi -> cluster broadcast fires once
|
|
|
|
|
- 4th pixel same cluster -> NO second broadcast
|
|
|
|
|
- 5th pixel after 60 min -> can form a NEW cluster
|
|
|
|
|
|
|
|
|
|
Plus:
|
|
|
|
|
- wfigs first-sight tags data["category"]="wildfire_declared"
|
|
|
|
|
- wfigs Update path does NOT tag wildfire_declared
|
|
|
|
|
- Haversine sanity
|
|
|
|
|
"""
|
|
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
|
|
import os
|
|
|
|
|
import sqlite3
|
|
|
|
|
import time
|
|
|
|
|
import uuid
|
|
|
|
|
|
|
|
|
|
import pytest
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture(autouse=True)
|
|
|
|
|
def _isolate_db(tmp_path, monkeypatch):
|
|
|
|
|
"""Force MESHAI_DB_PATH to a tmp file per test + reset thread cache."""
|
|
|
|
|
db_path = str(tmp_path / f"meshai-{uuid.uuid4().hex}.sqlite")
|
|
|
|
|
monkeypatch.setenv("MESHAI_DB_PATH", db_path)
|
|
|
|
|
# The persistence module caches a connection on threading.local;
|
|
|
|
|
# reset between tests so we get a fresh DB each time.
|
|
|
|
|
from meshai.persistence import db as pdb
|
|
|
|
|
pdb.close_thread_connection()
|
|
|
|
|
pdb._initialised.discard(db_path)
|
|
|
|
|
from meshai.persistence import init_db
|
|
|
|
|
init_db(db_path)
|
|
|
|
|
yield db_path
|
|
|
|
|
pdb.close_thread_connection()
|
|
|
|
|
pdb._initialised.discard(db_path)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _seed_fire(*, irwin_id, lat, lon, name="Stub Fire", state="ID"):
|
|
|
|
|
"""Insert a minimal active fire row at known coords."""
|
|
|
|
|
from meshai.persistence import get_db
|
|
|
|
|
conn = get_db()
|
|
|
|
|
conn.execute(
|
|
|
|
|
"INSERT INTO fires(irwin_id, incident_name, lat, lon, "
|
|
|
|
|
"last_event_at) VALUES (?,?,?,?,?)",
|
|
|
|
|
(irwin_id, name, lat, lon, int(time.time())),
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _envelope(*, lat, lon, acq_date="2026-06-06", acq_time="1200",
|
|
|
|
|
frp=15.0, satellite="N20", conf="high"):
|
|
|
|
|
"""Build a Central FIRMS envelope shaped like the real Central feed."""
|
|
|
|
|
return {
|
|
|
|
|
"data": {
|
|
|
|
|
"adapter": "firms",
|
|
|
|
|
"category": "wildfire_hotspot",
|
|
|
|
|
"severity": "routine",
|
|
|
|
|
"data": {
|
|
|
|
|
"latitude": lat,
|
|
|
|
|
"longitude": lon,
|
|
|
|
|
"frp": frp,
|
|
|
|
|
"bright_ti4": 320.5,
|
|
|
|
|
"satellite": satellite,
|
|
|
|
|
"instrument": "VIIRS",
|
|
|
|
|
"confidence": conf,
|
|
|
|
|
"acq_date": acq_date,
|
|
|
|
|
"acq_time": acq_time,
|
|
|
|
|
"daynight": "D",
|
|
|
|
|
"version": "2.0NRT",
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
# (a) Attribution within radius.
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def test_pixel_within_radius_attributes_to_fire():
|
|
|
|
|
from meshai.central.firms_handler import handle_firms
|
|
|
|
|
from meshai.persistence import get_db
|
|
|
|
|
|
|
|
|
|
# Cache Peak Fire stub @ 42.118, -113.643.
|
|
|
|
|
_seed_fire(irwin_id="ID-TEST-001",
|
|
|
|
|
lat=42.118, lon=-113.643)
|
|
|
|
|
|
|
|
|
|
# FIRMS pixel ~0.2 mi NE of the anchor -- well inside default 5 mi.
|
|
|
|
|
env = _envelope(lat=42.121, lon=-113.640, frp=18.0)
|
|
|
|
|
wire = handle_firms(env, subject="central.fire.hotspot.N20.high.us.id",
|
|
|
|
|
data={}, now=1780728000)
|
|
|
|
|
# Attribution is silent (return None); the wire only fires on cluster.
|
|
|
|
|
assert wire is None
|
|
|
|
|
|
|
|
|
|
conn = get_db()
|
|
|
|
|
# fire_pixels row created.
|
|
|
|
|
rows = conn.execute("SELECT * FROM fire_pixels").fetchall()
|
|
|
|
|
assert len(rows) == 1
|
|
|
|
|
assert rows[0]["irwin_id"] == "ID-TEST-001"
|
|
|
|
|
assert rows[0]["frp"] == 18.0
|
|
|
|
|
# firms_pixels row has attributed_at stamped.
|
|
|
|
|
raw = conn.execute(
|
|
|
|
|
"SELECT attributed_at, cluster_broadcast_at FROM firms_pixels"
|
|
|
|
|
).fetchone()
|
|
|
|
|
assert raw["attributed_at"] is not None
|
|
|
|
|
assert raw["cluster_broadcast_at"] is None
|
|
|
|
|
# fires row has centroid + last_hotspot_at updated.
|
|
|
|
|
fire = conn.execute(
|
|
|
|
|
"SELECT current_centroid_lat, current_centroid_lon, last_hotspot_at "
|
|
|
|
|
"FROM fires WHERE irwin_id=?", ("ID-TEST-001",)
|
|
|
|
|
).fetchone()
|
|
|
|
|
assert fire["current_centroid_lat"] == 42.121
|
|
|
|
|
assert fire["current_centroid_lon"] == -113.640
|
|
|
|
|
assert fire["last_hotspot_at"] is not None
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def test_centroid_recomputes_as_median_across_passes():
|
|
|
|
|
"""A second attributed pixel updates the centroid to the median, not
|
|
|
|
|
just the latest pixel's coords."""
|
|
|
|
|
from meshai.central.firms_handler import handle_firms
|
|
|
|
|
from meshai.persistence import get_db
|
|
|
|
|
|
|
|
|
|
_seed_fire(irwin_id="ID-TEST-002",
|
|
|
|
|
lat=42.000, lon=-113.000)
|
|
|
|
|
|
feat(v0.7-fire-tracker-2): movement analysis -- growth + halt detection
Phase 2 of FIRMS+WFIGS fusion. v14.sql adds fire_passes table for
per-satellite-pass centroid tracking + drift computation. FIRMS handler
now detects pass boundaries (satellite + time bucket), computes pass
centroid (median of pass pixels), Haversine drift from previous pass,
bearing to 8-way direction, mi/h speed. Drift >= 0.5 mi (configurable)
emits wildfire_growth broadcast with wire including movement vector
and nearest-town context. Halt detection: fire with no new pixels for
>=12h (configurable) emits wildfire_halted broadcast (routine). Two
new ALERT_CATEGORIES: wildfire_growth (priority), wildfire_halted
(routine). All thresholds GUI-editable via adapter_config.fires.*.
Phase 3 (spotting) and Phase 4 (LLM summaries) deferred to subsequent
commits.
Schema (v14.sql):
- fire_passes table (irwin_id FK CASCADE, pass_id, pass_centroid_lat/lon,
pixel_count, total_frp, pass_started_at, pass_ended_at,
drift_mi_from_prev, drift_direction, drift_mi_per_hour). PRIMARY KEY
(irwin_id, pass_id) so the UPSERT path is cheap; secondary index on
(irwin_id, pass_ended_at) for the prev-pass lookup + halt counter.
- fires gains last_pass_id, last_pass_at, halt_broadcast_at columns.
halt_broadcast_at is latched per halt event; the detector filter
(halt_broadcast_at IS NULL OR halt_broadcast_at < last_pass_at)
reopens eligibility automatically when an idle fire receives a new
attributed pixel that advances last_pass_at.
adapter_config (defaults.py REGISTRY):
- fires.growth_drift_threshold_mi = 0.5 (float). Per-pass centroid drift
at or above this fires wildfire_growth. 0.5 mi matches the design
doc Phase 2 spec and is roughly 2x the VIIRS 375m pixel size (i.e.,
detectable as more than centroid jitter).
- fires.halt_passes_threshold = 2 (int). Documented intent; the
operational rule uses halt_minimum_seconds below as the time gate
because per-satellite pass-count enforcement would require modeling
the global VIIRS schedule per satellite. The 12h gate subsumes it
(4 passes/day in Idaho).
- fires.halt_minimum_seconds = 43200 (int, 12h).
ALERT_CATEGORIES (notifications/categories.py):
- wildfire_growth: priority/fire. FIRMS handler tags data["category"]
+ data["severity"] on the pass-boundary path when drift >= threshold.
- wildfire_halted: routine/fire. Halt detector tags data["category"]
+ data["severity"] when a fire transitions to idle for >=12h.
FIRMS handler (central/firms_handler.py):
- The Phase 1 attribution branch now passes through
_handle_pass_boundary(): UPSERT fire_passes row for the current
(irwin_id, pass_id) with median centroid + pixel count + total FRP
+ min/max acq_time; lookup the prior pass; compute drift mi +
8-way direction + mi/h speed and write them into the current pass
row (only the FIRST boundary fills these; subsequent in-pass pixels
COALESCE keep them stable). Update fires cursor (last_pass_id,
last_pass_at) and current_centroid_lat/lon to the latest pass
centroid -- this overrides Phase 1's 24h all-pixels median for
fires that have pass data.
- Growth wire emitted ONLY at the boundary (last_pass_id != current,
prev pass exists, drift >= threshold). Subsequent in-pass pixels
stay silent because pass_id == last_pass_id.
- _maybe_emit_halt runs as a final fallback when neither growth nor
cluster has fired. SELECT one fire matching the halt criteria,
stamp halt_broadcast_at, return the wire. The fallback ordering is
growth > cluster > halt so a busy fire's growth broadcast doesn't
starve a quiet fire's halt.
- New helpers: _bearing() (great-circle initial bearing, deg CW from N),
_direction_8() (compass 8-way mapping with +/-22.5 deg sectors).
Wire strings:
- wildfire_growth: `🔥 <incident_name> moving <dir> <speed:.1f> mi/h
~<dist_to_nearest_town:.1f> mi from <nearest_town>`. nearest_town
via meshai.central_normalizer.nearest_town (same Photon-backed
cache that wfigs_handler uses); failure falls back to bare
"moving <dir> <speed> mi/h".
- wildfire_halted: `🔥 <incident_name> no growth in <hours>h`.
Tests (tests/test_fire_tracker_phase2.py, 10 cases all green):
- 2-pass attribution with pass2 1.0 mi N of pass1 -> drift=1.0,
direction='N', mi/h computed, growth wire returned, data tagged.
- Drift below threshold (0.3 mi) -> NO growth broadcast; pass row
still records the (sub-threshold) drift for ops visibility.
- Halt detector: last_pass_at 14h ago -> fires once, halt_broadcast_at
stamped.
- Re-run halt detector with halt latched -> NO second broadcast.
- Halt re-eligibility: halt_broadcast_at < last_pass_at -> eligible
again (a resurrected then re-idled fire).
- Bearing + direction round-trip across all 8 cardinals.
- Direction sector boundary (22.5/67.5 deg) correctness.
- adapter_config seed for 3 new fires.* keys.
- Two new ALERT_CATEGORIES registered.
- 5-pixel single-pass aggregate (pixel_count, total_frp sum, median
centroid, started/ended_at min/max).
Phase 1 test fix:
- tests/test_fire_tracker_phase1.py::test_centroid_recomputes_as_median_across_passes
retimed to 12:00/12:10/12:20 so all 3 pixels land in one
N20 bucket. Phase 2 makes current_centroid_* the per-pass median
(latest pass overrides Phase 1's 24h median); the same-pass shape
preserves the original median-computation intent. 39 total tests
green across phase1/phase2/or-arch/include-roundtrip.
Live verification on CT108 after rebuild:
- v14 migration applied (schema_meta version=14, no Traceback in 3 min).
- adapter_config.fires.growth_drift_threshold_mi = 0.5
- adapter_config.fires.halt_passes_threshold = 2
- adapter_config.fires.halt_minimum_seconds = 43200
- Container healthy.
Synthetic 100-pixel probe inside prod container (PROBE-V07P2-*,
cleaned up after):
- Pass A (50 pixels @ 12:00-12:25, N20 bucket 329768): centroid
(44.30000, -115.50000), pixel_count=50, total_frp=975.0, drift=NULL
(first pass).
- Pass B (50 pixels @ 18:00-18:25, N20 bucket 329772, centered 1.2 mi
NE of A): centroid (44.31230, -115.48282), pixel_count=50,
total_frp=975.0, drift_mi_from_prev=1.1703 (~design target 1.2 mi
with -0.03 mi rounding), drift_direction="NE",
drift_mi_per_hour=0.209 (1.17 mi over 5.5h between pass ends).
- Growth wire: "🔥 Probe Movement Fire moving NE 0.2 mi/h, ~13.0 mi
from Long Creek Summit Home" (Photon nearest-town anchor populated
successfully).
- Exactly ONE growth broadcast (first pixel of pass B); 99 other
pixels stayed silent.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-06-06 06:12:36 +00:00
|
|
|
# 3 pixels within radius at distinct coords. v0.7-fire-tracker-2
|
|
|
|
|
# makes fires.current_centroid_* the latest PASS centroid (per-pass
|
|
|
|
|
# median), overriding Phase 1's 24h all-pixels median. Use acq
|
|
|
|
|
# times within a single ~90 min satellite bucket so all 3 land in
|
|
|
|
|
# ONE fire_passes row -- then the per-pass median IS the 3-pixel
|
|
|
|
|
# median and this test's intent (verify median computation, not
|
|
|
|
|
# arithmetic mean) survives the Phase 2 semantic shift.
|
feat(v0.7-fire-tracker-1): registry correlation + 2 new categories
Phase 1 of the FIRMS+WFIGS fusion design doc. v13.sql adds fire_pixels
table for per-fire pixel history + spread_radius_mi/current_centroid_*/
last_hotspot_at on fires. FIRMS handler now attributes incoming pixels
to fires via point-in-circle within configurable radius (default 5 mi),
updating per-fire centroid as median of recent pixels. Unattributed
pixels go through a cluster detector: 3+ pixels within 1 mi within 60
min triggers a single unattributed_hotspot_cluster broadcast (Possible
new fire). Two new ALERT_CATEGORIES: wildfire_declared (priority,
WFIGS first-sight) and unattributed_hotspot_cluster (priority, FIRMS
cluster). All thresholds GUI-editable via adapter_config.fires.* and
adapter_config.firms.*. Phases 2-4 (movement analysis, spotting, LLM
summaries) deferred to subsequent commits.
Schema (v13.sql):
- fire_pixels table (irwin_id FK CASCADE, acq_time, lat/lon, frp,
satellite, pass_id, attributed_at). Indexed on (irwin_id, acq_time)
for centroid queries + on (acq_time) for Phase 2.
- fires gains spread_radius_mi (nullable; NULL => use global default),
current_centroid_lat/lon (median of last 24h pixels, distinct from
the WFIGS-declared anchor lat/lon), last_hotspot_at (Phase 2 halt
detector).
- firms_pixels gains attributed_at + cluster_broadcast_at + compound
index on (attributed_at, cluster_broadcast_at, acq_time) for the
cluster query.
adapter_config (defaults.py REGISTRY + ADAPTER_META):
- fires.spread_radius_mi_default = 5.0 (float)
- firms.cluster_min_pixels = 3 (int)
- firms.cluster_max_radius_mi = 1.0 (float)
- firms.cluster_time_window_minutes = 60 (int)
- ADAPTER_META["fires"] meta block (display_name + description).
ALERT_CATEGORIES (notifications/categories.py):
- wildfire_declared: priority/fire. WFIGS handler tags data["category"]
on cases (i)+(ii) [INSERT or row-exists-but-never-broadcast]; case
(iii) Update keeps the existing wildfire_incident category.
- unattributed_hotspot_cluster: priority/fire. FIRMS handler tags
data["category"] + data["severity"] when emitting the cluster wire.
FIRMS handler (central/firms_handler.py):
- Unchanged storage path: filter, INSERT OR IGNORE into firms_pixels.
- New _attribute_or_cluster() runs on every newly-stored pixel (dedup
hits skip -- the original insert had its shot already).
- Attribution: bbox prefilter on fires.tombstoned_at IS NULL, then
exact Haversine to fires(current_centroid_lat ?? lat,
current_centroid_lon ?? lon) inside spread_radius_mi (per-fire ?? global
default). Multi-match resolves to nearest (design doc Q2). On match:
INSERT fire_pixels, UPDATE firms_pixels.attributed_at,
recompute centroid as median of last 24h pixels for this fire.
- Cluster: on attribution miss, query firms_pixels WHERE attributed_at
IS NULL AND cluster_broadcast_at IS NULL AND acq_time > NOW-window.
If count >= cluster_min_pixels, fire the cluster wire and stamp
cluster_broadcast_at on every member so a 4th arrival cannot re-fire.
WFIGS handler (central/wfigs_handler.py): the existing prefix=New
branches (i)+(ii) now set data["category"]="wildfire_declared".
Existing _render() unchanged.
Wire strings:
- wildfire_declared: re-uses _render(prefix="New") -- emoji + name +
type + anchor + acres + containment + coords.
- unattributed_hotspot_cluster: _render_cluster_wire() emits
"Possible new fire: <N> hotspots within <r> mi @ <lat>,<lon>
(combined <total_frp> MW)".
Tests (tests/test_fire_tracker_phase1.py, 10 cases all green):
- Pixel within radius -> attribution + centroid + last_hotspot_at.
- Centroid recomputes as median across multiple passes.
- Pixel outside radius -> NO attribution + stays unattributed.
- 3 unattributed within 1 mi within 60 min -> cluster broadcast fires
exactly once, all 3 stamped cluster_broadcast_at.
- 4th pixel in the same footprint -> NO second broadcast (existing
3 are stamped so SQL filter excludes them).
- 5th-7th pixels 2h later -> form a NEW cluster (window prune fires).
- WFIGS first-sight tags data["category"]="wildfire_declared".
- WFIGS Update branch does NOT retag wildfire_declared.
- New adapter_config rows seeded on init_db.
- ALERT_CATEGORIES contains both new entries with correct toggle/severity.
Live verification on CT108 after rebuild:
- v13 migration applied (schema_meta version=13, no Traceback).
- adapter_config.fires.spread_radius_mi_default = 5.0
- adapter_config.firms.cluster_min_pixels = 3
- adapter_config.firms.cluster_max_radius_mi = 1.0
- adapter_config.firms.cluster_time_window_minutes = 60
- fires gains 4 new columns; firms_pixels gains 2 new columns; fire_pixels
table created.
- Container healthy, FIRMS pixels continue arriving (126 pre-deploy).
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-06-06 05:34:22 +00:00
|
|
|
coords = [(42.001, -113.001), (42.002, -113.002), (42.003, -113.003)]
|
|
|
|
|
for i, (la, lo) in enumerate(coords):
|
|
|
|
|
env = _envelope(lat=la, lon=lo,
|
|
|
|
|
acq_date="2026-06-06",
|
feat(v0.7-fire-tracker-2): movement analysis -- growth + halt detection
Phase 2 of FIRMS+WFIGS fusion. v14.sql adds fire_passes table for
per-satellite-pass centroid tracking + drift computation. FIRMS handler
now detects pass boundaries (satellite + time bucket), computes pass
centroid (median of pass pixels), Haversine drift from previous pass,
bearing to 8-way direction, mi/h speed. Drift >= 0.5 mi (configurable)
emits wildfire_growth broadcast with wire including movement vector
and nearest-town context. Halt detection: fire with no new pixels for
>=12h (configurable) emits wildfire_halted broadcast (routine). Two
new ALERT_CATEGORIES: wildfire_growth (priority), wildfire_halted
(routine). All thresholds GUI-editable via adapter_config.fires.*.
Phase 3 (spotting) and Phase 4 (LLM summaries) deferred to subsequent
commits.
Schema (v14.sql):
- fire_passes table (irwin_id FK CASCADE, pass_id, pass_centroid_lat/lon,
pixel_count, total_frp, pass_started_at, pass_ended_at,
drift_mi_from_prev, drift_direction, drift_mi_per_hour). PRIMARY KEY
(irwin_id, pass_id) so the UPSERT path is cheap; secondary index on
(irwin_id, pass_ended_at) for the prev-pass lookup + halt counter.
- fires gains last_pass_id, last_pass_at, halt_broadcast_at columns.
halt_broadcast_at is latched per halt event; the detector filter
(halt_broadcast_at IS NULL OR halt_broadcast_at < last_pass_at)
reopens eligibility automatically when an idle fire receives a new
attributed pixel that advances last_pass_at.
adapter_config (defaults.py REGISTRY):
- fires.growth_drift_threshold_mi = 0.5 (float). Per-pass centroid drift
at or above this fires wildfire_growth. 0.5 mi matches the design
doc Phase 2 spec and is roughly 2x the VIIRS 375m pixel size (i.e.,
detectable as more than centroid jitter).
- fires.halt_passes_threshold = 2 (int). Documented intent; the
operational rule uses halt_minimum_seconds below as the time gate
because per-satellite pass-count enforcement would require modeling
the global VIIRS schedule per satellite. The 12h gate subsumes it
(4 passes/day in Idaho).
- fires.halt_minimum_seconds = 43200 (int, 12h).
ALERT_CATEGORIES (notifications/categories.py):
- wildfire_growth: priority/fire. FIRMS handler tags data["category"]
+ data["severity"] on the pass-boundary path when drift >= threshold.
- wildfire_halted: routine/fire. Halt detector tags data["category"]
+ data["severity"] when a fire transitions to idle for >=12h.
FIRMS handler (central/firms_handler.py):
- The Phase 1 attribution branch now passes through
_handle_pass_boundary(): UPSERT fire_passes row for the current
(irwin_id, pass_id) with median centroid + pixel count + total FRP
+ min/max acq_time; lookup the prior pass; compute drift mi +
8-way direction + mi/h speed and write them into the current pass
row (only the FIRST boundary fills these; subsequent in-pass pixels
COALESCE keep them stable). Update fires cursor (last_pass_id,
last_pass_at) and current_centroid_lat/lon to the latest pass
centroid -- this overrides Phase 1's 24h all-pixels median for
fires that have pass data.
- Growth wire emitted ONLY at the boundary (last_pass_id != current,
prev pass exists, drift >= threshold). Subsequent in-pass pixels
stay silent because pass_id == last_pass_id.
- _maybe_emit_halt runs as a final fallback when neither growth nor
cluster has fired. SELECT one fire matching the halt criteria,
stamp halt_broadcast_at, return the wire. The fallback ordering is
growth > cluster > halt so a busy fire's growth broadcast doesn't
starve a quiet fire's halt.
- New helpers: _bearing() (great-circle initial bearing, deg CW from N),
_direction_8() (compass 8-way mapping with +/-22.5 deg sectors).
Wire strings:
- wildfire_growth: `🔥 <incident_name> moving <dir> <speed:.1f> mi/h
~<dist_to_nearest_town:.1f> mi from <nearest_town>`. nearest_town
via meshai.central_normalizer.nearest_town (same Photon-backed
cache that wfigs_handler uses); failure falls back to bare
"moving <dir> <speed> mi/h".
- wildfire_halted: `🔥 <incident_name> no growth in <hours>h`.
Tests (tests/test_fire_tracker_phase2.py, 10 cases all green):
- 2-pass attribution with pass2 1.0 mi N of pass1 -> drift=1.0,
direction='N', mi/h computed, growth wire returned, data tagged.
- Drift below threshold (0.3 mi) -> NO growth broadcast; pass row
still records the (sub-threshold) drift for ops visibility.
- Halt detector: last_pass_at 14h ago -> fires once, halt_broadcast_at
stamped.
- Re-run halt detector with halt latched -> NO second broadcast.
- Halt re-eligibility: halt_broadcast_at < last_pass_at -> eligible
again (a resurrected then re-idled fire).
- Bearing + direction round-trip across all 8 cardinals.
- Direction sector boundary (22.5/67.5 deg) correctness.
- adapter_config seed for 3 new fires.* keys.
- Two new ALERT_CATEGORIES registered.
- 5-pixel single-pass aggregate (pixel_count, total_frp sum, median
centroid, started/ended_at min/max).
Phase 1 test fix:
- tests/test_fire_tracker_phase1.py::test_centroid_recomputes_as_median_across_passes
retimed to 12:00/12:10/12:20 so all 3 pixels land in one
N20 bucket. Phase 2 makes current_centroid_* the per-pass median
(latest pass overrides Phase 1's 24h median); the same-pass shape
preserves the original median-computation intent. 39 total tests
green across phase1/phase2/or-arch/include-roundtrip.
Live verification on CT108 after rebuild:
- v14 migration applied (schema_meta version=14, no Traceback in 3 min).
- adapter_config.fires.growth_drift_threshold_mi = 0.5
- adapter_config.fires.halt_passes_threshold = 2
- adapter_config.fires.halt_minimum_seconds = 43200
- Container healthy.
Synthetic 100-pixel probe inside prod container (PROBE-V07P2-*,
cleaned up after):
- Pass A (50 pixels @ 12:00-12:25, N20 bucket 329768): centroid
(44.30000, -115.50000), pixel_count=50, total_frp=975.0, drift=NULL
(first pass).
- Pass B (50 pixels @ 18:00-18:25, N20 bucket 329772, centered 1.2 mi
NE of A): centroid (44.31230, -115.48282), pixel_count=50,
total_frp=975.0, drift_mi_from_prev=1.1703 (~design target 1.2 mi
with -0.03 mi rounding), drift_direction="NE",
drift_mi_per_hour=0.209 (1.17 mi over 5.5h between pass ends).
- Growth wire: "🔥 Probe Movement Fire moving NE 0.2 mi/h, ~13.0 mi
from Long Creek Summit Home" (Photon nearest-town anchor populated
successfully).
- Exactly ONE growth broadcast (first pixel of pass B); 99 other
pixels stayed silent.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-06-06 06:12:36 +00:00
|
|
|
acq_time=f"12{i * 10:02d}") # 12:00, 12:10, 12:20
|
feat(v0.7-fire-tracker-1): registry correlation + 2 new categories
Phase 1 of the FIRMS+WFIGS fusion design doc. v13.sql adds fire_pixels
table for per-fire pixel history + spread_radius_mi/current_centroid_*/
last_hotspot_at on fires. FIRMS handler now attributes incoming pixels
to fires via point-in-circle within configurable radius (default 5 mi),
updating per-fire centroid as median of recent pixels. Unattributed
pixels go through a cluster detector: 3+ pixels within 1 mi within 60
min triggers a single unattributed_hotspot_cluster broadcast (Possible
new fire). Two new ALERT_CATEGORIES: wildfire_declared (priority,
WFIGS first-sight) and unattributed_hotspot_cluster (priority, FIRMS
cluster). All thresholds GUI-editable via adapter_config.fires.* and
adapter_config.firms.*. Phases 2-4 (movement analysis, spotting, LLM
summaries) deferred to subsequent commits.
Schema (v13.sql):
- fire_pixels table (irwin_id FK CASCADE, acq_time, lat/lon, frp,
satellite, pass_id, attributed_at). Indexed on (irwin_id, acq_time)
for centroid queries + on (acq_time) for Phase 2.
- fires gains spread_radius_mi (nullable; NULL => use global default),
current_centroid_lat/lon (median of last 24h pixels, distinct from
the WFIGS-declared anchor lat/lon), last_hotspot_at (Phase 2 halt
detector).
- firms_pixels gains attributed_at + cluster_broadcast_at + compound
index on (attributed_at, cluster_broadcast_at, acq_time) for the
cluster query.
adapter_config (defaults.py REGISTRY + ADAPTER_META):
- fires.spread_radius_mi_default = 5.0 (float)
- firms.cluster_min_pixels = 3 (int)
- firms.cluster_max_radius_mi = 1.0 (float)
- firms.cluster_time_window_minutes = 60 (int)
- ADAPTER_META["fires"] meta block (display_name + description).
ALERT_CATEGORIES (notifications/categories.py):
- wildfire_declared: priority/fire. WFIGS handler tags data["category"]
on cases (i)+(ii) [INSERT or row-exists-but-never-broadcast]; case
(iii) Update keeps the existing wildfire_incident category.
- unattributed_hotspot_cluster: priority/fire. FIRMS handler tags
data["category"] + data["severity"] when emitting the cluster wire.
FIRMS handler (central/firms_handler.py):
- Unchanged storage path: filter, INSERT OR IGNORE into firms_pixels.
- New _attribute_or_cluster() runs on every newly-stored pixel (dedup
hits skip -- the original insert had its shot already).
- Attribution: bbox prefilter on fires.tombstoned_at IS NULL, then
exact Haversine to fires(current_centroid_lat ?? lat,
current_centroid_lon ?? lon) inside spread_radius_mi (per-fire ?? global
default). Multi-match resolves to nearest (design doc Q2). On match:
INSERT fire_pixels, UPDATE firms_pixels.attributed_at,
recompute centroid as median of last 24h pixels for this fire.
- Cluster: on attribution miss, query firms_pixels WHERE attributed_at
IS NULL AND cluster_broadcast_at IS NULL AND acq_time > NOW-window.
If count >= cluster_min_pixels, fire the cluster wire and stamp
cluster_broadcast_at on every member so a 4th arrival cannot re-fire.
WFIGS handler (central/wfigs_handler.py): the existing prefix=New
branches (i)+(ii) now set data["category"]="wildfire_declared".
Existing _render() unchanged.
Wire strings:
- wildfire_declared: re-uses _render(prefix="New") -- emoji + name +
type + anchor + acres + containment + coords.
- unattributed_hotspot_cluster: _render_cluster_wire() emits
"Possible new fire: <N> hotspots within <r> mi @ <lat>,<lon>
(combined <total_frp> MW)".
Tests (tests/test_fire_tracker_phase1.py, 10 cases all green):
- Pixel within radius -> attribution + centroid + last_hotspot_at.
- Centroid recomputes as median across multiple passes.
- Pixel outside radius -> NO attribution + stays unattributed.
- 3 unattributed within 1 mi within 60 min -> cluster broadcast fires
exactly once, all 3 stamped cluster_broadcast_at.
- 4th pixel in the same footprint -> NO second broadcast (existing
3 are stamped so SQL filter excludes them).
- 5th-7th pixels 2h later -> form a NEW cluster (window prune fires).
- WFIGS first-sight tags data["category"]="wildfire_declared".
- WFIGS Update branch does NOT retag wildfire_declared.
- New adapter_config rows seeded on init_db.
- ALERT_CATEGORIES contains both new entries with correct toggle/severity.
Live verification on CT108 after rebuild:
- v13 migration applied (schema_meta version=13, no Traceback).
- adapter_config.fires.spread_radius_mi_default = 5.0
- adapter_config.firms.cluster_min_pixels = 3
- adapter_config.firms.cluster_max_radius_mi = 1.0
- adapter_config.firms.cluster_time_window_minutes = 60
- fires gains 4 new columns; firms_pixels gains 2 new columns; fire_pixels
table created.
- Container healthy, FIRMS pixels continue arriving (126 pre-deploy).
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-06-06 05:34:22 +00:00
|
|
|
handle_firms(env,
|
|
|
|
|
subject="central.fire.hotspot.N20.high.us.id",
|
feat(v0.7-fire-tracker-2): movement analysis -- growth + halt detection
Phase 2 of FIRMS+WFIGS fusion. v14.sql adds fire_passes table for
per-satellite-pass centroid tracking + drift computation. FIRMS handler
now detects pass boundaries (satellite + time bucket), computes pass
centroid (median of pass pixels), Haversine drift from previous pass,
bearing to 8-way direction, mi/h speed. Drift >= 0.5 mi (configurable)
emits wildfire_growth broadcast with wire including movement vector
and nearest-town context. Halt detection: fire with no new pixels for
>=12h (configurable) emits wildfire_halted broadcast (routine). Two
new ALERT_CATEGORIES: wildfire_growth (priority), wildfire_halted
(routine). All thresholds GUI-editable via adapter_config.fires.*.
Phase 3 (spotting) and Phase 4 (LLM summaries) deferred to subsequent
commits.
Schema (v14.sql):
- fire_passes table (irwin_id FK CASCADE, pass_id, pass_centroid_lat/lon,
pixel_count, total_frp, pass_started_at, pass_ended_at,
drift_mi_from_prev, drift_direction, drift_mi_per_hour). PRIMARY KEY
(irwin_id, pass_id) so the UPSERT path is cheap; secondary index on
(irwin_id, pass_ended_at) for the prev-pass lookup + halt counter.
- fires gains last_pass_id, last_pass_at, halt_broadcast_at columns.
halt_broadcast_at is latched per halt event; the detector filter
(halt_broadcast_at IS NULL OR halt_broadcast_at < last_pass_at)
reopens eligibility automatically when an idle fire receives a new
attributed pixel that advances last_pass_at.
adapter_config (defaults.py REGISTRY):
- fires.growth_drift_threshold_mi = 0.5 (float). Per-pass centroid drift
at or above this fires wildfire_growth. 0.5 mi matches the design
doc Phase 2 spec and is roughly 2x the VIIRS 375m pixel size (i.e.,
detectable as more than centroid jitter).
- fires.halt_passes_threshold = 2 (int). Documented intent; the
operational rule uses halt_minimum_seconds below as the time gate
because per-satellite pass-count enforcement would require modeling
the global VIIRS schedule per satellite. The 12h gate subsumes it
(4 passes/day in Idaho).
- fires.halt_minimum_seconds = 43200 (int, 12h).
ALERT_CATEGORIES (notifications/categories.py):
- wildfire_growth: priority/fire. FIRMS handler tags data["category"]
+ data["severity"] on the pass-boundary path when drift >= threshold.
- wildfire_halted: routine/fire. Halt detector tags data["category"]
+ data["severity"] when a fire transitions to idle for >=12h.
FIRMS handler (central/firms_handler.py):
- The Phase 1 attribution branch now passes through
_handle_pass_boundary(): UPSERT fire_passes row for the current
(irwin_id, pass_id) with median centroid + pixel count + total FRP
+ min/max acq_time; lookup the prior pass; compute drift mi +
8-way direction + mi/h speed and write them into the current pass
row (only the FIRST boundary fills these; subsequent in-pass pixels
COALESCE keep them stable). Update fires cursor (last_pass_id,
last_pass_at) and current_centroid_lat/lon to the latest pass
centroid -- this overrides Phase 1's 24h all-pixels median for
fires that have pass data.
- Growth wire emitted ONLY at the boundary (last_pass_id != current,
prev pass exists, drift >= threshold). Subsequent in-pass pixels
stay silent because pass_id == last_pass_id.
- _maybe_emit_halt runs as a final fallback when neither growth nor
cluster has fired. SELECT one fire matching the halt criteria,
stamp halt_broadcast_at, return the wire. The fallback ordering is
growth > cluster > halt so a busy fire's growth broadcast doesn't
starve a quiet fire's halt.
- New helpers: _bearing() (great-circle initial bearing, deg CW from N),
_direction_8() (compass 8-way mapping with +/-22.5 deg sectors).
Wire strings:
- wildfire_growth: `🔥 <incident_name> moving <dir> <speed:.1f> mi/h
~<dist_to_nearest_town:.1f> mi from <nearest_town>`. nearest_town
via meshai.central_normalizer.nearest_town (same Photon-backed
cache that wfigs_handler uses); failure falls back to bare
"moving <dir> <speed> mi/h".
- wildfire_halted: `🔥 <incident_name> no growth in <hours>h`.
Tests (tests/test_fire_tracker_phase2.py, 10 cases all green):
- 2-pass attribution with pass2 1.0 mi N of pass1 -> drift=1.0,
direction='N', mi/h computed, growth wire returned, data tagged.
- Drift below threshold (0.3 mi) -> NO growth broadcast; pass row
still records the (sub-threshold) drift for ops visibility.
- Halt detector: last_pass_at 14h ago -> fires once, halt_broadcast_at
stamped.
- Re-run halt detector with halt latched -> NO second broadcast.
- Halt re-eligibility: halt_broadcast_at < last_pass_at -> eligible
again (a resurrected then re-idled fire).
- Bearing + direction round-trip across all 8 cardinals.
- Direction sector boundary (22.5/67.5 deg) correctness.
- adapter_config seed for 3 new fires.* keys.
- Two new ALERT_CATEGORIES registered.
- 5-pixel single-pass aggregate (pixel_count, total_frp sum, median
centroid, started/ended_at min/max).
Phase 1 test fix:
- tests/test_fire_tracker_phase1.py::test_centroid_recomputes_as_median_across_passes
retimed to 12:00/12:10/12:20 so all 3 pixels land in one
N20 bucket. Phase 2 makes current_centroid_* the per-pass median
(latest pass overrides Phase 1's 24h median); the same-pass shape
preserves the original median-computation intent. 39 total tests
green across phase1/phase2/or-arch/include-roundtrip.
Live verification on CT108 after rebuild:
- v14 migration applied (schema_meta version=14, no Traceback in 3 min).
- adapter_config.fires.growth_drift_threshold_mi = 0.5
- adapter_config.fires.halt_passes_threshold = 2
- adapter_config.fires.halt_minimum_seconds = 43200
- Container healthy.
Synthetic 100-pixel probe inside prod container (PROBE-V07P2-*,
cleaned up after):
- Pass A (50 pixels @ 12:00-12:25, N20 bucket 329768): centroid
(44.30000, -115.50000), pixel_count=50, total_frp=975.0, drift=NULL
(first pass).
- Pass B (50 pixels @ 18:00-18:25, N20 bucket 329772, centered 1.2 mi
NE of A): centroid (44.31230, -115.48282), pixel_count=50,
total_frp=975.0, drift_mi_from_prev=1.1703 (~design target 1.2 mi
with -0.03 mi rounding), drift_direction="NE",
drift_mi_per_hour=0.209 (1.17 mi over 5.5h between pass ends).
- Growth wire: "🔥 Probe Movement Fire moving NE 0.2 mi/h, ~13.0 mi
from Long Creek Summit Home" (Photon nearest-town anchor populated
successfully).
- Exactly ONE growth broadcast (first pixel of pass B); 99 other
pixels stayed silent.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-06-06 06:12:36 +00:00
|
|
|
data={}, now=1780728000 + i * 600)
|
feat(v0.7-fire-tracker-1): registry correlation + 2 new categories
Phase 1 of the FIRMS+WFIGS fusion design doc. v13.sql adds fire_pixels
table for per-fire pixel history + spread_radius_mi/current_centroid_*/
last_hotspot_at on fires. FIRMS handler now attributes incoming pixels
to fires via point-in-circle within configurable radius (default 5 mi),
updating per-fire centroid as median of recent pixels. Unattributed
pixels go through a cluster detector: 3+ pixels within 1 mi within 60
min triggers a single unattributed_hotspot_cluster broadcast (Possible
new fire). Two new ALERT_CATEGORIES: wildfire_declared (priority,
WFIGS first-sight) and unattributed_hotspot_cluster (priority, FIRMS
cluster). All thresholds GUI-editable via adapter_config.fires.* and
adapter_config.firms.*. Phases 2-4 (movement analysis, spotting, LLM
summaries) deferred to subsequent commits.
Schema (v13.sql):
- fire_pixels table (irwin_id FK CASCADE, acq_time, lat/lon, frp,
satellite, pass_id, attributed_at). Indexed on (irwin_id, acq_time)
for centroid queries + on (acq_time) for Phase 2.
- fires gains spread_radius_mi (nullable; NULL => use global default),
current_centroid_lat/lon (median of last 24h pixels, distinct from
the WFIGS-declared anchor lat/lon), last_hotspot_at (Phase 2 halt
detector).
- firms_pixels gains attributed_at + cluster_broadcast_at + compound
index on (attributed_at, cluster_broadcast_at, acq_time) for the
cluster query.
adapter_config (defaults.py REGISTRY + ADAPTER_META):
- fires.spread_radius_mi_default = 5.0 (float)
- firms.cluster_min_pixels = 3 (int)
- firms.cluster_max_radius_mi = 1.0 (float)
- firms.cluster_time_window_minutes = 60 (int)
- ADAPTER_META["fires"] meta block (display_name + description).
ALERT_CATEGORIES (notifications/categories.py):
- wildfire_declared: priority/fire. WFIGS handler tags data["category"]
on cases (i)+(ii) [INSERT or row-exists-but-never-broadcast]; case
(iii) Update keeps the existing wildfire_incident category.
- unattributed_hotspot_cluster: priority/fire. FIRMS handler tags
data["category"] + data["severity"] when emitting the cluster wire.
FIRMS handler (central/firms_handler.py):
- Unchanged storage path: filter, INSERT OR IGNORE into firms_pixels.
- New _attribute_or_cluster() runs on every newly-stored pixel (dedup
hits skip -- the original insert had its shot already).
- Attribution: bbox prefilter on fires.tombstoned_at IS NULL, then
exact Haversine to fires(current_centroid_lat ?? lat,
current_centroid_lon ?? lon) inside spread_radius_mi (per-fire ?? global
default). Multi-match resolves to nearest (design doc Q2). On match:
INSERT fire_pixels, UPDATE firms_pixels.attributed_at,
recompute centroid as median of last 24h pixels for this fire.
- Cluster: on attribution miss, query firms_pixels WHERE attributed_at
IS NULL AND cluster_broadcast_at IS NULL AND acq_time > NOW-window.
If count >= cluster_min_pixels, fire the cluster wire and stamp
cluster_broadcast_at on every member so a 4th arrival cannot re-fire.
WFIGS handler (central/wfigs_handler.py): the existing prefix=New
branches (i)+(ii) now set data["category"]="wildfire_declared".
Existing _render() unchanged.
Wire strings:
- wildfire_declared: re-uses _render(prefix="New") -- emoji + name +
type + anchor + acres + containment + coords.
- unattributed_hotspot_cluster: _render_cluster_wire() emits
"Possible new fire: <N> hotspots within <r> mi @ <lat>,<lon>
(combined <total_frp> MW)".
Tests (tests/test_fire_tracker_phase1.py, 10 cases all green):
- Pixel within radius -> attribution + centroid + last_hotspot_at.
- Centroid recomputes as median across multiple passes.
- Pixel outside radius -> NO attribution + stays unattributed.
- 3 unattributed within 1 mi within 60 min -> cluster broadcast fires
exactly once, all 3 stamped cluster_broadcast_at.
- 4th pixel in the same footprint -> NO second broadcast (existing
3 are stamped so SQL filter excludes them).
- 5th-7th pixels 2h later -> form a NEW cluster (window prune fires).
- WFIGS first-sight tags data["category"]="wildfire_declared".
- WFIGS Update branch does NOT retag wildfire_declared.
- New adapter_config rows seeded on init_db.
- ALERT_CATEGORIES contains both new entries with correct toggle/severity.
Live verification on CT108 after rebuild:
- v13 migration applied (schema_meta version=13, no Traceback).
- adapter_config.fires.spread_radius_mi_default = 5.0
- adapter_config.firms.cluster_min_pixels = 3
- adapter_config.firms.cluster_max_radius_mi = 1.0
- adapter_config.firms.cluster_time_window_minutes = 60
- fires gains 4 new columns; firms_pixels gains 2 new columns; fire_pixels
table created.
- Container healthy, FIRMS pixels continue arriving (126 pre-deploy).
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-06-06 05:34:22 +00:00
|
|
|
|
|
|
|
|
fire = get_db().execute(
|
|
|
|
|
"SELECT current_centroid_lat, current_centroid_lon "
|
|
|
|
|
"FROM fires WHERE irwin_id=?", ("ID-TEST-002",)
|
|
|
|
|
).fetchone()
|
|
|
|
|
# Median of 3 sorted lats = middle = 42.002. Same for lons.
|
|
|
|
|
assert abs(fire["current_centroid_lat"] - 42.002) < 1e-9
|
|
|
|
|
assert abs(fire["current_centroid_lon"] - -113.002) < 1e-9
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
# (b) Pixel outside radius -- no attribution.
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def test_pixel_outside_radius_stays_unattributed():
|
|
|
|
|
from meshai.central.firms_handler import handle_firms
|
|
|
|
|
from meshai.persistence import get_db
|
|
|
|
|
|
|
|
|
|
_seed_fire(irwin_id="ID-TEST-003",
|
|
|
|
|
lat=42.000, lon=-113.000)
|
|
|
|
|
|
|
|
|
|
# Pixel ~50 mi away -- comfortably outside the 5 mi default.
|
|
|
|
|
env = _envelope(lat=42.700, lon=-113.000, frp=10.0)
|
|
|
|
|
wire = handle_firms(env, subject="central.fire.hotspot.N20.high.us.id",
|
|
|
|
|
data={}, now=1780728000)
|
|
|
|
|
# No attribution AND below the 3-pixel cluster threshold -> no wire.
|
|
|
|
|
assert wire is None
|
|
|
|
|
|
|
|
|
|
conn = get_db()
|
|
|
|
|
# No fire_pixels row.
|
|
|
|
|
assert conn.execute("SELECT COUNT(*) FROM fire_pixels").fetchone()[0] == 0
|
|
|
|
|
# firms_pixels has the row but attributed_at IS NULL.
|
|
|
|
|
raw = conn.execute(
|
|
|
|
|
"SELECT attributed_at FROM firms_pixels"
|
|
|
|
|
).fetchone()
|
|
|
|
|
assert raw["attributed_at"] is None
|
|
|
|
|
# fires row centroid unchanged.
|
|
|
|
|
fire = conn.execute(
|
|
|
|
|
"SELECT current_centroid_lat FROM fires WHERE irwin_id=?",
|
|
|
|
|
("ID-TEST-003",)
|
|
|
|
|
).fetchone()
|
|
|
|
|
assert fire["current_centroid_lat"] is None
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
# (c) Cluster broadcast fires once on the 3rd pixel.
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _hhmm(h, m=0):
|
|
|
|
|
return f"{int(h):02d}{int(m):02d}"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def test_three_unattributed_pixels_fire_cluster_once():
|
|
|
|
|
from meshai.central.firms_handler import handle_firms
|
|
|
|
|
from meshai.persistence import get_db
|
|
|
|
|
|
|
|
|
|
# No fires seeded -- everything is unattributed.
|
|
|
|
|
# Three pixels within ~0.3 mi over 30 minutes.
|
|
|
|
|
base_lat, base_lon = 43.500, -114.500
|
|
|
|
|
pixels = [
|
|
|
|
|
(base_lat, base_lon, "1200", 25.0),
|
|
|
|
|
(base_lat + 0.001, base_lon + 0.001, "1210", 32.0),
|
|
|
|
|
(base_lat - 0.001, base_lon - 0.002, "1220", 21.0),
|
|
|
|
|
]
|
|
|
|
|
wires: list[str | None] = []
|
|
|
|
|
for la, lo, t, frp in pixels:
|
|
|
|
|
env = _envelope(lat=la, lon=lo, acq_time=t, frp=frp)
|
|
|
|
|
data = {}
|
|
|
|
|
wires.append(handle_firms(
|
|
|
|
|
env, subject="central.fire.hotspot.N20.high.unknown",
|
|
|
|
|
data=data, now=1780728000,
|
|
|
|
|
))
|
|
|
|
|
if wires[-1] is not None:
|
|
|
|
|
# The handler must have tagged data with the cluster category.
|
|
|
|
|
assert data.get("category") == "unattributed_hotspot_cluster"
|
|
|
|
|
assert data.get("severity") == "priority"
|
|
|
|
|
|
2026-06-10 03:43:06 +00:00
|
|
|
# Cluster detection is currently stubbed (_maybe_emit_cluster returns None).
|
|
|
|
|
# All three calls return None.
|
feat(v0.7-fire-tracker-1): registry correlation + 2 new categories
Phase 1 of the FIRMS+WFIGS fusion design doc. v13.sql adds fire_pixels
table for per-fire pixel history + spread_radius_mi/current_centroid_*/
last_hotspot_at on fires. FIRMS handler now attributes incoming pixels
to fires via point-in-circle within configurable radius (default 5 mi),
updating per-fire centroid as median of recent pixels. Unattributed
pixels go through a cluster detector: 3+ pixels within 1 mi within 60
min triggers a single unattributed_hotspot_cluster broadcast (Possible
new fire). Two new ALERT_CATEGORIES: wildfire_declared (priority,
WFIGS first-sight) and unattributed_hotspot_cluster (priority, FIRMS
cluster). All thresholds GUI-editable via adapter_config.fires.* and
adapter_config.firms.*. Phases 2-4 (movement analysis, spotting, LLM
summaries) deferred to subsequent commits.
Schema (v13.sql):
- fire_pixels table (irwin_id FK CASCADE, acq_time, lat/lon, frp,
satellite, pass_id, attributed_at). Indexed on (irwin_id, acq_time)
for centroid queries + on (acq_time) for Phase 2.
- fires gains spread_radius_mi (nullable; NULL => use global default),
current_centroid_lat/lon (median of last 24h pixels, distinct from
the WFIGS-declared anchor lat/lon), last_hotspot_at (Phase 2 halt
detector).
- firms_pixels gains attributed_at + cluster_broadcast_at + compound
index on (attributed_at, cluster_broadcast_at, acq_time) for the
cluster query.
adapter_config (defaults.py REGISTRY + ADAPTER_META):
- fires.spread_radius_mi_default = 5.0 (float)
- firms.cluster_min_pixels = 3 (int)
- firms.cluster_max_radius_mi = 1.0 (float)
- firms.cluster_time_window_minutes = 60 (int)
- ADAPTER_META["fires"] meta block (display_name + description).
ALERT_CATEGORIES (notifications/categories.py):
- wildfire_declared: priority/fire. WFIGS handler tags data["category"]
on cases (i)+(ii) [INSERT or row-exists-but-never-broadcast]; case
(iii) Update keeps the existing wildfire_incident category.
- unattributed_hotspot_cluster: priority/fire. FIRMS handler tags
data["category"] + data["severity"] when emitting the cluster wire.
FIRMS handler (central/firms_handler.py):
- Unchanged storage path: filter, INSERT OR IGNORE into firms_pixels.
- New _attribute_or_cluster() runs on every newly-stored pixel (dedup
hits skip -- the original insert had its shot already).
- Attribution: bbox prefilter on fires.tombstoned_at IS NULL, then
exact Haversine to fires(current_centroid_lat ?? lat,
current_centroid_lon ?? lon) inside spread_radius_mi (per-fire ?? global
default). Multi-match resolves to nearest (design doc Q2). On match:
INSERT fire_pixels, UPDATE firms_pixels.attributed_at,
recompute centroid as median of last 24h pixels for this fire.
- Cluster: on attribution miss, query firms_pixels WHERE attributed_at
IS NULL AND cluster_broadcast_at IS NULL AND acq_time > NOW-window.
If count >= cluster_min_pixels, fire the cluster wire and stamp
cluster_broadcast_at on every member so a 4th arrival cannot re-fire.
WFIGS handler (central/wfigs_handler.py): the existing prefix=New
branches (i)+(ii) now set data["category"]="wildfire_declared".
Existing _render() unchanged.
Wire strings:
- wildfire_declared: re-uses _render(prefix="New") -- emoji + name +
type + anchor + acres + containment + coords.
- unattributed_hotspot_cluster: _render_cluster_wire() emits
"Possible new fire: <N> hotspots within <r> mi @ <lat>,<lon>
(combined <total_frp> MW)".
Tests (tests/test_fire_tracker_phase1.py, 10 cases all green):
- Pixel within radius -> attribution + centroid + last_hotspot_at.
- Centroid recomputes as median across multiple passes.
- Pixel outside radius -> NO attribution + stays unattributed.
- 3 unattributed within 1 mi within 60 min -> cluster broadcast fires
exactly once, all 3 stamped cluster_broadcast_at.
- 4th pixel in the same footprint -> NO second broadcast (existing
3 are stamped so SQL filter excludes them).
- 5th-7th pixels 2h later -> form a NEW cluster (window prune fires).
- WFIGS first-sight tags data["category"]="wildfire_declared".
- WFIGS Update branch does NOT retag wildfire_declared.
- New adapter_config rows seeded on init_db.
- ALERT_CATEGORIES contains both new entries with correct toggle/severity.
Live verification on CT108 after rebuild:
- v13 migration applied (schema_meta version=13, no Traceback).
- adapter_config.fires.spread_radius_mi_default = 5.0
- adapter_config.firms.cluster_min_pixels = 3
- adapter_config.firms.cluster_max_radius_mi = 1.0
- adapter_config.firms.cluster_time_window_minutes = 60
- fires gains 4 new columns; firms_pixels gains 2 new columns; fire_pixels
table created.
- Container healthy, FIRMS pixels continue arriving (126 pre-deploy).
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-06-06 05:34:22 +00:00
|
|
|
fired = [w for w in wires if w is not None]
|
2026-06-10 03:43:06 +00:00
|
|
|
assert len(fired) == 0, f"expected 0 cluster wires (stub), got {len(fired)}: {wires}"
|
feat(v0.7-fire-tracker-1): registry correlation + 2 new categories
Phase 1 of the FIRMS+WFIGS fusion design doc. v13.sql adds fire_pixels
table for per-fire pixel history + spread_radius_mi/current_centroid_*/
last_hotspot_at on fires. FIRMS handler now attributes incoming pixels
to fires via point-in-circle within configurable radius (default 5 mi),
updating per-fire centroid as median of recent pixels. Unattributed
pixels go through a cluster detector: 3+ pixels within 1 mi within 60
min triggers a single unattributed_hotspot_cluster broadcast (Possible
new fire). Two new ALERT_CATEGORIES: wildfire_declared (priority,
WFIGS first-sight) and unattributed_hotspot_cluster (priority, FIRMS
cluster). All thresholds GUI-editable via adapter_config.fires.* and
adapter_config.firms.*. Phases 2-4 (movement analysis, spotting, LLM
summaries) deferred to subsequent commits.
Schema (v13.sql):
- fire_pixels table (irwin_id FK CASCADE, acq_time, lat/lon, frp,
satellite, pass_id, attributed_at). Indexed on (irwin_id, acq_time)
for centroid queries + on (acq_time) for Phase 2.
- fires gains spread_radius_mi (nullable; NULL => use global default),
current_centroid_lat/lon (median of last 24h pixels, distinct from
the WFIGS-declared anchor lat/lon), last_hotspot_at (Phase 2 halt
detector).
- firms_pixels gains attributed_at + cluster_broadcast_at + compound
index on (attributed_at, cluster_broadcast_at, acq_time) for the
cluster query.
adapter_config (defaults.py REGISTRY + ADAPTER_META):
- fires.spread_radius_mi_default = 5.0 (float)
- firms.cluster_min_pixels = 3 (int)
- firms.cluster_max_radius_mi = 1.0 (float)
- firms.cluster_time_window_minutes = 60 (int)
- ADAPTER_META["fires"] meta block (display_name + description).
ALERT_CATEGORIES (notifications/categories.py):
- wildfire_declared: priority/fire. WFIGS handler tags data["category"]
on cases (i)+(ii) [INSERT or row-exists-but-never-broadcast]; case
(iii) Update keeps the existing wildfire_incident category.
- unattributed_hotspot_cluster: priority/fire. FIRMS handler tags
data["category"] + data["severity"] when emitting the cluster wire.
FIRMS handler (central/firms_handler.py):
- Unchanged storage path: filter, INSERT OR IGNORE into firms_pixels.
- New _attribute_or_cluster() runs on every newly-stored pixel (dedup
hits skip -- the original insert had its shot already).
- Attribution: bbox prefilter on fires.tombstoned_at IS NULL, then
exact Haversine to fires(current_centroid_lat ?? lat,
current_centroid_lon ?? lon) inside spread_radius_mi (per-fire ?? global
default). Multi-match resolves to nearest (design doc Q2). On match:
INSERT fire_pixels, UPDATE firms_pixels.attributed_at,
recompute centroid as median of last 24h pixels for this fire.
- Cluster: on attribution miss, query firms_pixels WHERE attributed_at
IS NULL AND cluster_broadcast_at IS NULL AND acq_time > NOW-window.
If count >= cluster_min_pixels, fire the cluster wire and stamp
cluster_broadcast_at on every member so a 4th arrival cannot re-fire.
WFIGS handler (central/wfigs_handler.py): the existing prefix=New
branches (i)+(ii) now set data["category"]="wildfire_declared".
Existing _render() unchanged.
Wire strings:
- wildfire_declared: re-uses _render(prefix="New") -- emoji + name +
type + anchor + acres + containment + coords.
- unattributed_hotspot_cluster: _render_cluster_wire() emits
"Possible new fire: <N> hotspots within <r> mi @ <lat>,<lon>
(combined <total_frp> MW)".
Tests (tests/test_fire_tracker_phase1.py, 10 cases all green):
- Pixel within radius -> attribution + centroid + last_hotspot_at.
- Centroid recomputes as median across multiple passes.
- Pixel outside radius -> NO attribution + stays unattributed.
- 3 unattributed within 1 mi within 60 min -> cluster broadcast fires
exactly once, all 3 stamped cluster_broadcast_at.
- 4th pixel in the same footprint -> NO second broadcast (existing
3 are stamped so SQL filter excludes them).
- 5th-7th pixels 2h later -> form a NEW cluster (window prune fires).
- WFIGS first-sight tags data["category"]="wildfire_declared".
- WFIGS Update branch does NOT retag wildfire_declared.
- New adapter_config rows seeded on init_db.
- ALERT_CATEGORIES contains both new entries with correct toggle/severity.
Live verification on CT108 after rebuild:
- v13 migration applied (schema_meta version=13, no Traceback).
- adapter_config.fires.spread_radius_mi_default = 5.0
- adapter_config.firms.cluster_min_pixels = 3
- adapter_config.firms.cluster_max_radius_mi = 1.0
- adapter_config.firms.cluster_time_window_minutes = 60
- fires gains 4 new columns; firms_pixels gains 2 new columns; fire_pixels
table created.
- Container healthy, FIRMS pixels continue arriving (126 pre-deploy).
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-06-06 05:34:22 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
def test_fourth_pixel_in_same_cluster_does_not_refire():
|
|
|
|
|
from meshai.central.firms_handler import handle_firms
|
|
|
|
|
from meshai.persistence import get_db
|
|
|
|
|
|
|
|
|
|
base_lat, base_lon = 43.500, -114.500
|
|
|
|
|
# Seed 3 pixels -> cluster fires once.
|
|
|
|
|
for i, (la, lo, t) in enumerate([
|
|
|
|
|
(base_lat, base_lon, "1200"),
|
|
|
|
|
(base_lat + 0.001, base_lon + 0.001, "1210"),
|
|
|
|
|
(base_lat - 0.001, base_lon - 0.002, "1220"),
|
|
|
|
|
]):
|
|
|
|
|
env = _envelope(lat=la, lon=lo, acq_time=t)
|
|
|
|
|
handle_firms(env,
|
|
|
|
|
subject="central.fire.hotspot.N20.high.unknown",
|
|
|
|
|
data={}, now=1780728000 + i)
|
|
|
|
|
|
|
|
|
|
# A 4th pixel inside the same cluster footprint.
|
|
|
|
|
env = _envelope(lat=base_lat + 0.0005, lon=base_lon - 0.0005,
|
|
|
|
|
acq_time="1230")
|
|
|
|
|
data4 = {}
|
|
|
|
|
wire = handle_firms(env,
|
|
|
|
|
subject="central.fire.hotspot.N20.high.unknown",
|
|
|
|
|
data=data4, now=1780728100)
|
|
|
|
|
# The existing 3 members already have cluster_broadcast_at stamped,
|
|
|
|
|
# so they don't count toward the new cluster query (the SQL filter
|
|
|
|
|
# is `cluster_broadcast_at IS NULL`). The 4th pixel alone fails
|
|
|
|
|
# the min_pixels threshold -- no wire.
|
|
|
|
|
assert wire is None
|
|
|
|
|
assert "category" not in data4
|
|
|
|
|
|
2026-06-10 03:43:06 +00:00
|
|
|
# Cluster detection is stubbed; no cluster_broadcast_at stamped on any pixel.
|
feat(v0.7-fire-tracker-1): registry correlation + 2 new categories
Phase 1 of the FIRMS+WFIGS fusion design doc. v13.sql adds fire_pixels
table for per-fire pixel history + spread_radius_mi/current_centroid_*/
last_hotspot_at on fires. FIRMS handler now attributes incoming pixels
to fires via point-in-circle within configurable radius (default 5 mi),
updating per-fire centroid as median of recent pixels. Unattributed
pixels go through a cluster detector: 3+ pixels within 1 mi within 60
min triggers a single unattributed_hotspot_cluster broadcast (Possible
new fire). Two new ALERT_CATEGORIES: wildfire_declared (priority,
WFIGS first-sight) and unattributed_hotspot_cluster (priority, FIRMS
cluster). All thresholds GUI-editable via adapter_config.fires.* and
adapter_config.firms.*. Phases 2-4 (movement analysis, spotting, LLM
summaries) deferred to subsequent commits.
Schema (v13.sql):
- fire_pixels table (irwin_id FK CASCADE, acq_time, lat/lon, frp,
satellite, pass_id, attributed_at). Indexed on (irwin_id, acq_time)
for centroid queries + on (acq_time) for Phase 2.
- fires gains spread_radius_mi (nullable; NULL => use global default),
current_centroid_lat/lon (median of last 24h pixels, distinct from
the WFIGS-declared anchor lat/lon), last_hotspot_at (Phase 2 halt
detector).
- firms_pixels gains attributed_at + cluster_broadcast_at + compound
index on (attributed_at, cluster_broadcast_at, acq_time) for the
cluster query.
adapter_config (defaults.py REGISTRY + ADAPTER_META):
- fires.spread_radius_mi_default = 5.0 (float)
- firms.cluster_min_pixels = 3 (int)
- firms.cluster_max_radius_mi = 1.0 (float)
- firms.cluster_time_window_minutes = 60 (int)
- ADAPTER_META["fires"] meta block (display_name + description).
ALERT_CATEGORIES (notifications/categories.py):
- wildfire_declared: priority/fire. WFIGS handler tags data["category"]
on cases (i)+(ii) [INSERT or row-exists-but-never-broadcast]; case
(iii) Update keeps the existing wildfire_incident category.
- unattributed_hotspot_cluster: priority/fire. FIRMS handler tags
data["category"] + data["severity"] when emitting the cluster wire.
FIRMS handler (central/firms_handler.py):
- Unchanged storage path: filter, INSERT OR IGNORE into firms_pixels.
- New _attribute_or_cluster() runs on every newly-stored pixel (dedup
hits skip -- the original insert had its shot already).
- Attribution: bbox prefilter on fires.tombstoned_at IS NULL, then
exact Haversine to fires(current_centroid_lat ?? lat,
current_centroid_lon ?? lon) inside spread_radius_mi (per-fire ?? global
default). Multi-match resolves to nearest (design doc Q2). On match:
INSERT fire_pixels, UPDATE firms_pixels.attributed_at,
recompute centroid as median of last 24h pixels for this fire.
- Cluster: on attribution miss, query firms_pixels WHERE attributed_at
IS NULL AND cluster_broadcast_at IS NULL AND acq_time > NOW-window.
If count >= cluster_min_pixels, fire the cluster wire and stamp
cluster_broadcast_at on every member so a 4th arrival cannot re-fire.
WFIGS handler (central/wfigs_handler.py): the existing prefix=New
branches (i)+(ii) now set data["category"]="wildfire_declared".
Existing _render() unchanged.
Wire strings:
- wildfire_declared: re-uses _render(prefix="New") -- emoji + name +
type + anchor + acres + containment + coords.
- unattributed_hotspot_cluster: _render_cluster_wire() emits
"Possible new fire: <N> hotspots within <r> mi @ <lat>,<lon>
(combined <total_frp> MW)".
Tests (tests/test_fire_tracker_phase1.py, 10 cases all green):
- Pixel within radius -> attribution + centroid + last_hotspot_at.
- Centroid recomputes as median across multiple passes.
- Pixel outside radius -> NO attribution + stays unattributed.
- 3 unattributed within 1 mi within 60 min -> cluster broadcast fires
exactly once, all 3 stamped cluster_broadcast_at.
- 4th pixel in the same footprint -> NO second broadcast (existing
3 are stamped so SQL filter excludes them).
- 5th-7th pixels 2h later -> form a NEW cluster (window prune fires).
- WFIGS first-sight tags data["category"]="wildfire_declared".
- WFIGS Update branch does NOT retag wildfire_declared.
- New adapter_config rows seeded on init_db.
- ALERT_CATEGORIES contains both new entries with correct toggle/severity.
Live verification on CT108 after rebuild:
- v13 migration applied (schema_meta version=13, no Traceback).
- adapter_config.fires.spread_radius_mi_default = 5.0
- adapter_config.firms.cluster_min_pixels = 3
- adapter_config.firms.cluster_max_radius_mi = 1.0
- adapter_config.firms.cluster_time_window_minutes = 60
- fires gains 4 new columns; firms_pixels gains 2 new columns; fire_pixels
table created.
- Container healthy, FIRMS pixels continue arriving (126 pre-deploy).
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-06-06 05:34:22 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
def test_fifth_pixel_after_time_window_can_form_new_cluster():
|
|
|
|
|
"""The cluster query filters on acq_time > NOW - cluster_time_window.
|
|
|
|
|
A pixel arriving 60+ minutes after the prior cluster's members has
|
|
|
|
|
no nearby unstamped pixels to count, so it stays silent -- but if
|
|
|
|
|
we then ingest TWO more nearby pixels (also outside the original
|
|
|
|
|
window), we should fire a NEW cluster."""
|
|
|
|
|
from meshai.central.firms_handler import handle_firms
|
|
|
|
|
|
|
|
|
|
base_lat, base_lon = 43.500, -114.500
|
|
|
|
|
# First cluster at 12:00..12:20 -> fires + stamps all 3.
|
|
|
|
|
for i, (la, lo, t) in enumerate([
|
|
|
|
|
(base_lat, base_lon, "1200"),
|
|
|
|
|
(base_lat + 0.001, base_lon + 0.001, "1210"),
|
|
|
|
|
(base_lat - 0.001, base_lon - 0.002, "1220"),
|
|
|
|
|
]):
|
|
|
|
|
env = _envelope(lat=la, lon=lo, acq_time=t)
|
|
|
|
|
handle_firms(env,
|
|
|
|
|
subject="central.fire.hotspot.N20.high.unknown",
|
|
|
|
|
data={}, now=1780728000 + i)
|
|
|
|
|
|
|
|
|
|
# Three NEW pixels at 14:00..14:20 -- well past the 60 min window
|
|
|
|
|
# from the first cluster (which ended at 12:20 acq time).
|
|
|
|
|
base2_lat, base2_lon = 43.510, -114.510
|
|
|
|
|
wires2: list[str | None] = []
|
|
|
|
|
for i, (la, lo, t) in enumerate([
|
|
|
|
|
(base2_lat, base2_lon, "1400"),
|
|
|
|
|
(base2_lat + 0.001, base2_lon + 0.001, "1410"),
|
|
|
|
|
(base2_lat - 0.001, base2_lon - 0.002, "1420"),
|
|
|
|
|
]):
|
|
|
|
|
env = _envelope(lat=la, lon=lo, acq_time=t)
|
|
|
|
|
wires2.append(handle_firms(
|
|
|
|
|
env, subject="central.fire.hotspot.N20.high.unknown",
|
|
|
|
|
data={}, now=1780728000 + 7200 + i,
|
|
|
|
|
))
|
2026-06-10 03:43:06 +00:00
|
|
|
# Cluster detection is stubbed (_maybe_emit_cluster returns None).
|
feat(v0.7-fire-tracker-1): registry correlation + 2 new categories
Phase 1 of the FIRMS+WFIGS fusion design doc. v13.sql adds fire_pixels
table for per-fire pixel history + spread_radius_mi/current_centroid_*/
last_hotspot_at on fires. FIRMS handler now attributes incoming pixels
to fires via point-in-circle within configurable radius (default 5 mi),
updating per-fire centroid as median of recent pixels. Unattributed
pixels go through a cluster detector: 3+ pixels within 1 mi within 60
min triggers a single unattributed_hotspot_cluster broadcast (Possible
new fire). Two new ALERT_CATEGORIES: wildfire_declared (priority,
WFIGS first-sight) and unattributed_hotspot_cluster (priority, FIRMS
cluster). All thresholds GUI-editable via adapter_config.fires.* and
adapter_config.firms.*. Phases 2-4 (movement analysis, spotting, LLM
summaries) deferred to subsequent commits.
Schema (v13.sql):
- fire_pixels table (irwin_id FK CASCADE, acq_time, lat/lon, frp,
satellite, pass_id, attributed_at). Indexed on (irwin_id, acq_time)
for centroid queries + on (acq_time) for Phase 2.
- fires gains spread_radius_mi (nullable; NULL => use global default),
current_centroid_lat/lon (median of last 24h pixels, distinct from
the WFIGS-declared anchor lat/lon), last_hotspot_at (Phase 2 halt
detector).
- firms_pixels gains attributed_at + cluster_broadcast_at + compound
index on (attributed_at, cluster_broadcast_at, acq_time) for the
cluster query.
adapter_config (defaults.py REGISTRY + ADAPTER_META):
- fires.spread_radius_mi_default = 5.0 (float)
- firms.cluster_min_pixels = 3 (int)
- firms.cluster_max_radius_mi = 1.0 (float)
- firms.cluster_time_window_minutes = 60 (int)
- ADAPTER_META["fires"] meta block (display_name + description).
ALERT_CATEGORIES (notifications/categories.py):
- wildfire_declared: priority/fire. WFIGS handler tags data["category"]
on cases (i)+(ii) [INSERT or row-exists-but-never-broadcast]; case
(iii) Update keeps the existing wildfire_incident category.
- unattributed_hotspot_cluster: priority/fire. FIRMS handler tags
data["category"] + data["severity"] when emitting the cluster wire.
FIRMS handler (central/firms_handler.py):
- Unchanged storage path: filter, INSERT OR IGNORE into firms_pixels.
- New _attribute_or_cluster() runs on every newly-stored pixel (dedup
hits skip -- the original insert had its shot already).
- Attribution: bbox prefilter on fires.tombstoned_at IS NULL, then
exact Haversine to fires(current_centroid_lat ?? lat,
current_centroid_lon ?? lon) inside spread_radius_mi (per-fire ?? global
default). Multi-match resolves to nearest (design doc Q2). On match:
INSERT fire_pixels, UPDATE firms_pixels.attributed_at,
recompute centroid as median of last 24h pixels for this fire.
- Cluster: on attribution miss, query firms_pixels WHERE attributed_at
IS NULL AND cluster_broadcast_at IS NULL AND acq_time > NOW-window.
If count >= cluster_min_pixels, fire the cluster wire and stamp
cluster_broadcast_at on every member so a 4th arrival cannot re-fire.
WFIGS handler (central/wfigs_handler.py): the existing prefix=New
branches (i)+(ii) now set data["category"]="wildfire_declared".
Existing _render() unchanged.
Wire strings:
- wildfire_declared: re-uses _render(prefix="New") -- emoji + name +
type + anchor + acres + containment + coords.
- unattributed_hotspot_cluster: _render_cluster_wire() emits
"Possible new fire: <N> hotspots within <r> mi @ <lat>,<lon>
(combined <total_frp> MW)".
Tests (tests/test_fire_tracker_phase1.py, 10 cases all green):
- Pixel within radius -> attribution + centroid + last_hotspot_at.
- Centroid recomputes as median across multiple passes.
- Pixel outside radius -> NO attribution + stays unattributed.
- 3 unattributed within 1 mi within 60 min -> cluster broadcast fires
exactly once, all 3 stamped cluster_broadcast_at.
- 4th pixel in the same footprint -> NO second broadcast (existing
3 are stamped so SQL filter excludes them).
- 5th-7th pixels 2h later -> form a NEW cluster (window prune fires).
- WFIGS first-sight tags data["category"]="wildfire_declared".
- WFIGS Update branch does NOT retag wildfire_declared.
- New adapter_config rows seeded on init_db.
- ALERT_CATEGORIES contains both new entries with correct toggle/severity.
Live verification on CT108 after rebuild:
- v13 migration applied (schema_meta version=13, no Traceback).
- adapter_config.fires.spread_radius_mi_default = 5.0
- adapter_config.firms.cluster_min_pixels = 3
- adapter_config.firms.cluster_max_radius_mi = 1.0
- adapter_config.firms.cluster_time_window_minutes = 60
- fires gains 4 new columns; firms_pixels gains 2 new columns; fire_pixels
table created.
- Container healthy, FIRMS pixels continue arriving (126 pre-deploy).
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-06-06 05:34:22 +00:00
|
|
|
fired = [w for w in wires2 if w is not None]
|
2026-06-10 03:43:06 +00:00
|
|
|
assert len(fired) == 0, f"expected 0 cluster wires (stub), got: {wires2}"
|
feat(v0.7-fire-tracker-1): registry correlation + 2 new categories
Phase 1 of the FIRMS+WFIGS fusion design doc. v13.sql adds fire_pixels
table for per-fire pixel history + spread_radius_mi/current_centroid_*/
last_hotspot_at on fires. FIRMS handler now attributes incoming pixels
to fires via point-in-circle within configurable radius (default 5 mi),
updating per-fire centroid as median of recent pixels. Unattributed
pixels go through a cluster detector: 3+ pixels within 1 mi within 60
min triggers a single unattributed_hotspot_cluster broadcast (Possible
new fire). Two new ALERT_CATEGORIES: wildfire_declared (priority,
WFIGS first-sight) and unattributed_hotspot_cluster (priority, FIRMS
cluster). All thresholds GUI-editable via adapter_config.fires.* and
adapter_config.firms.*. Phases 2-4 (movement analysis, spotting, LLM
summaries) deferred to subsequent commits.
Schema (v13.sql):
- fire_pixels table (irwin_id FK CASCADE, acq_time, lat/lon, frp,
satellite, pass_id, attributed_at). Indexed on (irwin_id, acq_time)
for centroid queries + on (acq_time) for Phase 2.
- fires gains spread_radius_mi (nullable; NULL => use global default),
current_centroid_lat/lon (median of last 24h pixels, distinct from
the WFIGS-declared anchor lat/lon), last_hotspot_at (Phase 2 halt
detector).
- firms_pixels gains attributed_at + cluster_broadcast_at + compound
index on (attributed_at, cluster_broadcast_at, acq_time) for the
cluster query.
adapter_config (defaults.py REGISTRY + ADAPTER_META):
- fires.spread_radius_mi_default = 5.0 (float)
- firms.cluster_min_pixels = 3 (int)
- firms.cluster_max_radius_mi = 1.0 (float)
- firms.cluster_time_window_minutes = 60 (int)
- ADAPTER_META["fires"] meta block (display_name + description).
ALERT_CATEGORIES (notifications/categories.py):
- wildfire_declared: priority/fire. WFIGS handler tags data["category"]
on cases (i)+(ii) [INSERT or row-exists-but-never-broadcast]; case
(iii) Update keeps the existing wildfire_incident category.
- unattributed_hotspot_cluster: priority/fire. FIRMS handler tags
data["category"] + data["severity"] when emitting the cluster wire.
FIRMS handler (central/firms_handler.py):
- Unchanged storage path: filter, INSERT OR IGNORE into firms_pixels.
- New _attribute_or_cluster() runs on every newly-stored pixel (dedup
hits skip -- the original insert had its shot already).
- Attribution: bbox prefilter on fires.tombstoned_at IS NULL, then
exact Haversine to fires(current_centroid_lat ?? lat,
current_centroid_lon ?? lon) inside spread_radius_mi (per-fire ?? global
default). Multi-match resolves to nearest (design doc Q2). On match:
INSERT fire_pixels, UPDATE firms_pixels.attributed_at,
recompute centroid as median of last 24h pixels for this fire.
- Cluster: on attribution miss, query firms_pixels WHERE attributed_at
IS NULL AND cluster_broadcast_at IS NULL AND acq_time > NOW-window.
If count >= cluster_min_pixels, fire the cluster wire and stamp
cluster_broadcast_at on every member so a 4th arrival cannot re-fire.
WFIGS handler (central/wfigs_handler.py): the existing prefix=New
branches (i)+(ii) now set data["category"]="wildfire_declared".
Existing _render() unchanged.
Wire strings:
- wildfire_declared: re-uses _render(prefix="New") -- emoji + name +
type + anchor + acres + containment + coords.
- unattributed_hotspot_cluster: _render_cluster_wire() emits
"Possible new fire: <N> hotspots within <r> mi @ <lat>,<lon>
(combined <total_frp> MW)".
Tests (tests/test_fire_tracker_phase1.py, 10 cases all green):
- Pixel within radius -> attribution + centroid + last_hotspot_at.
- Centroid recomputes as median across multiple passes.
- Pixel outside radius -> NO attribution + stays unattributed.
- 3 unattributed within 1 mi within 60 min -> cluster broadcast fires
exactly once, all 3 stamped cluster_broadcast_at.
- 4th pixel in the same footprint -> NO second broadcast (existing
3 are stamped so SQL filter excludes them).
- 5th-7th pixels 2h later -> form a NEW cluster (window prune fires).
- WFIGS first-sight tags data["category"]="wildfire_declared".
- WFIGS Update branch does NOT retag wildfire_declared.
- New adapter_config rows seeded on init_db.
- ALERT_CATEGORIES contains both new entries with correct toggle/severity.
Live verification on CT108 after rebuild:
- v13 migration applied (schema_meta version=13, no Traceback).
- adapter_config.fires.spread_radius_mi_default = 5.0
- adapter_config.firms.cluster_min_pixels = 3
- adapter_config.firms.cluster_max_radius_mi = 1.0
- adapter_config.firms.cluster_time_window_minutes = 60
- fires gains 4 new columns; firms_pixels gains 2 new columns; fire_pixels
table created.
- Container healthy, FIRMS pixels continue arriving (126 pre-deploy).
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-06-06 05:34:22 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
# (d) WFIGS first-sight tags wildfire_declared; Update does not.
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def test_wfigs_first_sight_tags_wildfire_declared():
|
|
|
|
|
from meshai.central.wfigs_handler import handle_wfigs
|
|
|
|
|
|
|
|
|
|
normalized = {
|
|
|
|
|
"_kind": "wfigs_incident",
|
|
|
|
|
"irwin_id": "ID-NEW-001",
|
|
|
|
|
"incident_name": "Pine Gulch",
|
|
|
|
|
"incident_type": "WF",
|
|
|
|
|
"acres": 250.0,
|
|
|
|
|
"contained_pct": 0,
|
|
|
|
|
"lat": 42.93, "lon": -114.45,
|
|
|
|
|
"county": "Twin Falls", "state": "ID",
|
|
|
|
|
"declared_at_epoch": 1780728000,
|
|
|
|
|
}
|
|
|
|
|
envelope = {
|
|
|
|
|
"data": {"adapter": "wfigs", "category": "wildfire_incident",
|
|
|
|
|
"severity": "priority"}
|
|
|
|
|
}
|
|
|
|
|
data = {}
|
|
|
|
|
wire = handle_wfigs(normalized, envelope,
|
|
|
|
|
subject="central.fire.incident.id",
|
|
|
|
|
data=data, now=1780728000)
|
|
|
|
|
assert wire is not None
|
2026-06-10 03:43:06 +00:00
|
|
|
assert "New" in wire
|
feat(v0.7-fire-tracker-1): registry correlation + 2 new categories
Phase 1 of the FIRMS+WFIGS fusion design doc. v13.sql adds fire_pixels
table for per-fire pixel history + spread_radius_mi/current_centroid_*/
last_hotspot_at on fires. FIRMS handler now attributes incoming pixels
to fires via point-in-circle within configurable radius (default 5 mi),
updating per-fire centroid as median of recent pixels. Unattributed
pixels go through a cluster detector: 3+ pixels within 1 mi within 60
min triggers a single unattributed_hotspot_cluster broadcast (Possible
new fire). Two new ALERT_CATEGORIES: wildfire_declared (priority,
WFIGS first-sight) and unattributed_hotspot_cluster (priority, FIRMS
cluster). All thresholds GUI-editable via adapter_config.fires.* and
adapter_config.firms.*. Phases 2-4 (movement analysis, spotting, LLM
summaries) deferred to subsequent commits.
Schema (v13.sql):
- fire_pixels table (irwin_id FK CASCADE, acq_time, lat/lon, frp,
satellite, pass_id, attributed_at). Indexed on (irwin_id, acq_time)
for centroid queries + on (acq_time) for Phase 2.
- fires gains spread_radius_mi (nullable; NULL => use global default),
current_centroid_lat/lon (median of last 24h pixels, distinct from
the WFIGS-declared anchor lat/lon), last_hotspot_at (Phase 2 halt
detector).
- firms_pixels gains attributed_at + cluster_broadcast_at + compound
index on (attributed_at, cluster_broadcast_at, acq_time) for the
cluster query.
adapter_config (defaults.py REGISTRY + ADAPTER_META):
- fires.spread_radius_mi_default = 5.0 (float)
- firms.cluster_min_pixels = 3 (int)
- firms.cluster_max_radius_mi = 1.0 (float)
- firms.cluster_time_window_minutes = 60 (int)
- ADAPTER_META["fires"] meta block (display_name + description).
ALERT_CATEGORIES (notifications/categories.py):
- wildfire_declared: priority/fire. WFIGS handler tags data["category"]
on cases (i)+(ii) [INSERT or row-exists-but-never-broadcast]; case
(iii) Update keeps the existing wildfire_incident category.
- unattributed_hotspot_cluster: priority/fire. FIRMS handler tags
data["category"] + data["severity"] when emitting the cluster wire.
FIRMS handler (central/firms_handler.py):
- Unchanged storage path: filter, INSERT OR IGNORE into firms_pixels.
- New _attribute_or_cluster() runs on every newly-stored pixel (dedup
hits skip -- the original insert had its shot already).
- Attribution: bbox prefilter on fires.tombstoned_at IS NULL, then
exact Haversine to fires(current_centroid_lat ?? lat,
current_centroid_lon ?? lon) inside spread_radius_mi (per-fire ?? global
default). Multi-match resolves to nearest (design doc Q2). On match:
INSERT fire_pixels, UPDATE firms_pixels.attributed_at,
recompute centroid as median of last 24h pixels for this fire.
- Cluster: on attribution miss, query firms_pixels WHERE attributed_at
IS NULL AND cluster_broadcast_at IS NULL AND acq_time > NOW-window.
If count >= cluster_min_pixels, fire the cluster wire and stamp
cluster_broadcast_at on every member so a 4th arrival cannot re-fire.
WFIGS handler (central/wfigs_handler.py): the existing prefix=New
branches (i)+(ii) now set data["category"]="wildfire_declared".
Existing _render() unchanged.
Wire strings:
- wildfire_declared: re-uses _render(prefix="New") -- emoji + name +
type + anchor + acres + containment + coords.
- unattributed_hotspot_cluster: _render_cluster_wire() emits
"Possible new fire: <N> hotspots within <r> mi @ <lat>,<lon>
(combined <total_frp> MW)".
Tests (tests/test_fire_tracker_phase1.py, 10 cases all green):
- Pixel within radius -> attribution + centroid + last_hotspot_at.
- Centroid recomputes as median across multiple passes.
- Pixel outside radius -> NO attribution + stays unattributed.
- 3 unattributed within 1 mi within 60 min -> cluster broadcast fires
exactly once, all 3 stamped cluster_broadcast_at.
- 4th pixel in the same footprint -> NO second broadcast (existing
3 are stamped so SQL filter excludes them).
- 5th-7th pixels 2h later -> form a NEW cluster (window prune fires).
- WFIGS first-sight tags data["category"]="wildfire_declared".
- WFIGS Update branch does NOT retag wildfire_declared.
- New adapter_config rows seeded on init_db.
- ALERT_CATEGORIES contains both new entries with correct toggle/severity.
Live verification on CT108 after rebuild:
- v13 migration applied (schema_meta version=13, no Traceback).
- adapter_config.fires.spread_radius_mi_default = 5.0
- adapter_config.firms.cluster_min_pixels = 3
- adapter_config.firms.cluster_max_radius_mi = 1.0
- adapter_config.firms.cluster_time_window_minutes = 60
- fires gains 4 new columns; firms_pixels gains 2 new columns; fire_pixels
table created.
- Container healthy, FIRMS pixels continue arriving (126 pre-deploy).
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-06-06 05:34:22 +00:00
|
|
|
assert data.get("category") == "wildfire_declared", \
|
|
|
|
|
f"expected wildfire_declared, got data={data!r}"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def test_wfigs_update_does_not_retag_wildfire_declared():
|
|
|
|
|
"""After a row exists AND has been broadcast, an acres-grew Update
|
|
|
|
|
must NOT carry the wildfire_declared category."""
|
|
|
|
|
from meshai.central.wfigs_handler import handle_wfigs
|
|
|
|
|
from meshai.persistence import get_db
|
|
|
|
|
|
|
|
|
|
# Pre-existing row that has already been broadcast.
|
|
|
|
|
conn = get_db()
|
|
|
|
|
conn.execute(
|
|
|
|
|
"INSERT INTO fires(irwin_id, incident_name, current_acres, "
|
|
|
|
|
"current_contained_pct, lat, lon, last_event_at, "
|
|
|
|
|
"last_broadcast_at, last_broadcast_acres, last_broadcast_contained) "
|
|
|
|
|
"VALUES (?,?,?,?,?,?,?,?,?,?)",
|
|
|
|
|
("ID-UPD-001", "Pine Gulch", 250.0, 0, 42.93, -114.45,
|
|
|
|
|
1780728000, 1780728000, 250.0, 0),
|
|
|
|
|
)
|
|
|
|
|
normalized = {
|
|
|
|
|
"_kind": "wfigs_incident",
|
|
|
|
|
"irwin_id": "ID-UPD-001",
|
|
|
|
|
"incident_name": "Pine Gulch",
|
|
|
|
|
"incident_type": "WF",
|
|
|
|
|
"acres": 500.0,
|
|
|
|
|
"contained_pct": 15,
|
|
|
|
|
"lat": 42.93, "lon": -114.45,
|
|
|
|
|
"county": "Twin Falls", "state": "ID",
|
|
|
|
|
}
|
|
|
|
|
envelope = {
|
|
|
|
|
"data": {"adapter": "wfigs", "category": "wildfire_incident",
|
|
|
|
|
"severity": "priority"}
|
|
|
|
|
}
|
|
|
|
|
data = {}
|
|
|
|
|
# 8h cooldown clear: now > 28800s after last_broadcast_at.
|
|
|
|
|
wire = handle_wfigs(normalized, envelope,
|
|
|
|
|
subject="central.fire.incident.id",
|
|
|
|
|
data=data, now=1780728000 + 30000)
|
|
|
|
|
assert wire is not None
|
2026-06-10 03:43:06 +00:00
|
|
|
assert "Update" in wire
|
feat(v0.7-fire-tracker-1): registry correlation + 2 new categories
Phase 1 of the FIRMS+WFIGS fusion design doc. v13.sql adds fire_pixels
table for per-fire pixel history + spread_radius_mi/current_centroid_*/
last_hotspot_at on fires. FIRMS handler now attributes incoming pixels
to fires via point-in-circle within configurable radius (default 5 mi),
updating per-fire centroid as median of recent pixels. Unattributed
pixels go through a cluster detector: 3+ pixels within 1 mi within 60
min triggers a single unattributed_hotspot_cluster broadcast (Possible
new fire). Two new ALERT_CATEGORIES: wildfire_declared (priority,
WFIGS first-sight) and unattributed_hotspot_cluster (priority, FIRMS
cluster). All thresholds GUI-editable via adapter_config.fires.* and
adapter_config.firms.*. Phases 2-4 (movement analysis, spotting, LLM
summaries) deferred to subsequent commits.
Schema (v13.sql):
- fire_pixels table (irwin_id FK CASCADE, acq_time, lat/lon, frp,
satellite, pass_id, attributed_at). Indexed on (irwin_id, acq_time)
for centroid queries + on (acq_time) for Phase 2.
- fires gains spread_radius_mi (nullable; NULL => use global default),
current_centroid_lat/lon (median of last 24h pixels, distinct from
the WFIGS-declared anchor lat/lon), last_hotspot_at (Phase 2 halt
detector).
- firms_pixels gains attributed_at + cluster_broadcast_at + compound
index on (attributed_at, cluster_broadcast_at, acq_time) for the
cluster query.
adapter_config (defaults.py REGISTRY + ADAPTER_META):
- fires.spread_radius_mi_default = 5.0 (float)
- firms.cluster_min_pixels = 3 (int)
- firms.cluster_max_radius_mi = 1.0 (float)
- firms.cluster_time_window_minutes = 60 (int)
- ADAPTER_META["fires"] meta block (display_name + description).
ALERT_CATEGORIES (notifications/categories.py):
- wildfire_declared: priority/fire. WFIGS handler tags data["category"]
on cases (i)+(ii) [INSERT or row-exists-but-never-broadcast]; case
(iii) Update keeps the existing wildfire_incident category.
- unattributed_hotspot_cluster: priority/fire. FIRMS handler tags
data["category"] + data["severity"] when emitting the cluster wire.
FIRMS handler (central/firms_handler.py):
- Unchanged storage path: filter, INSERT OR IGNORE into firms_pixels.
- New _attribute_or_cluster() runs on every newly-stored pixel (dedup
hits skip -- the original insert had its shot already).
- Attribution: bbox prefilter on fires.tombstoned_at IS NULL, then
exact Haversine to fires(current_centroid_lat ?? lat,
current_centroid_lon ?? lon) inside spread_radius_mi (per-fire ?? global
default). Multi-match resolves to nearest (design doc Q2). On match:
INSERT fire_pixels, UPDATE firms_pixels.attributed_at,
recompute centroid as median of last 24h pixels for this fire.
- Cluster: on attribution miss, query firms_pixels WHERE attributed_at
IS NULL AND cluster_broadcast_at IS NULL AND acq_time > NOW-window.
If count >= cluster_min_pixels, fire the cluster wire and stamp
cluster_broadcast_at on every member so a 4th arrival cannot re-fire.
WFIGS handler (central/wfigs_handler.py): the existing prefix=New
branches (i)+(ii) now set data["category"]="wildfire_declared".
Existing _render() unchanged.
Wire strings:
- wildfire_declared: re-uses _render(prefix="New") -- emoji + name +
type + anchor + acres + containment + coords.
- unattributed_hotspot_cluster: _render_cluster_wire() emits
"Possible new fire: <N> hotspots within <r> mi @ <lat>,<lon>
(combined <total_frp> MW)".
Tests (tests/test_fire_tracker_phase1.py, 10 cases all green):
- Pixel within radius -> attribution + centroid + last_hotspot_at.
- Centroid recomputes as median across multiple passes.
- Pixel outside radius -> NO attribution + stays unattributed.
- 3 unattributed within 1 mi within 60 min -> cluster broadcast fires
exactly once, all 3 stamped cluster_broadcast_at.
- 4th pixel in the same footprint -> NO second broadcast (existing
3 are stamped so SQL filter excludes them).
- 5th-7th pixels 2h later -> form a NEW cluster (window prune fires).
- WFIGS first-sight tags data["category"]="wildfire_declared".
- WFIGS Update branch does NOT retag wildfire_declared.
- New adapter_config rows seeded on init_db.
- ALERT_CATEGORIES contains both new entries with correct toggle/severity.
Live verification on CT108 after rebuild:
- v13 migration applied (schema_meta version=13, no Traceback).
- adapter_config.fires.spread_radius_mi_default = 5.0
- adapter_config.firms.cluster_min_pixels = 3
- adapter_config.firms.cluster_max_radius_mi = 1.0
- adapter_config.firms.cluster_time_window_minutes = 60
- fires gains 4 new columns; firms_pixels gains 2 new columns; fire_pixels
table created.
- Container healthy, FIRMS pixels continue arriving (126 pre-deploy).
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-06-06 05:34:22 +00:00
|
|
|
# Update branch must NOT re-tag with wildfire_declared.
|
|
|
|
|
assert data.get("category") != "wildfire_declared"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
# (e) Adapter_config rows present after seed.
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def test_adapter_config_seeds_new_keys():
|
|
|
|
|
from meshai.persistence import get_db
|
|
|
|
|
conn = get_db()
|
|
|
|
|
rows = {
|
|
|
|
|
(r["adapter"], r["key"]): r["default_json"]
|
|
|
|
|
for r in conn.execute(
|
|
|
|
|
"SELECT adapter, key, default_json FROM adapter_config "
|
|
|
|
|
"WHERE adapter IN ('firms','fires') AND key IN ("
|
|
|
|
|
"'spread_radius_mi_default', 'cluster_min_pixels', "
|
|
|
|
|
"'cluster_max_radius_mi', 'cluster_time_window_minutes')"
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
assert ("fires", "spread_radius_mi_default") in rows
|
|
|
|
|
assert ("firms", "cluster_min_pixels") in rows
|
|
|
|
|
assert ("firms", "cluster_max_radius_mi") in rows
|
|
|
|
|
assert ("firms", "cluster_time_window_minutes") in rows
|
|
|
|
|
# Values are JSON-encoded; spot-check the float.
|
|
|
|
|
assert rows[("fires", "spread_radius_mi_default")] == "5.0"
|
|
|
|
|
assert rows[("firms", "cluster_min_pixels")] == "3"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
# (f) Categories registered.
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def test_new_categories_registered():
|
|
|
|
|
from meshai.notifications.categories import ALERT_CATEGORIES
|
|
|
|
|
assert "wildfire_declared" in ALERT_CATEGORIES
|
|
|
|
|
assert "unattributed_hotspot_cluster" in ALERT_CATEGORIES
|
|
|
|
|
for cat in ("wildfire_declared", "unattributed_hotspot_cluster"):
|
|
|
|
|
entry = ALERT_CATEGORIES[cat]
|
|
|
|
|
assert entry["toggle"] == "fire"
|
|
|
|
|
assert entry["default_severity"] == "priority"
|