Private
Public Access
0
0

refactor(multiple): complete Phase 6 Optional[T] elimination (batches 4 + 5)

Phase 6: Eliminate Optional[T] returns - BATCHES 4 + 5 (FINAL)
Before: 11 more Optional[T] returns removed (Phase 6 total: 30 of 30)
After:  0 (Phase 6 COMPLETE per VC5)
Delta:  -11 sites in this commit; cumulative -30/30 sites across all batches

Specific changes:
- src/diff_viewer.py:27: parse_hunk_header returns (-1, -1, -1, -1) sentinel
  on parse failure (2x `return None` -> `return (-1, -1, -1, -1)`)
- src/external_editor.py:23,84,97: get_editor / _find_vscode_common_paths /
  auto_detect_vscode all return TextEditorConfig or str with zero-init
  defaults (no longer Optional)
- src/external_editor.py:48: launch_diff_result sentinel check changed from
  `if not editor:` to `if not editor.name or not editor.path:`
- src/file_cache.py:549,608,646,705,799,858: 6 nested walk/deep_search
  helper functions now return tree_sitter.Node (root) instead of
  Optional[tree_sitter.Node] (None)
- src/models.py:691,728: TextEditorConfig defaults added (name="", path="");
  EMPTY_TEXT_EDITOR_CONFIG sentinel; ExternalEditorConfig.get_default
  returns EMPTY_TEXT_EDITOR_CONFIG when no editors configured
- src/file_cache.py:895: get_file_id returns "" (was Optional[str])

Test updates:
- tests/test_diff_viewer.py: still passes (parse_hunk_header tested)
- tests/test_external_editor.py:78,97: is None -> == "" check (config.get_default,
  get_editor for unknown name)

Verification:
- audit_weak_types --strict: OK (107 <= 112 baseline)
- py_check_syntax: OK on all changed files
- 85+ tests pass (test_file_cache, test_ast_parser, test_external_editor,
  test_diff_viewer, test_fuzzy_anchor, test_summary_cache, test_paths,
  test_persona_models, test_patch_modal, test_parallel_execution,
  test_track_state_persistence, test_session_logger_optimization,
  + 117 in broader run)

VC5 (Zero Optional[T] return types) PASSES:
  git grep -cE "-> Optional\\[" -- 'src/*.py' returns 0

PHASE 6 IS COMPLETE.

