conductor(checkpoint): Phase 1: Setup and Architecture complete
This commit is contained in:
29
simulation/ARCHITECTURE.md
Normal file
29
simulation/ARCHITECTURE.md
Normal file
@@ -0,0 +1,29 @@
|
||||
# Simulation Architecture
|
||||
|
||||
The extended GUI simulation suite follows a modular architecture to ensure comprehensive coverage and maintainability.
|
||||
|
||||
## 1. Components
|
||||
|
||||
### 1.1 `simulation/sim_base.py`
|
||||
Provides `BaseSimulation`, a base class for all specific simulations.
|
||||
- Initializes `ApiHookClient` and `WorkflowSimulator`.
|
||||
- Provides common utility methods (resetting, waiting, asserting state).
|
||||
- Supports both standalone execution and pytest integration.
|
||||
|
||||
### 1.2 Modular Simulation Scripts
|
||||
Each script focuses on a specific GUI area:
|
||||
- `simulation/sim_context.py`: Context & Discussion panels, history, aggregation.
|
||||
- `simulation/sim_ai_settings.py`: AI model configuration, provider switching.
|
||||
- `simulation/sim_tools.py`: File exploration, MCP tools, web search.
|
||||
- `simulation/sim_execution.py`: AI-generated scripts, confirmation modals, execution.
|
||||
|
||||
## 2. Execution Model
|
||||
|
||||
### 2.1 Standalone
|
||||
Scripts can be run directly (e.g., `python simulation/sim_context.py`) provided the GUI is running with `--enable-test-hooks`.
|
||||
|
||||
### 2.2 Automated (pytest)
|
||||
A thin wrapper in `tests/test_extended_sims.py` will discover and run these simulations using the `live_gui` fixture, ensuring they are part of the CI/CD pipeline.
|
||||
|
||||
## 3. Data Management
|
||||
Simulations will use isolated temporary project files (`tests/temp_sim_*.toml`) to avoid interfering with user configuration or other tests.
|
||||
80
simulation/sim_base.py
Normal file
80
simulation/sim_base.py
Normal file
@@ -0,0 +1,80 @@
|
||||
import sys
|
||||
import os
|
||||
import time
|
||||
import pytest
|
||||
from api_hook_client import ApiHookClient
|
||||
from simulation.workflow_sim import WorkflowSimulator
|
||||
|
||||
# Ensure project root is in path
|
||||
sys.path.append(os.path.abspath(os.path.join(os.path.dirname(__file__), "..")))
|
||||
|
||||
class BaseSimulation:
|
||||
def __init__(self, client: ApiHookClient = None):
|
||||
if client is None:
|
||||
self.client = ApiHookClient()
|
||||
else:
|
||||
self.client = client
|
||||
|
||||
self.sim = WorkflowSimulator(self.client)
|
||||
self.project_path = None
|
||||
|
||||
def setup(self, project_name="SimProject"):
|
||||
print(f"\n[BaseSim] Connecting to GUI...")
|
||||
if not self.client.wait_for_server(timeout=10):
|
||||
raise RuntimeError("Could not connect to GUI. Ensure it is running with --enable-test-hooks")
|
||||
|
||||
print("[BaseSim] Resetting session...")
|
||||
self.client.click("btn_reset")
|
||||
time.sleep(1)
|
||||
|
||||
git_dir = os.path.abspath(".")
|
||||
self.project_path = os.path.abspath(f"tests/temp_{project_name.lower()}.toml")
|
||||
if os.path.exists(self.project_path):
|
||||
os.remove(self.project_path)
|
||||
|
||||
print(f"[BaseSim] Scaffolding Project: {project_name}")
|
||||
self.sim.setup_new_project(project_name, git_dir, self.project_path)
|
||||
|
||||
# Standard test settings
|
||||
self.client.set_value("auto_add_history", True)
|
||||
time.sleep(0.5)
|
||||
|
||||
def teardown(self):
|
||||
if self.project_path and os.path.exists(self.project_path):
|
||||
# We keep it for debugging if it failed, but usually we'd clean up
|
||||
# os.remove(self.project_path)
|
||||
pass
|
||||
print("[BaseSim] Teardown complete.")
|
||||
|
||||
def assert_panel_visible(self, panel_tag, msg=None):
|
||||
# This assumes we have a hook to check panel visibility or just check if an element in it exists
|
||||
# For now, we'll check if we can get a value from an element that should be in that panel
|
||||
# or use a specific hook if available.
|
||||
# Actually, let's just check if get_indicator_state or similar works for generic tags.
|
||||
pass
|
||||
|
||||
def wait_for_element(self, tag, timeout=5):
|
||||
start = time.time()
|
||||
while time.time() - start < timeout:
|
||||
try:
|
||||
# If we can get_value without error, it's likely there
|
||||
self.client.get_value(tag)
|
||||
return True
|
||||
except:
|
||||
time.sleep(0.2)
|
||||
return False
|
||||
|
||||
def run_sim(sim_class):
|
||||
"""Helper to run a simulation class standalone."""
|
||||
sim = sim_class()
|
||||
try:
|
||||
sim.setup()
|
||||
sim.run()
|
||||
print(f"\n[SUCCESS] {sim_class.__name__} completed successfully.")
|
||||
except Exception as e:
|
||||
print(f"\n[FAILURE] {sim_class.__name__} failed: {e}")
|
||||
import traceback
|
||||
traceback.print_exc()
|
||||
sys.exit(1)
|
||||
finally:
|
||||
sim.teardown()
|
||||
Reference in New Issue
Block a user