"""Unit tests for LLM API Client."""

import pytest
from unittest.mock import Mock, patch, MagicMock
from src.utils.llm_api_client import LLMAPIClient


class TestLLMAPIClient:
    """Test cases for LLMAPIClient."""
    
    def test_init(self):
        """Test client initialization."""
        client = LLMAPIClient()
        assert client is not None
    
    def test_get_api_key_success(self):
        """Test successful API key retrieval."""
        with patch.dict('os.environ', {'GEMINI_API_KEY': 'test_key'}):
            client = LLMAPIClient()
            api_key = client._get_api_key("Gemini", "GEMINI_API_KEY")
            assert api_key == 'test_key'
    
    def test_get_api_key_missing(self):
        """Test API key retrieval when key is missing."""
        with patch.dict('os.environ', {}, clear=True):
            client = LLMAPIClient()
            with pytest.raises(ValueError, match="Missing Gemini API key"):
                client._get_api_key("Gemini", "GEMINI_API_KEY")
    
    @patch('src.utils.llm_api_client.genai')
    def test_embed_content_success(self, mock_genai):
        """Test successful embedding generation."""
        # Mock the genai module and its methods
        mock_genai.configure = Mock()
        mock_embedding_model = Mock()
        mock_result = Mock()
        mock_result.embedding = [0.1, 0.2, 0.3]
        mock_embedding_model.embed_content.return_value = mock_result
        mock_genai.get_embedding_model.return_value = mock_embedding_model
        
        with patch.dict('os.environ', {'GEMINI_API_KEY': 'test_key'}):
            client = LLMAPIClient()
            embeddings = client.embed_content("gemini-embedding-001", "test text")
            
            assert len(embeddings) == 1
            assert embeddings[0] == [0.1, 0.2, 0.3]
            mock_genai.configure.assert_called_once_with(api_key='test_key')
    
    @patch('src.utils.llm_api_client.genai')
    def test_embed_content_multiple_texts(self, mock_genai):
        """Test embedding generation for multiple texts."""
        # Mock the genai module and its methods
        mock_genai.configure = Mock()
        mock_embedding_model = Mock()
        mock_result1 = Mock()
        mock_result1.embedding = [0.1, 0.2, 0.3]
        mock_result2 = Mock()
        mock_result2.embedding = [0.4, 0.5, 0.6]
        mock_embedding_model.embed_content.side_effect = [mock_result1, mock_result2]
        mock_genai.get_embedding_model.return_value = mock_embedding_model
        
        with patch.dict('os.environ', {'GEMINI_API_KEY': 'test_key'}):
            client = LLMAPIClient()
            embeddings = client.embed_content("gemini-embedding-001", ["text1", "text2"])
            
            assert len(embeddings) == 2
            assert embeddings[0] == [0.1, 0.2, 0.3]
            assert embeddings[1] == [0.4, 0.5, 0.6]
    
    def test_embed_content_missing_api_key(self):
        """Test embedding generation when API key is missing."""
        with patch.dict('os.environ', {}, clear=True):
            client = LLMAPIClient()
            with pytest.raises(ValueError, match="Missing Gemini API key"):
                client.embed_content("gemini-embedding-001", "test text")
    
    def test_embed_content_invalid_model_name(self):
        """Test embedding generation with invalid model name."""
        with patch.dict('os.environ', {'GEMINI_API_KEY': 'test_key'}):
            client = LLMAPIClient()
            with pytest.raises(ValueError, match="model_name must be a non-empty string"):
                client.embed_content("", "test text")
    
    def test_embed_content_empty_contents(self):
        """Test embedding generation with empty contents."""
        with patch.dict('os.environ', {'GEMINI_API_KEY': 'test_key'}):
            client = LLMAPIClient()
            with pytest.raises(ValueError, match="contents must not be empty"):
                client.embed_content("gemini-embedding-001", "")
    
    @patch('src.utils.llm_api_client.genai')
    def test_embed_content_import_error(self, mock_genai):
        """Test embedding generation when genai module is not available."""
        mock_genai.configure.side_effect = ImportError("Module not found")
        
        with patch.dict('os.environ', {'GEMINI_API_KEY': 'test_key'}):
            client = LLMAPIClient()
            with pytest.raises(ImportError, match="Google Generative AI client library not installed"):
                client.embed_content("gemini-embedding-001", "test text")
    
    @patch('src.utils.llm_api_client.openai')
    def test_call_openai_returns_string(self, mock_openai):
        """Test that call_openai returns a string solution."""
        # Mock OpenAI response
        mock_response = Mock()
        mock_response.choices = [Mock()]
        mock_response.choices[0].message.content = "Test solution"
        mock_response.choices[0].finish_reason = "stop"
        mock_response.model = "gpt-4"
        mock_response.usage = Mock()
        mock_response.usage.prompt_tokens = 10
        mock_response.usage.completion_tokens = 20
        mock_response.usage.total_tokens = 30
        
        mock_client = Mock()
        mock_client.chat.completions.create.return_value = mock_response
        mock_openai.OpenAI.return_value = mock_client
        
        with patch.dict('os.environ', {'OPENAI_API_KEY': 'test_key'}):
            client = LLMAPIClient()
            solution_text = client.call_openai("test prompt", "gpt-4", 0.7)
            
            assert solution_text == "Test solution"
    
    @patch('src.utils.llm_api_client.openai')
    def test_call_llm_model_temperature_handling_o1(self, mock_openai):
        """Test that call_llm_model omits temperature for o1 model."""
        # Mock OpenAI response
        mock_response = Mock()
        mock_response.choices = [Mock()]
        mock_response.choices[0].message.content = "Test solution"
        
        mock_client = Mock()
        mock_client.chat.completions.create.return_value = mock_response
        mock_openai.OpenAI.return_value = mock_client
        
        with patch.dict('os.environ', {'OPENAI_API_KEY': 'test_key'}):
            client = LLMAPIClient()
            solution_text = client.call_llm_model("test prompt", "o1", 0.7)
            
            assert solution_text == "Test solution"
            # Verify that temperature was omitted for o1 model
            mock_client.chat.completions.create.assert_called_once_with(
                model="o1",
                messages=[{"role": "user", "content": "test prompt"}]
            )
    
    @patch('src.utils.llm_api_client.openai')
    def test_call_llm_model_temperature_handling_gpt5(self, mock_openai):
        """Test that call_llm_model omits temperature for gpt-5 models."""
        # Mock OpenAI response
        mock_response = Mock()
        mock_response.choices = [Mock()]
        mock_response.choices[0].message.content = "Test solution"
        
        mock_client = Mock()
        mock_client.chat.completions.create.return_value = mock_response
        mock_openai.OpenAI.return_value = mock_client
        
        with patch.dict('os.environ', {'OPENAI_API_KEY': 'test_key'}):
            client = LLMAPIClient()
            solution_text = client.call_llm_model("test prompt", "gpt-5", 0.7)
            
            assert solution_text == "Test solution"
            # Verify that temperature was omitted for gpt-5 model
            mock_client.chat.completions.create.assert_called_once_with(
                model="gpt-5",
                messages=[{"role": "user", "content": "test prompt"}]
            )
    
    @patch('src.utils.llm_api_client.openai')
    def test_call_llm_model_temperature_handling_other_models(self, mock_openai):
        """Test that call_llm_model includes temperature for non-o1 models."""
        # Mock OpenAI response
        mock_response = Mock()
        mock_response.choices = [Mock()]
        mock_response.choices[0].message.content = "Test solution"
        
        mock_client = Mock()
        mock_client.chat.completions.create.return_value = mock_response
        mock_openai.OpenAI.return_value = mock_client
        
        with patch.dict('os.environ', {'OPENAI_API_KEY': 'test_key'}):
            client = LLMAPIClient()
            solution_text = client.call_llm_model("test prompt", "gpt-4o", 0.7)
            
            assert solution_text == "Test solution"
            # Verify that temperature was included for non-o1 model
            mock_client.chat.completions.create.assert_called_once_with(
                model="gpt-4o",
                messages=[{"role": "user", "content": "test prompt"}],
                temperature=0.7
            )
    