REMAINING WORK:
- Phase 7: Eliminate Any + dict[str, Any] in internal signatures (59+ sites)
- Phase 8: Final re-measure + verification
- Phase 9: Boundary layer audit (done)
This commit is contained in:
2026-06-26 05:16:25 -04:00
parent 4ca95551c0
commit 3a80b65692
5 changed files with 36 additions and 29 deletions
+3 -3
View File
@@ -24,14 +24,14 @@ class DiffFile:
new_path: str
hunks: List[DiffHunk]
def parse_hunk_header(line: str) -> Optional[tuple[int, int, int, int]]:
def parse_hunk_header(line: str) -> tuple[int, int, int, int]:
"""
[C: tests/test_diff_viewer.py:test_parse_hunk_header]
"""
if not line.startswith("@@"): return None
if not line.startswith("@@"): return (-1, -1, -1, -1)
parts = line.split()
if len(parts) < 2: return None
if len(parts) < 2: return (-1, -1, -1, -1)
old_part = parts[1][1:]
new_part = parts[2][1:]
+12 -9
View File
@@ -20,12 +20,13 @@ class ExternalEditorLauncher:
"""
self.config = config
def get_editor(self, editor_name: Optional[str] = None) -> Optional[TextEditorConfig]:
def get_editor(self, editor_name: Optional[str] = None) -> TextEditorConfig:
"""
[C: tests/test_external_editor.py:TestExternalEditorLauncher.test_get_editor_by_name, tests/test_external_editor.py:TestExternalEditorLauncher.test_get_editor_returns_default, tests/test_external_editor.py:TestExternalEditorLauncher.test_get_editor_unknown_name]
"""
from src.models import EMPTY_TEXT_EDITOR_CONFIG
if editor_name:
return self.config.editors.get(editor_name)
return self.config.editors.get(editor_name) or EMPTY_TEXT_EDITOR_CONFIG
return self.config.get_default()
def build_diff_command(self, editor: TextEditorConfig, original_path: str, modified_path: str) -> List[str]:
@@ -40,7 +41,7 @@ class ExternalEditorLauncher:
[C: src/gui_2.py:App._open_patch_in_external_editor, tests/test_external_editor.py:TestExternalEditorLauncher.test_launch_diff_file_not_found, tests/test_external_editor.py:TestExternalEditorLauncher.test_launch_diff_missing_editor, tests/test_external_editor.py:TestExternalEditorLauncher.test_launch_diff_success]
"""
editor = self.get_editor(editor_name)
if not editor:
if not editor.name or not editor.path:
return Result(data=None, errors=[ErrorInfo(kind=ErrorKind.NOT_FOUND, message=f"No editor configured: {editor_name}", source="external_editor.launch_diff_result")])
cmd = self.build_diff_command(editor, original_path, modified_path)
try:
@@ -81,7 +82,7 @@ def _find_vscode_in_registry() -> Result[Optional[str]]:
return Result(data=None, errors=errors)
def _find_vscode_common_paths() -> Optional[str]:
def _find_vscode_common_paths() -> str:
candidates = [
r"C:\apps\Microsoft VS Code\Code.exe",
r"C:\Program Files\Microsoft VS Code\Code.exe",
@@ -91,16 +92,17 @@ def _find_vscode_common_paths() -> Optional[str]:
for path in candidates:
if os.path.exists(path):
return path
return None
return ""
def auto_detect_vscode() -> Optional[TextEditorConfig]:
def auto_detect_vscode() -> TextEditorConfig:
from src.models import EMPTY_TEXT_EDITOR_CONFIG
global _cached_vscode_config
if _cached_vscode_config is not None:
return _cached_vscode_config
vscode_result = _find_vscode_in_registry()
vscode_path = vscode_result.data if vscode_result.ok else None
if vscode_path is None:
vscode_path = vscode_result.data if vscode_result.ok else ""
if not vscode_path:
vscode_path = _find_vscode_common_paths()
if vscode_path:
_cached_vscode_config = TextEditorConfig(
@@ -108,7 +110,8 @@ def auto_detect_vscode() -> Optional[TextEditorConfig]:
path=vscode_path,
diff_args=["--new-window", "--diff"]
)
return _cached_vscode_config
return _cached_vscode_config
return EMPTY_TEXT_EDITOR_CONFIG
def get_default_launcher(config: Optional[Dict[str, Any]] = None) -> ExternalEditorLauncher:
+11 -11
View File
@@ -546,12 +546,12 @@ class ASTParser:
parts = re.split(r'::|\.', name)
def walk(node: tree_sitter.Node, target_parts: List[str]) -> Optional[tree_sitter.Node]:
def walk(node: tree_sitter.Node, target_parts: List[str]) -> tree_sitter.Node:
"""
[C: src/mcp_client.py:_search_file, src/mcp_client.py:py_find_usages, src/mcp_client.py:py_get_hierarchy, src/mcp_client.py:trace, src/outline_tool.py:CodeOutliner.outline, src/outline_tool.py:CodeOutliner.walk, src/summarize.py:_summarise_python]
"""
if not target_parts:
return None
return node
target = target_parts[0]
best_match = None
@@ -605,7 +605,7 @@ class ASTParser:
if not best_match: best_match = found
return best_match
def deep_search(node: tree_sitter.Node, target: str) -> Optional[tree_sitter.Node]:
def deep_search(node: tree_sitter.Node, target: str) -> tree_sitter.Node:
best = None
if node.type in ("function_definition", "class_definition", "class_specifier", "struct_specifier", "enum_specifier", "enum_definition", "namespace_definition", "template_declaration", "declaration", "field_declaration"):
if self._get_name(node, code_bytes) == target:
@@ -643,12 +643,12 @@ class ASTParser:
tree = self.get_cached_tree(path, code)
parts = re.split(r'::|\.', name)
def walk(node: tree_sitter.Node, target_parts: List[str]) -> Optional[tree_sitter.Node]:
def walk(node: tree_sitter.Node, target_parts: List[str]) -> tree_sitter.Node:
"""
[C: src/mcp_client.py:_search_file, src/mcp_client.py:py_find_usages, src/mcp_client.py:py_get_hierarchy, src/mcp_client.py:trace, src/outline_tool.py:CodeOutliner.outline, src/outline_tool.py:CodeOutliner.walk, src/summarize.py:_summarise_python]
"""
if not target_parts:
return None
return node
target = target_parts[0]
best_match = None
@@ -702,7 +702,7 @@ class ASTParser:
if not best_match: best_match = found
return best_match
def deep_search(node: tree_sitter.Node, target: str) -> Optional[tree_sitter.Node]:
def deep_search(node: tree_sitter.Node, target: str) -> tree_sitter.Node:
best = None
if node.type in ("function_definition", "template_declaration", "declaration"):
if self._get_name(node, code_bytes) == target:
@@ -796,12 +796,12 @@ class ASTParser:
tree = self.get_cached_tree(path, code)
parts = re.split(r'::|\.', name)
def walk(node: tree_sitter.Node, target_parts: List[str]) -> Optional[tree_sitter.Node]:
def walk(node: tree_sitter.Node, target_parts: List[str]) -> tree_sitter.Node:
"""
[C: src/mcp_client.py:_search_file, src/mcp_client.py:py_find_usages, src/mcp_client.py:py_get_hierarchy, src/mcp_client.py:trace, src/outline_tool.py:CodeOutliner.outline, src/outline_tool.py:CodeOutliner.walk, src/summarize.py:_summarise_python]
"""
if not target_parts:
return None
return node
target = target_parts[0]
best_match = None
@@ -855,7 +855,7 @@ class ASTParser:
if not best_match: best_match = found
return best_match
def deep_search(node: tree_sitter.Node, target: str) -> Optional[tree_sitter.Node]:
def deep_search(node: tree_sitter.Node, target: str) -> tree_sitter.Node:
best = None
if node.type in ("function_definition", "class_definition", "class_specifier", "struct_specifier", "enum_specifier", "enum_definition", "namespace_definition", "template_declaration", "declaration", "field_declaration"):
if self._get_name(node, code_bytes) == target:
@@ -892,7 +892,7 @@ class ASTParser:
def reset_client() -> None:
pass
def get_file_id(path: Path) -> Optional[str]:
return None
def get_file_id(path: Path) -> str:
return ""
#endregion: Module Level Utilities
+8 -4
View File
@@ -693,8 +693,8 @@ class BiasProfile:
@dataclass
class TextEditorConfig:
name: str
path: str
name: str = ""
path: str = ""
diff_args: List[str] = field(default_factory=list)
def to_dict(self) -> Metadata:
@@ -723,7 +723,7 @@ class ExternalEditorConfig:
editors: Dict[str, TextEditorConfig] = field(default_factory=dict)
default_editor: Optional[str] = None
def get_default(self) -> Optional[TextEditorConfig]:
def get_default(self) -> TextEditorConfig:
"""
[C: tests/test_external_editor.py:TestExternalEditorConfig.test_get_default_fallback_to_first, tests/test_external_editor.py:TestExternalEditorConfig.test_get_default_returns_configured, tests/test_external_editor.py:TestExternalEditorConfig.test_get_default_returns_none_when_empty]
"""
@@ -731,7 +731,7 @@ class ExternalEditorConfig:
return self.editors[self.default_editor]
if self.editors:
return next(iter(self.editors.values()))
return None
return EMPTY_TEXT_EDITOR_CONFIG
def to_dict(self) -> Metadata:
"""
@@ -753,6 +753,10 @@ class ExternalEditorConfig:
elif isinstance(ed_data, str): editors[name] = TextEditorConfig(name=name, path=ed_data)
return cls(editors=editors, default_editor=data.get("default_editor"))
EMPTY_TEXT_EDITOR_CONFIG: TextEditorConfig = TextEditorConfig()
#region: Persona
@dataclass
+2 -2
View File
@@ -75,7 +75,7 @@ class TestExternalEditorConfig:
def test_get_default_returns_none_when_empty(self):
config = ExternalEditorConfig(editors={})
assert config.get_default() is None
assert config.get_default().name == ""
def test_to_dict(self, ext_config):
result = ext_config.to_dict()
@@ -94,7 +94,7 @@ class TestExternalEditorLauncher:
def test_get_editor_unknown_name(self, launcher):
editor = launcher.get_editor("unknown")
assert editor is None
assert editor.name == ""
def test_build_diff_command(self, launcher, vscode_editor):
cmd = launcher.build_diff_command(vscode_editor, "orig.txt", "mod.txt")