Files
DECNET/tests/profiler/behave_shell/test_extract_smoke.py
anti 879f5e731b feat(profiler/behave_shell): emit motor.input_modality
BEHAVE-EXTRACTOR.md Phase A Step 2. The first primitive — picked
first because it has the highest discriminative value (HUMAN vs
everyone) and the simplest implementation (paste-event ratio over
total inputs).

* _features/motor.py:input_modality(ctx) emits one Observation
  per session in {typed, pasted, mixed} with confidence 0.75 / 0.70.
* _features/_emit.py centralises the make_observation helper so
  every feature module gets the same Window/source/evidence_ref
  boilerplate without copy-paste.
* Thresholds inherited from the prototype's calibration history
  (MODALITY_PASTED_MIN=0.40, MODALITY_TYPED_MAX=0.05).
* Zero-input session skips emission — registry doesn't admit
  "unknown" here.

Tests: pure-typed → typed, pure-pasted → pasted, mixed → mixed,
output-only session → no observation, full envelope round-trip.
2026-05-03 07:47:38 -04:00

97 lines
3.2 KiB
Python

"""Step 0 smoke: prove the wiring before any logic.
Before any feature function lands, verify:
* the public ``extract_session`` import path resolves;
* an empty event stream yields zero observations and a well-formed
zero-duration ``SessionContext``;
* a single input event yields a context with ``t_start == t_end``
and ``duration_s == 0.0``;
* a multi-event stream populates ``t_start`` / ``t_end`` /
``duration_s`` correctly and routes events into the
``input_events`` / ``output_events`` slots by kind;
* ``FEATURES`` is empty at Step 0 — the empty contract is the gate
that the next step must intentionally break.
"""
from __future__ import annotations
from decnet.profiler.behave_shell import (
DEFAULT_SOURCE,
build_context,
extract_session,
)
from decnet.profiler.behave_shell._features import FEATURES
from decnet.profiler.behave_shell._parse import AsciinemaEvent
def test_features_tuple_is_populated() -> None:
# Step 2+: at least one feature is registered. Exact membership is
# asserted in per-feature tests; this test only pins "the registry
# is non-empty" so the empty-FEATURES regression doesn't sneak back.
assert len(FEATURES) >= 1
def test_default_source_is_canonical_path() -> None:
assert DEFAULT_SOURCE == "decnet/profiler/behave_shell/extract.py"
def test_extract_session_empty_stream_yields_no_observations() -> None:
out = list(extract_session([], sid="sess-empty"))
assert out == []
def test_build_context_empty_stream_zero_duration() -> None:
ctx = build_context([], sid="sess-empty")
assert ctx.sid == "sess-empty"
assert ctx.source == DEFAULT_SOURCE
assert ctx.evidence_ref == "session:sess-empty"
assert ctx.t_start == 0.0
assert ctx.t_end == 0.0
assert ctx.duration_s == 0.0
assert ctx.input_events == ()
assert ctx.output_events == ()
def test_build_context_single_input_event() -> None:
events: list[AsciinemaEvent] = [(1.5, "i", "a")]
ctx = build_context(events, sid="sess-1")
assert ctx.t_start == 1.5
assert ctx.t_end == 1.5
assert ctx.duration_s == 0.0
assert ctx.input_events == ((1.5, "i", "a"),)
assert ctx.output_events == ()
def test_build_context_multi_event_routes_by_kind() -> None:
events: list[AsciinemaEvent] = [
(0.0, "i", "l"),
(0.1, "i", "s"),
(0.2, "o", "ls\r\n"),
(0.3, "o", "file.txt\r\n"),
(0.5, "i", "\r"),
]
ctx = build_context(events, sid="sess-multi")
assert ctx.t_start == 0.0
assert ctx.t_end == 0.5
assert ctx.duration_s == 0.5
assert len(ctx.input_events) == 3
assert len(ctx.output_events) == 2
# Order preserved
assert ctx.input_events[0] == (0.0, "i", "l")
assert ctx.output_events[-1] == (0.3, "o", "file.txt\r\n")
def test_extract_session_explicit_evidence_ref_overrides_default() -> None:
ctx = build_context(
[(0.0, "i", "x")],
sid="sess-x",
evidence_ref="shard:/var/log/d/sess-x.cast",
)
assert ctx.evidence_ref == "shard:/var/log/d/sess-x.cast"
def test_extract_session_zero_inputs_yields_nothing() -> None:
"""No input events → no feature emits (input_modality skips on empty)."""
out = list(extract_session([(0.0, "o", "hi\r\n")], sid="sess-no-input"))
assert out == []