shinkan-jinkendo/backend/tests/test_planning_exercise_path_ai_fill.py
Lars dd0fae4bf5
Some checks failed
Deploy Development / deploy (push) Successful in 49s
Test Suite / pytest-backend (push) Failing after 43s
Test Suite / lint-backend (push) Successful in 0s
Test Suite / build-frontend (push) Successful in 15s
Test Suite / k6 /health Baseline (push) Successful in 44s
Test Suite / playwright-tests (push) Successful in 1m15s
Enhance Planning AI with Roadmap-First Architecture and New Features
- Introduced a roadmap-first approach for the planning AI, allowing for a structured progression graph that aligns with the overall project roadmap.
- Added new functionality to strip off-topic steps from exercise paths, improving the relevance of generated exercise suggestions.
- Implemented a detailed goal text generation for AI proposals, enhancing the context provided for new exercises.
- Updated the ExerciseProgressionPathBuilder component to support new features, including roadmap previews and improved focus area handling.
- Incremented application version to 0.8.205 and updated database schema version to 20260606086 to reflect these changes.
2026-06-08 08:10:53 +02:00

94 lines
3.2 KiB
Python

"""Tests Planungs-KI Phase E3 — Lücken-Angebote und Off-Topic."""
from planning_exercise_path_ai_fill import build_gap_fill_goal_text, collect_gap_fill_specs
from planning_exercise_path_qa import parse_llm_suggested_new_exercises, strip_off_topic_steps_from_path
from planning_exercise_semantics import build_semantic_brief
def test_parse_llm_suggested_new_exercises():
brief = build_semantic_brief("Mae Geri Perfektion")
llm_qa = {
"suggested_new_exercises": [
{
"title_hint": "Mae Geri Kraft am Sandsack",
"sketch": "Kraft und Schnelligkeit",
"phase": "vertiefung",
"insert_after_step_index": 1,
"rationale": "Zwischenschritt",
}
]
}
specs = parse_llm_suggested_new_exercises(llm_qa, brief=brief, step_count=5)
assert len(specs) == 1
assert specs[0]["insert_after_index"] == 1
assert "Mae Geri" in specs[0]["title_hint"]
def test_collect_gap_fill_specs_off_topic_and_unfilled():
brief = build_semantic_brief("Mae Geri Perfektion")
steps = [
{"exercise_id": 1, "title": "Mae Geri Kihon"},
{"exercise_id": 2, "title": "Präzision"},
{"exercise_id": 3, "title": "One Leg Squat"},
{"exercise_id": 4, "title": "Gleichgewichtstritt"},
]
unfilled = [
{
"from_exercise_id": 2,
"to_exercise_id": 3,
"expected_phase": "vertiefung",
"from_title": "Präzision",
"to_title": "One Leg Squat",
}
]
off_topic = [
{
"step_index": 2,
"exercise_id": 3,
"title": "One Leg Squat",
"expected_phase": "vertiefung",
}
]
specs = collect_gap_fill_specs(
steps=steps,
unfilled_gaps=unfilled,
off_topic_steps=off_topic,
llm_specs=[],
brief=brief,
goal_query="Mae Geri Perfektion",
)
sources = {s["source"] for s in specs}
assert "unfilled_gap" in sources
assert "off_topic" in sources
off = next(s for s in specs if s["source"] == "off_topic")
assert off["replace_step_index"] == 2
assert off["insert_after_index"] == 1
def test_strip_off_topic_steps_from_path():
steps = [
{"exercise_id": 1, "title": "A"},
{"exercise_id": 2, "title": "B"},
{"exercise_id": 3, "title": "One Leg Squat"},
{"exercise_id": 4, "title": "D"},
]
off_topic = [{"step_index": 2, "title": "One Leg Squat", "exercise_id": 3}]
out, removed = strip_off_topic_steps_from_path(steps, off_topic)
assert len(out) == 3
assert len(removed) == 1
assert removed[0]["removed_title"] == "One Leg Squat"
assert [s["exercise_id"] for s in out] == [1, 2, 4]
def test_build_gap_fill_goal_text_includes_topic():
brief = build_semantic_brief("Mae Geri Perfektion")
text = build_gap_fill_goal_text(
goal_query="Mae Geri Perfektion",
brief=brief,
spec={"phase": "anwendung", "rationale": "Fehlt Kombinationstraining"},
step_a={"title": "Kihon"},
step_b={"title": "Kumite"},
)
assert "Mae Geri" in text or "mae geri" in text.lower()
assert "anwendung" in text
assert "Kihon" in text