fix(rag): stop live_gui tests from polluting session-scoped subprocess
Per Tier 1 investigation (docs/reports/INVESTIGATION_rag_phase4_final_verify_20260627.md), two live_gui tests were leaking temp/relative paths into the shared subprocess's ui_files_base_dir, which survived across @clean_baseline tests and caused RAGEngine.index_file to silently no-op on a dead base_dir. Three fixes: 1. tests/test_rag_visual_sim.py: stop using tempfile.mkdtemp() (which defaults to C:\Users\Ed\AppData\Local\Temp\tmpXXXX) and instead use tempfile.mkdtemp(dir="tests/artifacts", ...). Also restore files_base_dir and rag_enabled in finally so the next live_gui test in the session doesn't inherit the dead path. 2. tests/test_visual_sim_mma_v2.py: stop changing files_base_dir to 'tests/artifacts/temp_workspace' and stop clicking btn_project_save (which persisted the path to manual_slop.toml). The MMA lifecycle does not depend on a specific files_base_dir. 3. src/app_controller.py _handle_reset_session: defensive fix that resets ui_files_base_dir from the default project's base_dir. This makes reset_session() robust to any future polluter (not just the two known ones). Without this, a test that sets files_base_dir via set_value leaves a dead path in the session-scoped subprocess even after reset_session(). Verified: tests/test_rag_visual_sim.py passes 2/2 after the fix.
This commit is contained in:
@@ -3893,6 +3893,14 @@ class AppController:
|
|||||||
self.ui_ai_input = ""
|
self.ui_ai_input = ""
|
||||||
self.ui_manual_approve = False
|
self.ui_manual_approve = False
|
||||||
self.ui_auto_add_history = False
|
self.ui_auto_add_history = False
|
||||||
|
# Reset ui_files_base_dir to a sane default so a prior live_gui test
|
||||||
|
# that set a temp/relative path via set_value('files_base_dir', ...)
|
||||||
|
# does not pollute the session-scoped subprocess with a dead path.
|
||||||
|
# See tests/test_rag_visual_sim.py and tests/test_visual_sim_mma_v2.py
|
||||||
|
# for the original polluters. Per Tier 1 investigation:
|
||||||
|
# docs/reports/INVESTIGATION_rag_phase4_final_verify_20260627.md
|
||||||
|
self.ui_files_base_dir = self.project.get("files", {}).get("base_dir", ".")
|
||||||
|
self.ui_shots_base_dir = self.project.get("screenshots", {}).get("base_dir", ".")
|
||||||
self.active_track = None
|
self.active_track = None
|
||||||
self.active_tier = None
|
self.active_tier = None
|
||||||
self.mma_status = 'idle'
|
self.mma_status = 'idle'
|
||||||
|
|||||||
@@ -17,13 +17,20 @@ def test_rag_full_lifecycle_sim(live_gui):
|
|||||||
assert client.wait_for_server(timeout=15), "Hook server did not start"
|
assert client.wait_for_server(timeout=15), "Hook server did not start"
|
||||||
|
|
||||||
# 1. Setup mock project data
|
# 1. Setup mock project data
|
||||||
test_dir = tempfile.mkdtemp()
|
# Per Tier 1 investigation (docs/reports/INVESTIGATION_rag_phase4_final_verify_20260627.md):
|
||||||
|
# Use a temp dir under tests/artifacts/ (NOT tempfile.mkdtemp() which defaults
|
||||||
|
# to C:\Users\Ed\AppData\Local\Temp\tmpXXXX and pollutes the session-scoped
|
||||||
|
# subprocess's ui_files_base_dir with a dead path that persists across
|
||||||
|
# live_gui tests). Also restore files_base_dir and rag_enabled in finally
|
||||||
|
# to prevent state leakage into subsequent live_gui tests.
|
||||||
|
test_dir = tempfile.mkdtemp(dir="tests/artifacts", prefix="rag_visual_sim_")
|
||||||
|
previous_files_base_dir = client.get_value('files_base_dir')
|
||||||
try:
|
try:
|
||||||
(Path(test_dir) / "test_file.txt").write_text("This is a test file about RAG integration. It should be indexed.")
|
(Path(test_dir) / "test_file.txt").write_text("This is a test file about RAG integration. It should be indexed.")
|
||||||
(Path(test_dir) / "other_file.py").write_text("# This is another file\ndef hello():\n print('world')")
|
(Path(test_dir) / "other_file.py").write_text("# This is another file\ndef hello():\n print('world')")
|
||||||
|
|
||||||
# 2. Configure project through Hook API
|
# 2. Configure project through Hook API
|
||||||
client.set_value('files_base_dir', test_dir)
|
client.set_value('files_base_dir', str(Path(test_dir).resolve()))
|
||||||
client.set_value('rag_enabled', True)
|
client.set_value('rag_enabled', True)
|
||||||
client.set_value('rag_source', 'mock') # Use mock to avoid sentence-transformers dependency in CI
|
client.set_value('rag_source', 'mock') # Use mock to avoid sentence-transformers dependency in CI
|
||||||
|
|
||||||
@@ -73,7 +80,20 @@ def test_rag_full_lifecycle_sim(live_gui):
|
|||||||
assert client.get_value('rag_enabled') is False
|
assert client.get_value('rag_enabled') is False
|
||||||
|
|
||||||
finally:
|
finally:
|
||||||
shutil.rmtree(test_dir)
|
# Restore state to prevent pollution of session-scoped subprocess.
|
||||||
|
# ui_files_base_dir and rag_enabled are sticky; without restoration,
|
||||||
|
# the next live_gui test (e.g. test_rag_phase4_final_verify) inherits
|
||||||
|
# the dead temp path and its RAG search silently no-ops on missing files.
|
||||||
|
try:
|
||||||
|
client.set_value('rag_enabled', False)
|
||||||
|
except Exception:
|
||||||
|
pass
|
||||||
|
if previous_files_base_dir is not None:
|
||||||
|
try:
|
||||||
|
client.set_value('files_base_dir', previous_files_base_dir)
|
||||||
|
except Exception:
|
||||||
|
pass
|
||||||
|
shutil.rmtree(test_dir, ignore_errors=True)
|
||||||
|
|
||||||
@pytest.mark.integration
|
@pytest.mark.integration
|
||||||
def test_rag_settings_persistence_sim(live_gui):
|
def test_rag_settings_persistence_sim(live_gui):
|
||||||
|
|||||||
@@ -71,10 +71,15 @@ def test_mma_complete_lifecycle(live_gui) -> None:
|
|||||||
time.sleep(0.3)
|
time.sleep(0.3)
|
||||||
client.set_value('gcli_path', f'"{sys.executable}" "{os.path.abspath("tests/mock_gemini_cli.py")}"')
|
client.set_value('gcli_path', f'"{sys.executable}" "{os.path.abspath("tests/mock_gemini_cli.py")}"')
|
||||||
time.sleep(0.3)
|
time.sleep(0.3)
|
||||||
client.set_value('files_base_dir', 'tests/artifacts/temp_workspace')
|
# Per Tier 1 investigation: do NOT change files_base_dir here and do NOT
|
||||||
|
# click btn_project_save. The previous version set files_base_dir to
|
||||||
|
# 'tests/artifacts/temp_workspace' and persisted it via btn_project_save,
|
||||||
|
# which polluted the session-scoped subprocess's ui_files_base_dir (and
|
||||||
|
# the workspace's manual_slop.toml) with a test-only path that persisted
|
||||||
|
# into subsequent live_gui tests (e.g. test_rag_phase4_final_verify).
|
||||||
|
# The MMA lifecycle does not depend on a specific files_base_dir — the
|
||||||
|
# mock_gemini_cli returns canned responses regardless.
|
||||||
time.sleep(0.3)
|
time.sleep(0.3)
|
||||||
client.click('btn_project_save')
|
|
||||||
time.sleep(1.0) # one full second — let GUI process all set_value tasks
|
|
||||||
|
|
||||||
# ------------------------------------------------------------------
|
# ------------------------------------------------------------------
|
||||||
# Stage 2: Start epic planning
|
# Stage 2: Start epic planning
|
||||||
|
|||||||
Reference in New Issue
Block a user