diff --git a/app.py b/app.py index 95040d8..b3dabff 100644 --- a/app.py +++ b/app.py @@ -9,6 +9,7 @@ import time import hashlib import base64 +from data.data_loader import load_military_job_codes, translate_military_code # Configure page settings and styling svg_content = ''' @@ -17,15 +18,17 @@ ''' + def get_svg_base64(svg_content): - b64 = base64.b64encode(svg_content.encode('utf-8')).decode('utf-8') - return f"data:image/svg+xml;base64,{b64}" + b64 = base64.b64encode(svg_content.encode('utf-8')).decode('utf-8') + return f"data:image/svg+xml;base64,{b64}" + # Configure the page with your brand colors st.set_page_config( - page_title="VetsAI: Vets Who Code Assistant", - page_icon=get_svg_base64(svg_content), - layout="wide" + page_title="VetsAI: Vets Who Code Assistant", + page_icon=get_svg_base64(svg_content), + layout="wide" ) # Define VWC brand colors and add custom CSS @@ -114,198 +117,6 @@ def get_svg_base64(svg_content): if not client.api_key: # Changed from openai.api_key to client.api_key raise ValueError("OpenAI API key not found in Streamlit secrets.") -def parse_mos_file(file_content: str) -> dict: - """Parse military job code text file content into a structured dictionary.""" - lines = file_content.strip().split('\n') - job_code, title, description = "", "", [] - parsing_description = False - - for line in lines: - line = line.strip() - if not line: - continue - if line.startswith("Job Code:"): - job_code = line.replace("Job Code:", "").strip() - elif line.startswith("Description:"): - parsing_description = True - elif parsing_description: - description.append(line) - - for line in description: - if line: - title = line - break - - full_text = ' '.join(description).lower() - category = "general" - category_keywords = { - "information_technology": ["technology", "computer", "network", "data", "software", "hardware", "system", "database"], - "communications": ["communications", "signal", "radio", "transmission", "telecom"], - "intelligence": ["intelligence", "analysis", "surveillance", "reconnaissance"], - "maintenance": ["maintenance", "repair", "technical", "equipment"], - "cyber": ["cyber", "security", "information assurance", "cryptographic"] - } - - for cat, keywords in category_keywords.items(): - if any(keyword in full_text for keyword in keywords): - category = cat - break - - return { - "title": title or "Military Professional", - "category": category, - "skills": [line for line in description if line and len(line) > 10] - } - -def load_military_job_codes() -> dict: - base_path = "data/employment_transitions/job_codes" - job_codes = {} - branches = { - "army": {"path": "army", "prefix": "MOS"}, - "air_force": {"path": "air_force", "prefix": "AFSC"}, - "coast_guard": {"path": "coast_guard", "prefix": "RATE"}, - "navy": {"path": "navy", "prefix": "RATE"}, - "marine_corps": {"path": "marine_corps", "prefix": "MOS"} - } - - for branch, info in branches.items(): - branch_path = os.path.join(base_path, info["path"]) - if os.path.exists(branch_path): - for file in os.listdir(branch_path): - if file.endswith('.txt'): - try: - with open(os.path.join(branch_path, file), 'r') as f: - content = f.read() - code = file.replace('.txt', '') - details = parse_mos_file(content) - vwc_mapping = map_to_vwc_path(details.get('category', ''), details.get('skills', [])) - details.update({ - 'vwc_path': vwc_mapping['path'], - 'tech_focus': vwc_mapping['tech_focus'], - 'branch': branch, - 'code_type': info['prefix'] - }) - job_codes[f"{info['prefix']}_{code}"] = details - except Exception as e: - logger.error(f"Error loading {file}: {e}") - continue - return job_codes - -def map_to_vwc_path(category: str, skills: List[str]) -> dict: - """Map military job categories and skills to VWC tech stack paths.""" - default_path = { - "path": "Full Stack Development", - "tech_focus": [ - "JavaScript/TypeScript fundamentals", - "Next.js and Tailwind for frontend", - "Python with FastAPI/Django for backend" - ] - } - - tech_paths = { - "information_technology": { - "path": "Full Stack Development", - "tech_focus": [ - "JavaScript/TypeScript with focus on system architecture", - "Next.js for complex web applications", - "Python backend services with FastAPI" - ] - }, - "cyber": { - "path": "Security-Focused Development", - "tech_focus": [ - "TypeScript for type-safe applications", - "Secure API development with FastAPI/Django", - "AI/ML for security applications" - ] - }, - "communications": { - "path": "Frontend Development", - "tech_focus": [ - "JavaScript/TypeScript specialization", - "Advanced Next.js and Tailwind", - "API integration with Python backends" - ] - }, - "intelligence": { - "path": "AI/ML Development", - "tech_focus": [ - "Python for data processing", - "ML model deployment with FastAPI", - "Next.js for ML application frontends" - ] - }, - "maintenance": { - "path": "Backend Development", - "tech_focus": [ - "Python backend development", - "API design with FastAPI/Django", - "Basic frontend with Next.js" - ] - } - } - - skill_keywords = { - "programming": "software", - "database": "data", - "network": "communications", - "security": "cyber", - "analysis": "intelligence" - } - - if category.lower() in tech_paths: - return tech_paths[category.lower()] - - for skill in skills: - skill_lower = skill.lower() - for keyword, category in skill_keywords.items(): - if keyword in skill_lower and category in tech_paths: - return tech_paths[category] - - return default_path - -def translate_military_code(code: str, job_codes: dict) -> dict: - """Translate military code to VWC development path.""" - code = code.upper().strip() - prefixes = ["MOS", "AFSC", "RATE"] - for prefix in prefixes: - if code.startswith(prefix): - code = code.replace(prefix, "").strip() - - possible_codes = [f"MOS_{code}", f"AFSC_{code}", f"RATE_{code}"] - - for possible_code in possible_codes: - if possible_code in job_codes: - job_data = job_codes[possible_code] - return { - "found": True, - "data": { - "title": job_data.get('title', 'Military Professional'), - "branch": job_data.get('branch', 'Military'), - "dev_path": job_data.get('vwc_path', 'Full Stack Development'), - "tech_focus": job_data.get('tech_focus', []), - "skills": job_data.get('skills', []) - } - } - - return { - "found": False, - "data": { - "title": "Military Professional", - "branch": "Military", - "dev_path": "Full Stack Development", - "tech_focus": [ - "Start with JavaScript/TypeScript fundamentals", - "Build projects with Next.js and Tailwind", - "Learn Python backend development with FastAPI" - ], - "skills": [ - "Leadership and team coordination", - "Problem-solving and adaptation", - "Project planning and execution" - ] - } - } def get_chat_response(messages: List[Dict]) -> str: """Get response from OpenAI chat completion.""" @@ -320,6 +131,7 @@ def get_chat_response(messages: List[Dict]) -> str: logger.error(f"OpenAI API error: {e}") raise + def export_chat_history(chat_history: List[Dict]) -> str: """Export chat history to JSON.""" export_data = { @@ -328,6 +140,7 @@ def export_chat_history(chat_history: List[Dict]) -> str: } return json.dumps(export_data, indent=2) + def save_feedback(feedback: Dict): """Save user feedback to file.""" feedback_dir = "feedback" @@ -339,37 +152,39 @@ def save_feedback(feedback: Dict): with open(feedback_file, 'w') as f: json.dump(feedback, f, indent=2) + def handle_command(command: str) -> str: """Handle special commands including MOS translation.""" parts = command.lower().split() if not parts: return None - + cmd = parts[0] if cmd in ['/mos', '/afsc', '/rate']: if len(parts) < 2: return "Please provide a military job code. Example: `/mos 25B`" - + code = parts[1] translation = translate_military_code(code, st.session_state.job_codes) if translation['found']: return ( - f"πŸŽ–οΈ **{translation['data']['title']}** ({translation['data']['branch']})\n\n" - f"πŸ’» **VWC Development Path**: {translation['data']['dev_path']}\n\n" - "πŸ”§ **Military Skills**:\n" + - "\n".join(f"- {skill}" for skill in translation['data']['skills']) + - "\n\nπŸ“š **VWC Tech Focus**:\n" + - "\n".join(f"{i+1}. {focus}" for i, focus in enumerate(translation['data']['tech_focus'])) + f"πŸŽ–οΈ **{translation['data']['title']}** ({translation['data']['branch']})\n\n" + f"πŸ’» **VWC Development Path**: {translation['data']['dev_path']}\n\n" + "πŸ”§ **Military Skills**:\n" + + "\n".join(f"- {skill}" for skill in translation['data']['skills']) + + "\n\nπŸ“š **VWC Tech Focus**:\n" + + "\n".join(f"{i + 1}. {focus}" for i, focus in enumerate(translation['data']['tech_focus'])) ) else: return ( - "I don't have that specific code in my database, but here's a recommended " - "VWC learning path based on general military experience:\n\n" + - "\n".join(f"{i+1}. {focus}" for i, focus in enumerate(translation['data']['tech_focus'])) + "I don't have that specific code in my database, but here's a recommended " + "VWC learning path based on general military experience:\n\n" + + "\n".join(f"{i + 1}. {focus}" for i, focus in enumerate(translation['data']['tech_focus'])) ) - + return None + def initialize_chat(): """Initialize the chat with a VWC-focused welcome message.""" welcome_message = { @@ -396,24 +211,25 @@ def initialize_chat(): } return [welcome_message] + def main(): """Main application function.""" st.title("πŸ‡ΊπŸ‡Έ VetsAI: Vets Who Code Assistant") - + # Initialize session if 'session_id' not in st.session_state: st.session_state.session_id = hashlib.md5(str(time.time()).encode()).hexdigest() - + if 'job_codes' not in st.session_state: try: st.session_state.job_codes = load_military_job_codes() except Exception as e: logger.error(f"Error loading job codes: {e}") st.session_state.job_codes = {} - + if 'messages' not in st.session_state: st.session_state.messages = initialize_chat() - + # Sidebar with VWC tech stack resources with st.sidebar: st.markdown(""" @@ -481,7 +297,7 @@ def main(): "content": ( "You are a specialized AI assistant for Vets Who Code members, designed to provide clear, practical technical guidance " "to veterans transitioning into software development careers.\n\n" - + "CORE TECH STACK:\n" "- Frontend: JavaScript, TypeScript, React, Next.js\n" "- Styling: CSS, Tailwind CSS\n" @@ -490,34 +306,34 @@ def main(): "- Advanced: AI/ML fundamentals\n" "- Development Tools: Git, GitHub, VS Code\n" "- Testing: Jest, Pytest\n\n" - + "CAREER TRANSITION GUIDANCE:\n" "1. Resume Development:\n" " - Technical Skills: Programming Languages, Frameworks, Tools, Cloud, Testing\n" " - Military Experience Translation: Leadership, Problem-solving, Team Collaboration\n\n" - + "2. Portfolio Development:\n" " - Clean code and documentation\n" " - Version control and API integration\n" " - Responsive design and performance\n" " - Testing and TypeScript implementation\n" " - Security and accessibility standards\n\n" - + "LEARNING PATHS:\n" "1. Fundamentals: HTML, CSS, JavaScript, Git\n" "2. Intermediate: TypeScript, React, Python\n" "3. Advanced: Next.js, FastAPI, Streamlit, AI/ML\n\n" - + "PROJECT FOCUS:\n" "1. Portfolio Projects: Personal website, APIs, Data visualization\n" "2. Technical Skills: Code quality, Testing, Security, Performance\n" "3. Career Materials: GitHub profile, Technical blog, Documentation\n\n" - + "Remember: Provide practical guidance for building technical skills and transitioning to software development careers. " "Focus on concrete examples and best practices." ) }) - + response = get_chat_response(messages) st.markdown(response) st.session_state.messages.append({ @@ -547,5 +363,6 @@ def main(): save_feedback(feedback) st.success("Thank you for your feedback!") + if __name__ == "__main__": - main() \ No newline at end of file + main() diff --git a/data/data_loader.py b/data/data_loader.py new file mode 100644 index 0000000..ec4fb5b --- /dev/null +++ b/data/data_loader.py @@ -0,0 +1,214 @@ +import os +from typing import List +import logging +from pathlib import Path + +# Configure logging +ROOT_DIR = Path(__file__).parent.parent +logging.basicConfig( + level=logging.INFO, + format='%(asctime)s - %(name)s - %(levelname)s - %(message)s', + handlers=[ + logging.FileHandler(ROOT_DIR / 'vetsai.log'), + logging.StreamHandler() + ] +) +logger = logging.getLogger(__name__) + + +def parse_mos_file(file_content: str) -> dict: + """Parse military job code text file content into a structured dictionary.""" + lines = file_content.strip().split('\n') + job_code, title, description = "", "", [] + parsing_description = False + + for line in lines: + line = line.strip() + if not line: + continue + if line.startswith("Job Code:"): + job_code = line.replace("Job Code:", "").strip() + elif line.startswith("Description:"): + parsing_description = True + elif parsing_description: + description.append(line) + + for line in description: + if line: + title = line + break + + full_text = ' '.join(description).lower() + category = "general" + category_keywords = { + "information_technology": ["technology", "computer", "network", "data", "software", "hardware", "system", + "database"], + "communications": ["communications", "signal", "radio", "transmission", "telecom"], + "intelligence": ["intelligence", "analysis", "surveillance", "reconnaissance"], + "maintenance": ["maintenance", "repair", "technical", "equipment"], + "cyber": ["cyber", "security", "information assurance", "cryptographic"] + } + + for cat, keywords in category_keywords.items(): + if any(keyword in full_text for keyword in keywords): + category = cat + break + + return { + "title": title or "Military Professional", + "category": category, + "skills": [line for line in description if line and len(line) > 10] + } + + +def load_military_job_codes() -> dict: + base_path = "data/employment_transitions/job_codes" + job_codes = {} + branches = { + "army": {"path": "army", "prefix": "MOS"}, + "air_force": {"path": "air_force", "prefix": "AFSC"}, + "coast_guard": {"path": "coast_guard", "prefix": "RATE"}, + "navy": {"path": "navy", "prefix": "RATE"}, + "marine_corps": {"path": "marine_corps", "prefix": "MOS"} + } + + for branch, info in branches.items(): + branch_path = os.path.join(base_path, info["path"]) + if os.path.exists(branch_path): + for file in os.listdir(branch_path): + if file.endswith('.txt'): + try: + with open(os.path.join(branch_path, file), 'r') as f: + content = f.read() + code = file.replace('.txt', '') + details = parse_mos_file(content) + vwc_mapping = map_to_vwc_path(details.get('category', ''), details.get('skills', [])) + details.update({ + 'vwc_path': vwc_mapping['path'], + 'tech_focus': vwc_mapping['tech_focus'], + 'branch': branch, + 'code_type': info['prefix'] + }) + job_codes[f"{info['prefix']}_{code}"] = details + except Exception as e: + logger.error(f"Error loading {file}: {e}") + continue + return job_codes + + +def map_to_vwc_path(category: str, skills: List[str]) -> dict: + """Map military job categories and skills to VWC tech stack paths.""" + default_path = { + "path": "Full Stack Development", + "tech_focus": [ + "JavaScript/TypeScript fundamentals", + "Next.js and Tailwind for frontend", + "Python with FastAPI/Django for backend" + ] + } + + tech_paths = { + "information_technology": { + "path": "Full Stack Development", + "tech_focus": [ + "JavaScript/TypeScript with focus on system architecture", + "Next.js for complex web applications", + "Python backend services with FastAPI" + ] + }, + "cyber": { + "path": "Security-Focused Development", + "tech_focus": [ + "TypeScript for type-safe applications", + "Secure API development with FastAPI/Django", + "AI/ML for security applications" + ] + }, + "communications": { + "path": "Frontend Development", + "tech_focus": [ + "JavaScript/TypeScript specialization", + "Advanced Next.js and Tailwind", + "API integration with Python backends" + ] + }, + "intelligence": { + "path": "AI/ML Development", + "tech_focus": [ + "Python for data processing", + "ML model deployment with FastAPI", + "Next.js for ML application frontends" + ] + }, + "maintenance": { + "path": "Backend Development", + "tech_focus": [ + "Python backend development", + "API design with FastAPI/Django", + "Basic frontend with Next.js" + ] + } + } + + skill_keywords = { + "programming": "software", + "database": "data", + "network": "communications", + "security": "cyber", + "analysis": "intelligence" + } + + if category.lower() in tech_paths: + return tech_paths[category.lower()] + + for skill in skills: + skill_lower = skill.lower() + for keyword, category in skill_keywords.items(): + if keyword in skill_lower and category in tech_paths: + return tech_paths[category] + + return default_path + + +def translate_military_code(code: str, job_codes: dict) -> dict: + """Translate military code to VWC development path.""" + code = code.upper().strip() + prefixes = ["MOS", "AFSC", "RATE"] + for prefix in prefixes: + if code.startswith(prefix): + code = code.replace(prefix, "").strip() + + possible_codes = [f"MOS_{code}", f"AFSC_{code}", f"RATE_{code}"] + + for possible_code in possible_codes: + if possible_code in job_codes: + job_data = job_codes[possible_code] + return { + "found": True, + "data": { + "title": job_data.get('title', 'Military Professional'), + "branch": job_data.get('branch', 'Military'), + "dev_path": job_data.get('vwc_path', 'Full Stack Development'), + "tech_focus": job_data.get('tech_focus', []), + "skills": job_data.get('skills', []) + } + } + + return { + "found": False, + "data": { + "title": "Military Professional", + "branch": "Military", + "dev_path": "Full Stack Development", + "tech_focus": [ + "Start with JavaScript/TypeScript fundamentals", + "Build projects with Next.js and Tailwind", + "Learn Python backend development with FastAPI" + ], + "skills": [ + "Leadership and team coordination", + "Problem-solving and adaptation", + "Project planning and execution" + ] + } + } diff --git a/data/tests/test_data_loader.py b/data/tests/test_data_loader.py new file mode 100644 index 0000000..924a096 --- /dev/null +++ b/data/tests/test_data_loader.py @@ -0,0 +1,166 @@ +import pytest +from unittest.mock import patch, mock_open + +from data.data_loader import ( + load_military_job_codes, + map_to_vwc_path, + translate_military_code, + parse_mos_file +) + +# Sample text content +SAMPLE_MOS_TEXT = """ +Job Code: 25B + +Description: +Manages or supervises a specific automated system or node in a data or communications network. + +Manages or supervises a specific automated system or node in a data or communications network supporting tactical, theater, strategic or base operations; provides detailed technical direction and advice to commanders, staffs and other Command, Control, and Communications (C3) users at all echelons on the installation, operation and maintenance of distributed operating and data base systems, teleprocessing systems, and data communications supporting Battlefield Automated Systems (BAS); requires the practical application of automation theory to the design, implementation and successful interoperation of hardware and software for automated telecommunications and teleprocessing systems. +""" + + +@pytest.fixture +def mock_job_codes(): + return { + "MOS_25B": { + "title": "Information Technology Specialist", + "branch": "army", + "category": "information_technology", + "skills": ["Network administration", "System maintenance"], + "vwc_path": "Full Stack Development", + "tech_focus": [ + "JavaScript/TypeScript with focus on system architecture", + "Next.js for complex web applications", + "Python backend services with FastAPI" + ], + "code_type": "MOS" + } + } + + +@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 + + def mock_exists(path): + return True + + def mock_listdir(path): + if path.endswith("job_codes"): + 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): + + job_codes = load_military_job_codes() + + # Basic validations + assert isinstance(job_codes, dict) + assert len(job_codes) > 0 + + # 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) + + # 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) + + +@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 + + +if __name__ == "__main__": + pytest.main(["-v"]) diff --git a/run.sh b/run.sh old mode 100644 new mode 100755 diff --git a/tests/test_streamlit_app.py b/tests/test_streamlit_app.py index 992cffe..f02dc03 100644 --- a/tests/test_streamlit_app.py +++ b/tests/test_streamlit_app.py @@ -12,146 +12,12 @@ sys.path.append(str(ROOT_DIR)) from app import ( - load_military_job_codes, - map_to_vwc_path, - translate_military_code, get_chat_response, handle_command, export_chat_history, - save_feedback, - parse_mos_file + save_feedback ) -# Sample text content -SAMPLE_MOS_TEXT = """ -Job Code: 25B - -Description: -Manages or supervises a specific automated system or node in a data or communications network. - -Manages or supervises a specific automated system or node in a data or communications network supporting tactical, theater, strategic or base operations; provides detailed technical direction and advice to commanders, staffs and other Command, Control, and Communications (C3) users at all echelons on the installation, operation and maintenance of distributed operating and data base systems, teleprocessing systems, and data communications supporting Battlefield Automated Systems (BAS); requires the practical application of automation theory to the design, implementation and successful interoperation of hardware and software for automated telecommunications and teleprocessing systems. -""" - -@pytest.fixture -def mock_job_codes(): - return { - "MOS_25B": { - "title": "Information Technology Specialist", - "branch": "army", - "category": "information_technology", - "skills": ["Network administration", "System maintenance"], - "vwc_path": "Full Stack Development", - "tech_focus": [ - "JavaScript/TypeScript with focus on system architecture", - "Next.js for complex web applications", - "Python backend services with FastAPI" - ], - "code_type": "MOS" - } - } - -@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 - - def mock_exists(path): - return True - - def mock_listdir(path): - if path.endswith("job_codes"): - 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): - - job_codes = load_military_job_codes() - - # Basic validations - assert isinstance(job_codes, dict) - assert len(job_codes) > 0 - - # 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) - - # 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("openai.chat.completions.create") def test_get_chat_response(mock_create): @@ -165,6 +31,7 @@ def test_get_chat_response(mock_create): assert response == "Test response" mock_create.assert_called_once() + def test_handle_command_mos(mock_job_codes): with patch("streamlit.session_state") as mock_session: mock_session.job_codes = mock_job_codes @@ -173,14 +40,17 @@ def test_handle_command_mos(mock_job_codes): 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"}, @@ -188,13 +58,14 @@ def test_export_chat_history(): ] 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): @@ -203,26 +74,26 @@ def test_save_feedback(mock_makedirs, mock_file): "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: parsed_data = json.loads(written_data) @@ -232,18 +103,6 @@ def test_save_feedback(mock_makedirs, mock_file): 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 if __name__ == "__main__": - pytest.main(["-v"]) \ No newline at end of file + pytest.main(["-v"])