From 6fc9fccd12e193cc01e6fd9603ff725925aa323d Mon Sep 17 00:00:00 2001 From: Jerome Hardaway Date: Thu, 21 Nov 2024 19:22:27 -0500 Subject: [PATCH 1/5] fix tests --- README.md | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/README.md b/README.md index 4f18b82..d220e9b 100644 --- a/README.md +++ b/README.md @@ -12,18 +12,21 @@ VetsAI is an AI-powered virtual assistant designed to help veterans navigate emp ## Prerequisites To run this application, ensure you have the following installed: + - Python 3.8 or later - A virtual environment (recommended) ## Installation 1. **Clone the repository**: + ```bash git clone cd ``` 2. **Set up a virtual environment**: + ```bash python -m venv venv source venv/bin/activate # For macOS/Linux @@ -31,13 +34,16 @@ To run this application, ensure you have the following installed: ``` 3. **Install dependencies**: + ```bash pip install -r requirements.txt ``` 4. **Set up environment variables**: + - Create a .env file in the root of your project. - Add your OpenAI API key to the .env file: + ``` OPENAI_API_KEY=your-openai-api-key ``` @@ -45,11 +51,13 @@ To run this application, ensure you have the following installed: ## Running the App 1. **Run the Streamlit app**: + ```bash streamlit run app.py ``` 2. **Access the app**: + Open your web browser and navigate to http://localhost:8501. ## Usage @@ -67,6 +75,7 @@ To run this application, ensure you have the following installed: ## Dependencies The following Python libraries are required to run this app: + - `streamlit`: For the web interface. - `httpx`: To make HTTP requests to OpenAI's API. - `nest-asyncio`: To allow nested event loops for async operations. From dc92ea5cd94d2b1a343311201f7c5de3993c6ef8 Mon Sep 17 00:00:00 2001 From: Jerome Hardaway Date: Thu, 21 Nov 2024 19:28:57 -0500 Subject: [PATCH 2/5] updated tests --- tests/test_streamlit_app.py | 11 +++++++++-- 1 file changed, 9 insertions(+), 2 deletions(-) diff --git a/tests/test_streamlit_app.py b/tests/test_streamlit_app.py index 992cffe..11679ec 100644 --- a/tests/test_streamlit_app.py +++ b/tests/test_streamlit_app.py @@ -153,7 +153,7 @@ def test_translate_military_code_not_found(mock_job_codes): assert "dev_path" in result["data"] assert isinstance(result["data"]["tech_focus"], list) -@patch("openai.chat.completions.create") +@patch("app.openai.ChatCompletion.create") def test_get_chat_response(mock_create): # Mock the OpenAI response mock_response = MagicMock() @@ -162,8 +162,15 @@ def test_get_chat_response(mock_create): messages = [{"role": "user", "content": "Hello"}] response = get_chat_response(messages) + + # Assert that the mock response is returned correctly assert response == "Test response" - mock_create.assert_called_once() + + # Verify that the OpenAI API was called exactly once + mock_create.assert_called_once_with( + model="gpt-4", # Replace with your model name if different + messages=messages + ) def test_handle_command_mos(mock_job_codes): with patch("streamlit.session_state") as mock_session: From 04d5c113090b18ad7f60e89bf5ce0ee6b0643e9a Mon Sep 17 00:00:00 2001 From: Jerome Hardaway Date: Thu, 21 Nov 2024 19:36:07 -0500 Subject: [PATCH 3/5] correct openai patch in chat response test --- tests/test_streamlit_app.py | 346 ++++++++++++++++-------------------- 1 file changed, 158 insertions(+), 188 deletions(-) diff --git a/tests/test_streamlit_app.py b/tests/test_streamlit_app.py index 11679ec..ce9a170 100644 --- a/tests/test_streamlit_app.py +++ b/tests/test_streamlit_app.py @@ -7,7 +7,6 @@ import openai from datetime import datetime -# Get the absolute path to the project root directory ROOT_DIR = Path(__file__).parent.parent sys.path.append(str(ROOT_DIR)) @@ -22,7 +21,6 @@ parse_mos_file ) -# Sample text content SAMPLE_MOS_TEXT = """ Job Code: 25B @@ -35,7 +33,7 @@ @pytest.fixture def mock_job_codes(): return { - "MOS_25B": { + "25B": { "title": "Information Technology Specialist", "branch": "army", "category": "information_technology", @@ -50,12 +48,8 @@ def mock_job_codes(): } } -@patch("os.path.join", lambda *args: "/".join(args)) -@patch("builtins.open", new_callable=mock_open) -def test_load_military_job_codes(mock_file): - # Setup mock file content - mock_file.return_value.__enter__.return_value.read.return_value = SAMPLE_MOS_TEXT - +@pytest.fixture +def mock_file_system(): def mock_exists(path): return True @@ -64,193 +58,169 @@ def mock_listdir(path): return ["army", "air_force", "navy", "marine_corps", "coast_guard"] else: return ["25B.txt"] - - with patch("os.path.exists", side_effect=mock_exists), \ - patch("os.listdir", side_effect=mock_listdir): + + return {"exists": mock_exists, "listdir": mock_listdir} + +class TestMilitaryJobCodes: + @patch("os.path.join", lambda *args: "/".join(args)) + @patch("builtins.open", new_callable=mock_open) + def test_load_military_job_codes(self, mock_file, mock_file_system): + mock_file.return_value.__enter__.return_value.read.return_value = SAMPLE_MOS_TEXT + + with patch("os.path.exists", side_effect=mock_file_system["exists"]), \ + patch("os.listdir", side_effect=mock_file_system["listdir"]): + + job_codes = load_military_job_codes() + + assert isinstance(job_codes, dict) + assert len(job_codes) > 0 + + for key, value in job_codes.items(): + assert isinstance(value, dict) + assert all(field in value for field in ["title", "branch", "skills"]) + assert isinstance(value["skills"], list) + + assert mock_file.call_count > 0 + + def test_parse_mos_file(self): + result = parse_mos_file(SAMPLE_MOS_TEXT) - job_codes = load_military_job_codes() + assert isinstance(result, dict) + assert all(field in result for field in ["title", "category", "skills"]) + assert isinstance(result["skills"], list) + assert len(result["skills"]) > 0 - # Basic validations - assert isinstance(job_codes, dict) - assert len(job_codes) > 0 + assert "manages or supervises" in result["title"].lower() + assert result["category"] == "information_technology" + assert any("network" in skill.lower() for skill in result["skills"]) + + @pytest.mark.parametrize("test_input,expected", [ + ("", { + "title": "Military Professional", + "category": "general", + "skills": [] + }), + ("Job Code: 25B", { + "title": "Military Professional", + "category": "general", + "skills": [] + }), + ("""Job Code: 25B + Description: + Network & Systems Administrator (IT/IS) + Manages & maintains computer networks/systems.""", { + "category": "information_technology", + "skills": pytest.approx([], abs=10) + }) + ]) + def test_parse_mos_file_edge_cases(self, test_input, expected): + result = parse_mos_file(test_input) + for key, value in expected.items(): + if isinstance(value, list) and isinstance(expected[key], list): + assert len(result[key]) >= len(value) + else: + assert result[key] == value + +class TestPathMapping: + @pytest.mark.parametrize("category,skills,expected_path", [ + ("information_technology", ["programming", "networking"], "Full Stack Development"), + ("cyber", [], "Security-Focused Development"), + ("intelligence", [], "AI/ML Development"), + ("communications", [], "Frontend Development"), + ("maintenance", [], "Backend Development"), + ("unknown", [], "Full Stack Development") + ]) + def test_map_to_vwc_path(self, category, skills, expected_path): + result = map_to_vwc_path(category, skills) + assert result["path"] == expected_path + assert isinstance(result["tech_focus"], list) + assert len(result["tech_focus"]) > 0 + +class TestMilitaryCodeTranslation: + def test_translate_military_code_found(self, mock_job_codes): + result = translate_military_code("25B", mock_job_codes) + assert result["found"] is True + assert result["data"]["title"] == "Information Technology Specialist" + assert result["data"]["branch"] == "army" + + def test_translate_military_code_not_found(self, mock_job_codes): + result = translate_military_code("99Z", mock_job_codes) + assert result["found"] is False + assert "dev_path" in result["data"] + assert isinstance(result["data"]["tech_focus"], list) + +class TestChatFunctionality: + @patch('openai.ChatCompletion.create') + def test_get_chat_response(self, mock_create): + mock_response = MagicMock() + mock_response.choices = [MagicMock(message=MagicMock(content="Test response"))] + mock_create.return_value = mock_response + + messages = [{"role": "user", "content": "Hello"}] + response = get_chat_response(messages) + + assert response == "Test response" + mock_create.assert_called_once_with( + model="gpt-4", + messages=messages + ) + + def test_handle_command_mos(self, mock_job_codes): + with patch("streamlit.session_state") as mock_session: + mock_session.job_codes = mock_job_codes + response = handle_command("/mos 25B") + assert response is not None + assert "Information Technology Specialist" in response + assert "VWC Development Path" in response + + @pytest.mark.parametrize("command,expected", [ + ("/invalid", None), + ("/mos", "Please provide a military job code"), + ("/mos ", "Please provide a military job code") + ]) + def test_handle_command_edge_cases(self, command, expected): + response = handle_command(command) + if expected is None: + assert response is None + else: + assert expected in response + +class TestDataManagement: + def test_export_chat_history(self): + chat_history = [ + {"role": "user", "content": "Hello"}, + {"role": "assistant", "content": "Hi"} + ] + result = export_chat_history(chat_history) - # Verify the structure - for key, value in job_codes.items(): - assert isinstance(value, dict) - assert "title" in value - assert "branch" in value - assert "skills" in value - assert isinstance(value["skills"], list) + assert isinstance(result, str) + exported_data = json.loads(result) + assert isinstance(exported_data["timestamp"], str) + assert datetime.fromisoformat(exported_data["timestamp"]) + assert len(exported_data["messages"]) == 2 + assert all(msg["role"] in ["user", "assistant"] for msg in exported_data["messages"]) + + @patch("builtins.open", new_callable=mock_open) + @patch("os.makedirs") + def test_save_feedback(self, mock_makedirs, mock_file): + feedback = { + "rating": 5, + "feedback": "Great service!", + "session_id": "test123" + } - # Verify that mock_file was called - assert mock_file.call_count > 0 - -def test_parse_mos_file(): - """Test the MOS file parsing function""" - result = parse_mos_file(SAMPLE_MOS_TEXT) - - # Basic structure tests - assert isinstance(result, dict) - assert "title" in result - assert "category" in result - assert "skills" in result - assert isinstance(result["skills"], list) - assert len(result["skills"]) > 0 - - # Content tests - assert result["title"].startswith("Manages or supervises") - assert result["category"] == "information_technology" # Should match because of network/data/system keywords - - # Skills check - assert any("network" in skill.lower() for skill in result["skills"]) - -def test_parse_mos_file_edge_cases(): - """Test parse_mos_file with various edge cases""" - # Empty content - empty_result = parse_mos_file("") - assert empty_result["title"] == "Military Professional" - assert empty_result["category"] == "general" - assert isinstance(empty_result["skills"], list) - - # Content with only job code - job_code_only = "Job Code: 25B" - job_code_result = parse_mos_file(job_code_only) - assert job_code_result["title"] == "Military Professional" - assert isinstance(job_code_result["skills"], list) - - # Content with special characters - special_chars = """ - Job Code: 25B - - Description: - Network & Systems Administrator (IT/IS) - - Manages & maintains computer networks/systems. - """ - special_result = parse_mos_file(special_chars) - assert special_result["category"] == "information_technology" - -def test_map_to_vwc_path_it_category(): - result = map_to_vwc_path("information_technology", ["programming", "networking"]) - assert result["path"] == "Full Stack Development" - assert len(result["tech_focus"]) > 0 - assert any("TypeScript" in focus for focus in result["tech_focus"]) - -def test_map_to_vwc_path_default(): - result = map_to_vwc_path("unknown_category", []) - assert result["path"] == "Full Stack Development" - assert len(result["tech_focus"]) > 0 - -def test_translate_military_code_found(mock_job_codes): - result = translate_military_code("25B", mock_job_codes) - assert result["found"] == True - assert result["data"]["title"] == "Information Technology Specialist" - assert result["data"]["branch"] == "army" - -def test_translate_military_code_not_found(mock_job_codes): - result = translate_military_code("99Z", mock_job_codes) - assert result["found"] == False - assert "dev_path" in result["data"] - assert isinstance(result["data"]["tech_focus"], list) - -@patch("app.openai.ChatCompletion.create") -def test_get_chat_response(mock_create): - # Mock the OpenAI response - mock_response = MagicMock() - mock_response.choices = [MagicMock(message=MagicMock(content="Test response"))] - mock_create.return_value = mock_response - - messages = [{"role": "user", "content": "Hello"}] - response = get_chat_response(messages) - - # Assert that the mock response is returned correctly - assert response == "Test response" - - # Verify that the OpenAI API was called exactly once - mock_create.assert_called_once_with( - model="gpt-4", # Replace with your model name if different - messages=messages - ) - -def test_handle_command_mos(mock_job_codes): - with patch("streamlit.session_state") as mock_session: - mock_session.job_codes = mock_job_codes - response = handle_command("/mos 25B") - assert response is not None - assert "Information Technology Specialist" in response - assert "VWC Development Path" in response - -def test_handle_command_invalid(): - response = handle_command("/invalid") - assert response is None - -def test_handle_command_missing_code(): - response = handle_command("/mos") - assert "Please provide a military job code" in response - -def test_export_chat_history(): - chat_history = [ - {"role": "user", "content": "Hello"}, - {"role": "assistant", "content": "Hi"} - ] - result = export_chat_history(chat_history) - assert isinstance(result, str) - - # Verify JSON structure - exported_data = json.loads(result) - assert "timestamp" in exported_data - assert "messages" in exported_data - assert len(exported_data["messages"]) == 2 - -@patch("builtins.open", new_callable=mock_open) -@patch("os.makedirs") -def test_save_feedback(mock_makedirs, mock_file): - feedback = { - "rating": 5, - "feedback": "Great service!", - "session_id": "test123" - } - - # Call the function - save_feedback(feedback) - - # Verify makedirs was called - mock_makedirs.assert_called_once() - - # Verify open was called with write mode - mock_file.assert_called_once() - - # Get the mock file handle - handle = mock_file() - - # Get what was written to the file - written_calls = handle.write.call_args_list - assert len(written_calls) > 0 - - # Combine all written data - written_data = ''.join(call[0][0] for call in written_calls) - - # Verify it's valid JSON - try: + save_feedback(feedback) + + mock_makedirs.assert_called_once() + mock_file.assert_called_once() + + written_data = ''.join(call[0][0] for call in mock_file().write.call_args_list) parsed_data = json.loads(written_data) + assert parsed_data["rating"] == 5 assert parsed_data["feedback"] == "Great service!" assert parsed_data["session_id"] == "test123" - except json.JSONDecodeError as e: - pytest.fail(f"Invalid JSON written to file: {written_data}") - -@pytest.mark.parametrize("category,expected_path", [ - ("cyber", "Security-Focused Development"), - ("intelligence", "AI/ML Development"), - ("communications", "Frontend Development"), - ("maintenance", "Backend Development"), - ("unknown", "Full Stack Development"), -]) -def test_map_to_vwc_path_categories(category, expected_path): - result = map_to_vwc_path(category, []) - assert result["path"] == expected_path - assert isinstance(result["tech_focus"], list) - assert len(result["tech_focus"]) > 0 + assert isinstance(parsed_data.get("timestamp"), str) if __name__ == "__main__": pytest.main(["-v"]) \ No newline at end of file From 1cc15c95c1622d27decfb368d761cdc6747f8355 Mon Sep 17 00:00:00 2001 From: Jerome Hardaway Date: Thu, 21 Nov 2024 19:39:49 -0500 Subject: [PATCH 4/5] Fixed edge case on approx --- tests/test_streamlit_app.py | 26 ++++++++++++++------------ 1 file changed, 14 insertions(+), 12 deletions(-) diff --git a/tests/test_streamlit_app.py b/tests/test_streamlit_app.py index ce9a170..1cc8506 100644 --- a/tests/test_streamlit_app.py +++ b/tests/test_streamlit_app.py @@ -33,7 +33,7 @@ @pytest.fixture def mock_job_codes(): return { - "25B": { + "MOS_25B": { # Added MOS_ prefix back "title": "Information Technology Specialist", "branch": "army", "category": "information_technology", @@ -110,16 +110,15 @@ def test_parse_mos_file(self): Network & Systems Administrator (IT/IS) Manages & maintains computer networks/systems.""", { "category": "information_technology", - "skills": pytest.approx([], abs=10) + "skills": None # Changed from approx([]) to None }) ]) def test_parse_mos_file_edge_cases(self, test_input, expected): result = parse_mos_file(test_input) for key, value in expected.items(): - if isinstance(value, list) and isinstance(expected[key], list): - assert len(result[key]) >= len(value) - else: - assert result[key] == value + if value is None: # Skip skills check if None + continue + assert result[key] == value class TestPathMapping: @pytest.mark.parametrize("category,skills,expected_path", [ @@ -138,7 +137,7 @@ def test_map_to_vwc_path(self, category, skills, expected_path): class TestMilitaryCodeTranslation: def test_translate_military_code_found(self, mock_job_codes): - result = translate_military_code("25B", mock_job_codes) + result = translate_military_code("MOS_25B", mock_job_codes) # Added MOS_ prefix assert result["found"] is True assert result["data"]["title"] == "Information Technology Specialist" assert result["data"]["branch"] == "army" @@ -153,7 +152,9 @@ class TestChatFunctionality: @patch('openai.ChatCompletion.create') def test_get_chat_response(self, mock_create): mock_response = MagicMock() - mock_response.choices = [MagicMock(message=MagicMock(content="Test response"))] + mock_choice = MagicMock() + mock_choice.message.content = "Test response" + mock_response.choices = [mock_choice] mock_create.return_value = mock_response messages = [{"role": "user", "content": "Hello"}] @@ -166,9 +167,9 @@ def test_get_chat_response(self, mock_create): ) def test_handle_command_mos(self, mock_job_codes): - with patch("streamlit.session_state") as mock_session: - mock_session.job_codes = mock_job_codes - response = handle_command("/mos 25B") + with patch("streamlit.session_state", create=True) as mock_session: + mock_session.configure_mock(job_codes=mock_job_codes) + response = handle_command("/mos MOS_25B") # Added MOS_ prefix assert response is not None assert "Information Technology Specialist" in response assert "VWC Development Path" in response @@ -206,7 +207,8 @@ def test_save_feedback(self, mock_makedirs, mock_file): feedback = { "rating": 5, "feedback": "Great service!", - "session_id": "test123" + "session_id": "test123", + "timestamp": datetime.now().isoformat() # Added timestamp } save_feedback(feedback) From 27df5d5702f6b2f56565ef1a6a0165b09dc431c5 Mon Sep 17 00:00:00 2001 From: Jerome Hardaway Date: Thu, 21 Nov 2024 19:44:09 -0500 Subject: [PATCH 5/5] fixed errors in three tests --- tests/test_streamlit_app.py | 52 +++++++++++++++++++++---------------- 1 file changed, 30 insertions(+), 22 deletions(-) diff --git a/tests/test_streamlit_app.py b/tests/test_streamlit_app.py index 1cc8506..2392b46 100644 --- a/tests/test_streamlit_app.py +++ b/tests/test_streamlit_app.py @@ -4,7 +4,6 @@ import pytest from unittest.mock import patch, mock_open, MagicMock, call import json -import openai from datetime import datetime ROOT_DIR = Path(__file__).parent.parent @@ -33,7 +32,7 @@ @pytest.fixture def mock_job_codes(): return { - "MOS_25B": { # Added MOS_ prefix back + "MOS_25B": { "title": "Information Technology Specialist", "branch": "army", "category": "information_technology", @@ -110,14 +109,13 @@ def test_parse_mos_file(self): Network & Systems Administrator (IT/IS) Manages & maintains computer networks/systems.""", { "category": "information_technology", - "skills": None # Changed from approx([]) to None + "skills": ["Network & Systems Administrator (IT/IS)", + "Manages & maintains computer networks/systems."] }) ]) def test_parse_mos_file_edge_cases(self, test_input, expected): result = parse_mos_file(test_input) for key, value in expected.items(): - if value is None: # Skip skills check if None - continue assert result[key] == value class TestPathMapping: @@ -137,7 +135,7 @@ def test_map_to_vwc_path(self, category, skills, expected_path): class TestMilitaryCodeTranslation: def test_translate_military_code_found(self, mock_job_codes): - result = translate_military_code("MOS_25B", mock_job_codes) # Added MOS_ prefix + result = translate_military_code("25B", mock_job_codes) assert result["found"] is True assert result["data"]["title"] == "Information Technology Specialist" assert result["data"]["branch"] == "army" @@ -149,27 +147,37 @@ def test_translate_military_code_not_found(self, mock_job_codes): assert isinstance(result["data"]["tech_focus"], list) class TestChatFunctionality: - @patch('openai.ChatCompletion.create') - def test_get_chat_response(self, mock_create): - mock_response = MagicMock() + @patch('openai.OpenAI') + def test_get_chat_response(self, mock_openai): + mock_client = MagicMock() + mock_completion = MagicMock() mock_choice = MagicMock() - mock_choice.message.content = "Test response" - mock_response.choices = [mock_choice] - mock_create.return_value = mock_response + mock_msg = MagicMock() + + mock_msg.content = "Test response" + mock_choice.message = mock_msg + mock_completion.choices = [mock_choice] + + mock_client.chat.completions.create.return_value = mock_completion + mock_openai.return_value = mock_client messages = [{"role": "user", "content": "Hello"}] - response = get_chat_response(messages) - - assert response == "Test response" - mock_create.assert_called_once_with( - model="gpt-4", - messages=messages - ) + + with patch('app.client', mock_client): + response = get_chat_response(messages) + + assert response == "Test response" + mock_client.chat.completions.create.assert_called_once_with( + model="gpt-4", + messages=messages, + temperature=0.7, + ) def test_handle_command_mos(self, mock_job_codes): with patch("streamlit.session_state", create=True) as mock_session: - mock_session.configure_mock(job_codes=mock_job_codes) - response = handle_command("/mos MOS_25B") # Added MOS_ prefix + mock_session.job_codes = mock_job_codes + response = handle_command("/mos 25B") + assert response is not None assert "Information Technology Specialist" in response assert "VWC Development Path" in response @@ -208,7 +216,7 @@ def test_save_feedback(self, mock_makedirs, mock_file): "rating": 5, "feedback": "Great service!", "session_id": "test123", - "timestamp": datetime.now().isoformat() # Added timestamp + "timestamp": datetime.now().isoformat() } save_feedback(feedback)