"""Unit tests for main.py argument parsing functionality."""

import pytest
import sys
from unittest.mock import patch, MagicMock
from src.main import main


class TestMainArgumentParsing:
    """Test cases for main.py argument parsing."""
    
    def test_rerun_evaluation_flag_parsing(self):
        """Test that --rerun-evaluation flag is properly parsed."""
        test_args = ["--rerun-evaluation"]
        
        with patch('sys.argv', ['main.py'] + test_args):
            with patch('src.main.run_evaluation_only_mode') as mock_eval_mode:
                mock_eval_mode.return_value = None
                
                # Should call evaluation mode
                main()
                mock_eval_mode.assert_called_once()
    
    def test_normal_mode_with_required_args(self):
        """Test normal mode with all required arguments."""
        test_args = [
            "--task-name", "bridge",
            "--algorithm-name", "test_algorithm",
            "--backbone-llm-name", "gpt-4o"
        ]
        
        with patch('sys.argv', ['main.py'] + test_args):
            with patch('src.main.load_task_description') as mock_load_task:
                with patch('src.main.load_feasibility_check_points') as mock_load_feasibility:
                    with patch('src.main.load_known_solutions') as mock_load_known:
                        with patch('src.main.load_known_solutions_concept') as mock_load_known_concept:
                            with patch('src.main.load_calibration_anchors') as mock_load_calibration:
                                with patch('src.main.load_optimal_solutions') as mock_load_optimal:
                                    with patch('src.main.load_and_run_algorithm') as mock_run_algorithm:
                                        with patch('src.main.run_evaluation') as mock_run_evaluation:
                                            with patch('src.main.log_results') as mock_log_results:
                                                with patch('src.main.save_intermediate_logs') as mock_save_logs:
                                                    # Mock all the loading functions
                                                    mock_load_task.return_value = "Test task"
                                                    mock_load_feasibility.return_value = ["Check 1"]
                                                    mock_load_known.return_value = ["Solution 1"]
                                                    mock_load_known_concept.return_value = ["Concept 1"]
                                                    mock_load_calibration.return_value = ["Anchor 1"]
                                                    mock_load_optimal.return_value = (["Optimal 1"], ["Optimal concept 1"])
                                                    
                                                    # Mock algorithm execution
                                                    mock_run_algorithm.return_value = ("Test solution", [])
                                                    
                                                    # Mock evaluation
                                                    from src.data_models.evaluation_result import EvaluationResult
                                                    mock_eval_result = EvaluationResult(
                                                        original_solution_id="sol_1",
                                                        individual_solution_text="Test solution",
                                                        feasibility_score=0.8,
                                                        utility_score=0.7,
                                                        novelty_score=0.6,
                                                        creativity_score=0.7,
                                                        intermediate_log_filename="test.json",
                                                        feasibility_reasoning="Test reasoning",
                                                        utility_reasoning="Test reasoning",
                                                        novelty_theme="Test theme"
                                                    )
                                                    mock_run_evaluation.return_value = [mock_eval_result]
                                                    
                                                    # Mock intermediate logs saving
                                                    mock_save_logs.return_value = "test_log.json"
                                                    
                                                    # Should not call evaluation mode
                                                    main()
                                                    
                                                    # Verify normal workflow was called
                                                    mock_load_task.assert_called_once_with("bridge")
                                                    mock_run_algorithm.assert_called_once()
                                                    mock_run_evaluation.assert_called_once()
    
    def test_missing_required_args_in_normal_mode(self):
        """Test that missing required arguments in normal mode cause exit."""
        # Test missing task-name
        test_args = [
            "--algorithm-name", "test_algorithm",
            "--backbone-llm-name", "gpt-4o"
        ]
        
        with patch('sys.argv', ['main.py'] + test_args):
            with patch('sys.exit') as mock_exit:
                main()
                mock_exit.assert_called_once_with(1)
        
        # Test missing algorithm-name
        test_args = [
            "--task-name", "bridge",
            "--backbone-llm-name", "gpt-4o"
        ]
        
        with patch('sys.argv', ['main.py'] + test_args):
            with patch('sys.exit') as mock_exit:
                main()
                mock_exit.assert_called_once_with(1)
        
        # Test missing backbone-llm-name
        test_args = [
            "--task-name", "bridge",
            "--algorithm-name", "test_algorithm"
        ]
        
        with patch('sys.argv', ['main.py'] + test_args):
            with patch('sys.exit') as mock_exit:
                main()
                mock_exit.assert_called_once_with(1)
    
    def test_default_values_for_optional_args(self):
        """Test that optional arguments have correct default values."""
        test_args = [
            "--task-name", "bridge",
            "--algorithm-name", "test_algorithm",
            "--backbone-llm-name", "gpt-4o"
        ]
        
        with patch('sys.argv', ['main.py'] + test_args):
            with patch('src.main.load_task_description') as mock_load_task:
                with patch('src.main.load_feasibility_check_points') as mock_load_feasibility:
                    with patch('src.main.load_known_solutions') as mock_load_known:
                        with patch('src.main.load_known_solutions_concept') as mock_load_known_concept:
                            with patch('src.main.load_calibration_anchors') as mock_load_calibration:
                                with patch('src.main.load_optimal_solutions') as mock_load_optimal:
                                    with patch('src.main.load_and_run_algorithm') as mock_run_algorithm:
                                        with patch('src.main.run_evaluation') as mock_run_evaluation:
                                            with patch('src.main.log_results') as mock_log_results:
                                                with patch('src.main.save_intermediate_logs') as mock_save_logs:
                                                    # Mock all the loading functions
                                                    mock_load_task.return_value = "Test task"
                                                    mock_load_feasibility.return_value = ["Check 1"]
                                                    mock_load_known.return_value = ["Solution 1"]
                                                    mock_load_known_concept.return_value = ["Concept 1"]
                                                    mock_load_calibration.return_value = ["Anchor 1"]
                                                    mock_load_optimal.return_value = (["Optimal 1"], ["Optimal concept 1"])
                                                    
                                                    # Mock algorithm execution
                                                    mock_run_algorithm.return_value = ("Test solution", [])
                                                    
                                                    # Mock evaluation
                                                    from src.data_models.evaluation_result import EvaluationResult
                                                    mock_eval_result = EvaluationResult(
                                                        original_solution_id="sol_1",
                                                        individual_solution_text="Test solution",
                                                        feasibility_score=0.8,
                                                        utility_score=0.7,
                                                        novelty_score=0.6,
                                                        creativity_score=0.7,
                                                        intermediate_log_filename="test.json",
                                                        feasibility_reasoning="Test reasoning",
                                                        utility_reasoning="Test reasoning",
                                                        novelty_theme="Test theme"
                                                    )
                                                    mock_run_evaluation.return_value = [mock_eval_result]
                                                    
                                                    # Mock intermediate logs saving
                                                    mock_save_logs.return_value = "test_log.json"
                                                    
                                                    main()
                                                    
                                                    # Verify default values were used
                                                    call_args = mock_run_algorithm.call_args
                                                    assert call_args[1]['num_analogous_problems'] == 10
                                                    assert call_args[1]['num_solutions_per_problem'] == 5
                                                    assert call_args[1]['num_exploratory_ideas'] == 50
                                                    assert call_args[1]['num_new_rule_sets'] == 3
                                                    assert call_args[1]['num_final_solutions'] == 3
                                                    assert call_args[1]['num_solutions_combinational'] == 20
                                                    assert call_args[1]['num_thoughts_per_step'] == 5
                                                    assert call_args[1]['search_depth'] == 3
    
    def test_custom_values_for_optional_args(self):
        """Test that custom values for optional arguments are properly passed."""
        test_args = [
            "--task-name", "bridge",
            "--algorithm-name", "test_algorithm",
            "--backbone-llm-name", "gpt-4o",
            "--num-analogous-problems", "15",
            "--num-solutions-per-problem", "7",
            "--num-exploratory-ideas", "60",
            "--num-new-rule-sets", "5",
            "--num-final-solutions", "4",
            "--num-solutions-combinational", "25",
            "--num-thoughts-per-step", "8",
            "--search-depth", "4"
        ]
        
        with patch('sys.argv', ['main.py'] + test_args):
            with patch('src.main.load_task_description') as mock_load_task:
                with patch('src.main.load_feasibility_check_points') as mock_load_feasibility:
                    with patch('src.main.load_known_solutions') as mock_load_known:
                        with patch('src.main.load_known_solutions_concept') as mock_load_known_concept:
                            with patch('src.main.load_calibration_anchors') as mock_load_calibration:
                                with patch('src.main.load_optimal_solutions') as mock_load_optimal:
                                    with patch('src.main.load_and_run_algorithm') as mock_run_algorithm:
                                        with patch('src.main.run_evaluation') as mock_run_evaluation:
                                            with patch('src.main.log_results') as mock_log_results:
                                                with patch('src.main.save_intermediate_logs') as mock_save_logs:
                                                    # Mock all the loading functions
                                                    mock_load_task.return_value = "Test task"
                                                    mock_load_feasibility.return_value = ["Check 1"]
                                                    mock_load_known.return_value = ["Solution 1"]
                                                    mock_load_known_concept.return_value = ["Concept 1"]
                                                    mock_load_calibration.return_value = ["Anchor 1"]
                                                    mock_load_optimal.return_value = (["Optimal 1"], ["Optimal concept 1"])
                                                    
                                                    # Mock algorithm execution
                                                    mock_run_algorithm.return_value = ("Test solution", [])
                                                    
                                                    # Mock evaluation
                                                    from src.data_models.evaluation_result import EvaluationResult
                                                    mock_eval_result = EvaluationResult(
                                                        original_solution_id="sol_1",
                                                        individual_solution_text="Test solution",
                                                        feasibility_score=0.8,
                                                        utility_score=0.7,
                                                        novelty_score=0.6,
                                                        creativity_score=0.7,
                                                        intermediate_log_filename="test.json",
                                                        feasibility_reasoning="Test reasoning",
                                                        utility_reasoning="Test reasoning",
                                                        novelty_theme="Test theme"
                                                    )
                                                    mock_run_evaluation.return_value = [mock_eval_result]
                                                    
                                                    # Mock intermediate logs saving
                                                    mock_save_logs.return_value = "test_log.json"
                                                    
                                                    main()
                                                    
                                                    # Verify custom values were used
                                                    call_args = mock_run_algorithm.call_args
                                                    assert call_args[1]['num_analogous_problems'] == 15
                                                    assert call_args[1]['num_solutions_per_problem'] == 7
                                                    assert call_args[1]['num_exploratory_ideas'] == 60
                                                    assert call_args[1]['num_new_rule_sets'] == 5
                                                    assert call_args[1]['num_final_solutions'] == 4
                                                    assert call_args[1]['num_solutions_combinational'] == 25
                                                    assert call_args[1]['num_thoughts_per_step'] == 8
                                                    assert call_args[1]['search_depth'] == 4
