add --wipe-project-memory flag

This commit is contained in:
AI Christianson 2025-03-07 12:06:40 -05:00
parent 95f9d8c2c4
commit 089bd9b368
2 changed files with 173 additions and 1 deletions

View File

@ -275,6 +275,11 @@ Examples:
default=8080,
help="Port to listen on for web interface (default: 8080)",
)
parser.add_argument(
"--wipe-project-memory",
action="store_true",
help="Delete the project database file (.ra-aid/pk.db) before starting, effectively wiping all stored memory",
)
if args is None:
args = sys.argv[1:]
parsed_args = parser.parse_args(args)
@ -351,6 +356,31 @@ def is_stage_requested(stage: str) -> bool:
return False
def wipe_project_memory():
"""Delete the project database file to wipe all stored memory.
Returns:
str: A message indicating the result of the operation
"""
import os
from pathlib import Path
cwd = os.getcwd()
ra_aid_dir = Path(os.path.join(cwd, ".ra-aid"))
db_path = os.path.join(ra_aid_dir, "pk.db")
if not os.path.exists(db_path):
return "No project memory found to wipe."
try:
os.remove(db_path)
return "Project memory wiped successfully."
except PermissionError:
return "Error: Could not wipe project memory due to permission issues."
except Exception as e:
return f"Error: Failed to wipe project memory: {str(e)}"
def build_status():
"""Build status panel with model and feature information.
@ -424,8 +454,10 @@ def build_status():
except RuntimeError as e:
logger.debug(f"Failed to get research notes count: {e}")
# Add memory statistics line
# Add memory statistics line with reset option note
status.append(f"\n💾 Memory: {fact_count} facts, {snippet_count} snippets, {note_count} notes")
if fact_count > 0 or snippet_count > 0 or note_count > 0:
status.append(" (use --wipe-project-memory to reset)")
# Check for newer version
version_message = check_for_newer_version()
@ -441,6 +473,12 @@ def main():
args = parse_arguments()
setup_logging(args.log_mode, args.pretty_logger, args.log_level)
logger.debug("Starting RA.Aid with arguments: %s", args)
# Check if we need to wipe project memory before starting
if args.wipe_project_memory:
result = wipe_project_memory()
logger.info(result)
print(f"📋 {result}")
# Launch web interface if requested
if args.webui:

View File

@ -0,0 +1,134 @@
"""Tests for wipe_project_memory functionality."""
import os
import tempfile
from pathlib import Path
from unittest.mock import patch, MagicMock
import pytest
from ra_aid.__main__ import wipe_project_memory, parse_arguments
def test_wipe_project_memory_function():
"""Test that wipe_project_memory function correctly deletes the database file."""
# Create a temporary directory to simulate the project directory
with tempfile.TemporaryDirectory() as temp_dir:
# Create a fake .ra-aid directory and pk.db file
ra_aid_dir = Path(os.path.join(temp_dir, ".ra-aid"))
os.makedirs(ra_aid_dir, exist_ok=True)
db_path = os.path.join(ra_aid_dir, "pk.db")
# Create an empty file
with open(db_path, "w") as f:
f.write("")
# Verify the file exists
assert os.path.exists(db_path)
# Mock getcwd to return our temp directory
with patch("os.getcwd", return_value=temp_dir):
# Call the function
result = wipe_project_memory()
# Verify the file no longer exists
assert not os.path.exists(db_path)
assert result == "Project memory wiped successfully."
def test_wipe_project_memory_no_file():
"""Test wipe_project_memory when no database file exists."""
# Create a temporary directory without a pk.db file
with tempfile.TemporaryDirectory() as temp_dir:
# Mock getcwd to return our temp directory
with patch("os.getcwd", return_value=temp_dir):
# Call the function
result = wipe_project_memory()
# Verify the result message
assert result == "No project memory found to wipe."
def test_wipe_project_memory_flag():
"""Test that the --wipe-project-memory flag is properly parsed."""
# Test without the flag
args = parse_arguments(["-m", "test message"])
assert not hasattr(args, "wipe_project_memory") or not args.wipe_project_memory
# Test with the flag
args = parse_arguments(["-m", "test message", "--wipe-project-memory"])
assert args.wipe_project_memory is True
def test_build_status_shows_reset_option():
"""Test that build_status function shows reset option when there are items in memory."""
from unittest.mock import patch, MagicMock
from ra_aid.__main__ import build_status
# Mock repositories to return different numbers of items
with patch("ra_aid.__main__.get_key_fact_repository") as mock_fact_repo, \
patch("ra_aid.__main__.get_key_snippet_repository") as mock_snippet_repo, \
patch("ra_aid.__main__.get_research_note_repository") as mock_note_repo, \
patch("ra_aid.__main__.get_config_repository") as mock_config_repo:
# Set up mock repositories
mock_fact_repo.return_value.get_all.return_value = [1, 2, 3] # 3 facts
mock_snippet_repo.return_value.get_all.return_value = [1] # 1 snippet
mock_note_repo.return_value.get_all.return_value = [1, 2] # 2 notes
mock_config_repo.return_value.get.return_value = None
# Call build_status
status = build_status()
# Convert status to string for easier assertion
status_str = str(status)
# Verify it includes the memory statistics with reset option
assert "Memory: 3 facts, 1 snippets, 2 notes" in status_str
assert "use --wipe-project-memory to reset" in status_str
# Test with empty memory - should not show reset option
mock_fact_repo.return_value.get_all.return_value = []
mock_snippet_repo.return_value.get_all.return_value = []
mock_note_repo.return_value.get_all.return_value = []
# Call build_status again
status = build_status()
status_str = str(status)
# Verify it doesn't include the reset option
assert "Memory: 0 facts, 0 snippets, 0 notes" in status_str
assert "use --wipe-project-memory to reset" not in status_str
def test_main_with_wipe_project_memory_flag():
"""Test that the main function properly calls wipe_project_memory when flag is set."""
from ra_aid.__main__ import main
# Create a mock args object with wipe_project_memory=True
mock_args = MagicMock()
mock_args.wipe_project_memory = True
# Mock the wipe_project_memory function to raise SystemExit after being called
def mock_wipe_side_effect():
raise SystemExit(0)
mock_wipe = MagicMock(side_effect=mock_wipe_side_effect)
# Mock all necessary dependencies to prevent actual operations
with patch("ra_aid.__main__.wipe_project_memory", mock_wipe), \
patch("ra_aid.__main__.parse_arguments", return_value=mock_args), \
patch("ra_aid.__main__.setup_logging"), \
patch("ra_aid.__main__.get_config_repository"), \
patch("ra_aid.__main__.launch_webui"), \
patch("ra_aid.__main__.DatabaseManager"):
# Call main() and catch SystemExit since we're raising it
try:
main()
except SystemExit:
pass
# Verify wipe_project_memory was called
mock_wipe.assert_called_once()