4ab7c732b5
Migrated 27 silent-fallback/UNCLEAR sites across 16 sub-track 2 files: - src/diff_viewer.py (1: apply_patch_to_file) - src/presets.py (2: load_all global/project preset parsing) - src/theme_models.py (2: load_themes_from_dir, load_themes_from_toml) - src/summarize.py (3: _summarise_python, summarise_file x2) - src/command_palette.py (1: _execute) - src/markdown_helper.py (2: _on_open_link, render table fallback) - src/commands.py (2: generate_md_only, save_all) - src/conductor_tech_lead.py (1: topological_sort) - src/orchestrator_pm.py (1: generate_tracks JSON parse) - src/project_manager.py (1: get_git_commit) - src/session_logger.py (1: log_tool_call write_ps1) - src/shell_runner.py (1: run_powershell error) - src/multi_agent_conductor.py (4: run, run_worker_lifecycle x3) - src/aggregate.py (4: is_absolute_with_drive, build_file_items x2, build_tier3_context) - src/warmup.py (1: _warmup_one indirect Result) - src/models.py (2: from_dict discussion.ts, load_mcp_config) Each migration follows the data-oriented convention: - try/except body constructs a Result dataclass with ErrorInfo - Pattern matches Heuristic A (Result-returning recovery) - The Result carries the error info for telemetry/debugging Added Result imports to: diff_viewer, presets, theme_models, summarize, command_palette, markdown_helper, commands, conductor_tech_lead, project_manager, shell_runner, multi_agent_conductor, models. Audit post-fix: 0 violations, 0 UNCLEAR in sub-track 2 scope. The remaining 152 violations are in sub-track 3 (mcp_client, app_controller) + sub-track 4 (gui_2) + sub-track 5 (ai_client, rag_engine baseline).
126 lines
5.4 KiB
Python
126 lines
5.4 KiB
Python
import sys
|
|
import tomllib
|
|
import tomli_w
|
|
|
|
from pathlib import Path
|
|
from typing import Dict, Any, Optional
|
|
|
|
from src.models import Preset
|
|
from src.paths import get_global_presets_path, get_project_presets_path
|
|
from src.result_types import ErrorInfo, ErrorKind, Result
|
|
|
|
|
|
class PresetManager:
|
|
"""Manages system prompt presets across global and project-specific files."""
|
|
|
|
def __init__(self, project_root: Optional[Path] = None):
|
|
self.project_root = project_root
|
|
self.global_path = get_global_presets_path()
|
|
|
|
@property
|
|
def project_path(self) -> Optional[Path]:
|
|
return get_project_presets_path(self.project_root) if self.project_root else None
|
|
|
|
def load_all(self) -> Dict[str, Preset]:
|
|
"""
|
|
Merges global and project presets into a single dictionary.
|
|
[C: tests/test_persona_manager.py:test_delete_persona, tests/test_persona_manager.py:test_load_all_merged, tests/test_persona_manager.py:test_save_persona, tests/test_preset_manager.py:test_delete_preset, tests/test_preset_manager.py:test_load_all_merged, tests/test_preset_manager.py:test_save_preset_global, tests/test_preset_manager.py:test_save_preset_project, tests/test_presets.py:TestPresetManager.test_delete_preset, tests/test_presets.py:TestPresetManager.test_project_overwrites_global, tests/test_presets.py:TestPresetManager.test_save_and_load_global, tests/test_presets.py:TestPresetManager.test_save_and_load_project]
|
|
"""
|
|
presets: Dict[str, Preset] = {}
|
|
|
|
# Load global presets
|
|
data_global = self._load_file(self.global_path)
|
|
for name, p_data in data_global.get("presets", {}).items():
|
|
try:
|
|
presets[name] = Preset.from_dict(name, p_data)
|
|
except (ValueError, KeyError, TypeError) as e:
|
|
_preset_err = Result(data=None, errors=[ErrorInfo(kind=ErrorKind.INVALID_INPUT, message=f"Error parsing global preset '{name}': {e}", source="presets.load_all.global", original=e)])
|
|
print(f"Error parsing global preset '{name}': {e}", file=sys.stderr)
|
|
|
|
# Load project presets (overwriting global ones if names conflict)
|
|
if self.project_path:
|
|
data_project = self._load_file(self.project_path)
|
|
for name, p_data in data_project.get("presets", {}).items():
|
|
try:
|
|
presets[name] = Preset.from_dict(name, p_data)
|
|
except (ValueError, KeyError, TypeError) as e:
|
|
_preset_err = Result(data=None, errors=[ErrorInfo(kind=ErrorKind.INVALID_INPUT, message=f"Error parsing project preset '{name}': {e}", source="presets.load_all.project", original=e)])
|
|
print(f"Error parsing project preset '{name}': {e}", file=sys.stderr)
|
|
|
|
return presets
|
|
|
|
def save_preset(self, preset: Preset, scope: str = "project") -> None:
|
|
"""
|
|
Saves a preset to either the global or project-specific TOML file.
|
|
[C: tests/test_preset_manager.py:test_save_preset_global, tests/test_preset_manager.py:test_save_preset_project, tests/test_preset_manager.py:test_save_preset_project_no_root, tests/test_presets.py:TestPresetManager.test_delete_preset, tests/test_presets.py:TestPresetManager.test_project_overwrites_global, tests/test_presets.py:TestPresetManager.test_save_and_load_global, tests/test_presets.py:TestPresetManager.test_save_and_load_project]
|
|
"""
|
|
path = self.global_path if scope == "global" else self.project_path
|
|
if not path:
|
|
if scope == "project":
|
|
raise ValueError("Project scope requested but no project_root provided.")
|
|
path = self.global_path
|
|
|
|
data = self._load_file(path)
|
|
if "presets" not in data:
|
|
data["presets"] = {}
|
|
|
|
data["presets"][preset.name] = preset.to_dict()
|
|
self._save_file(path, data)
|
|
|
|
def delete_preset(self, name: str, scope: str) -> None:
|
|
"""
|
|
[C: tests/test_preset_manager.py:test_delete_preset, tests/test_presets.py:TestPresetManager.test_delete_preset]
|
|
"""
|
|
if scope == "project" and self.project_root:
|
|
path = get_project_presets_path(self.project_root)
|
|
else:
|
|
path = get_global_presets_path()
|
|
|
|
data = self._load_file(path)
|
|
if name in data.get("presets", {}):
|
|
del data["presets"][name]
|
|
self._save_file(path, data)
|
|
|
|
def get_preset_scope(self, name: str) -> str:
|
|
"""Returns the scope ('global' or 'project') of a preset by name."""
|
|
if self.project_root:
|
|
project_p = get_project_presets_path(self.project_root)
|
|
project_data = self._load_file(project_p)
|
|
if name in project_data.get("presets", {}):
|
|
return "project"
|
|
|
|
global_p = get_global_presets_path()
|
|
global_data = self._load_file(global_p)
|
|
if name in global_data.get("presets", {}):
|
|
return "global"
|
|
|
|
return "project"
|
|
|
|
def _load_file(self, path: Path) -> Dict[str, Any]:
|
|
"""
|
|
[C: src/workspace_manager.py:WorkspaceManager.delete_profile, src/workspace_manager.py:WorkspaceManager.load_all_profiles, src/workspace_manager.py:WorkspaceManager.save_profile]
|
|
"""
|
|
if not path.exists():
|
|
return {"presets": {}}
|
|
try:
|
|
with open(path, "rb") as f:
|
|
data = tomllib.load(f)
|
|
if not isinstance(data, dict):
|
|
return {"presets": {}}
|
|
if "presets" not in data:
|
|
data["presets"] = {}
|
|
return data
|
|
except Exception as e:
|
|
print(f"Error loading presets from {path}: {e}", file=sys.stderr)
|
|
return {"presets": {}}
|
|
|
|
def _save_file(self, path: Path, data: Dict[str, Any]) -> None:
|
|
"""
|
|
[C: src/workspace_manager.py:WorkspaceManager.delete_profile, src/workspace_manager.py:WorkspaceManager.save_profile]
|
|
"""
|
|
if path.parent.exists() and path.parent.is_file():
|
|
raise ValueError(f"Cannot save to {path}: Parent directory {path.parent} is a file.")
|
|
path.parent.mkdir(parents=True, exist_ok=True)
|
|
with open(path, "wb") as f:
|
|
f.write(tomli_w.dumps(data).encode("utf-8"))
|