"""Tests for the OpenAI integration.""" from unittest.mock import AsyncMock, mock_open, patch import httpx from openai import ( APIConnectionError, AuthenticationError, BadRequestError, RateLimitError, ) from openai.types.image import Image from openai.types.images_response import ImagesResponse from openai.types.responses import Response, ResponseOutputMessage, ResponseOutputText import pytest from homeassistant.components.openai_conversation import CONF_FILENAMES from homeassistant.core import HomeAssistant from homeassistant.exceptions import HomeAssistantError, ServiceValidationError from homeassistant.setup import async_setup_component from tests.common import MockConfigEntry @pytest.mark.parametrize( ("service_data", "expected_args"), [ ( {"prompt": "Picture of a dog"}, { "prompt": "Picture of a dog", "size": "1024x1024", "quality": "standard", "style": "vivid", }, ), ( { "prompt": "Picture of a dog", "size": "1024x1792", "quality": "hd", "style": "vivid", }, { "prompt": "Picture of a dog", "size": "1024x1792", "quality": "hd", "style": "vivid", }, ), ( { "prompt": "Picture of a dog", "size": "1792x1024", "quality": "standard", "style": "natural", }, { "prompt": "Picture of a dog", "size": "1792x1024", "quality": "standard", "style": "natural", }, ), ], ) async def test_generate_image_service( hass: HomeAssistant, mock_config_entry: MockConfigEntry, mock_init_component, service_data, expected_args, ) -> None: """Test generate image service.""" service_data["config_entry"] = mock_config_entry.entry_id expected_args["model"] = "dall-e-3" expected_args["response_format"] = "url" expected_args["n"] = 1 with patch( "openai.resources.images.AsyncImages.generate", return_value=ImagesResponse( created=1700000000, data=[ Image( b64_json=None, revised_prompt="A clear and detailed picture of an ordinary canine", url="A", ) ], ), ) as mock_create: response = await hass.services.async_call( "openai_conversation", "generate_image", service_data, blocking=True, return_response=True, ) assert response == { "url": "A", "revised_prompt": "A clear and detailed picture of an ordinary canine", } assert len(mock_create.mock_calls) == 1 assert mock_create.mock_calls[0][2] == expected_args @pytest.mark.usefixtures("mock_init_component") async def test_generate_image_service_error( hass: HomeAssistant, mock_config_entry: MockConfigEntry, ) -> None: """Test generate image service handles errors.""" with ( patch( "openai.resources.images.AsyncImages.generate", side_effect=RateLimitError( response=httpx.Response( status_code=500, request=httpx.Request(method="GET", url="") ), body=None, message="Reason", ), ), pytest.raises(HomeAssistantError, match="Error generating image: Reason"), ): await hass.services.async_call( "openai_conversation", "generate_image", { "config_entry": mock_config_entry.entry_id, "prompt": "Image of an epic fail", }, blocking=True, return_response=True, ) with ( patch( "openai.resources.images.AsyncImages.generate", return_value=ImagesResponse( created=1700000000, data=[ Image( b64_json=None, revised_prompt=None, url=None, ) ], ), ), pytest.raises(HomeAssistantError, match="No image returned"), ): await hass.services.async_call( "openai_conversation", "generate_image", { "config_entry": mock_config_entry.entry_id, "prompt": "Image of an epic fail", }, blocking=True, return_response=True, ) @pytest.mark.usefixtures("mock_init_component") async def test_generate_content_service_with_image_not_allowed_path( hass: HomeAssistant, mock_config_entry: MockConfigEntry, ) -> None: """Test generate content service with an image in a not allowed path.""" with ( patch("pathlib.Path.exists", return_value=True), patch.object(hass.config, "is_allowed_path", return_value=False), pytest.raises( HomeAssistantError, match=( "Cannot read `doorbell_snapshot.jpg`, no access to path; " "`allowlist_external_dirs` may need to be adjusted in " "`configuration.yaml`" ), ), ): await hass.services.async_call( "openai_conversation", "generate_content", { "config_entry": mock_config_entry.entry_id, "prompt": "Describe this image from my doorbell camera", "filenames": "doorbell_snapshot.jpg", }, blocking=True, return_response=True, ) @pytest.mark.parametrize( ("service_name", "error"), [ ("generate_image", "Invalid config entry provided. Got invalid_entry"), ("generate_content", "Invalid config entry provided. Got invalid_entry"), ], ) async def test_invalid_config_entry( hass: HomeAssistant, mock_config_entry: MockConfigEntry, mock_init_component, service_name: str, error: str, ) -> None: """Assert exception when invalid config entry is provided.""" service_data = { "prompt": "Picture of a dog", "config_entry": "invalid_entry", } with pytest.raises(ServiceValidationError, match=error): await hass.services.async_call( "openai_conversation", service_name, service_data, blocking=True, return_response=True, ) @pytest.mark.parametrize( ("side_effect", "error"), [ ( APIConnectionError(request=httpx.Request(method="GET", url="test")), "Connection error", ), ( AuthenticationError( response=httpx.Response( status_code=500, request=httpx.Request(method="GET", url="test") ), body=None, message="", ), "Invalid API key", ), ( BadRequestError( response=httpx.Response( status_code=500, request=httpx.Request(method="GET", url="test") ), body=None, message="", ), "openai_conversation integration not ready yet", ), ], ) async def test_init_error( hass: HomeAssistant, mock_config_entry: MockConfigEntry, caplog: pytest.LogCaptureFixture, side_effect, error, ) -> None: """Test initialization errors.""" with patch( "openai.resources.models.AsyncModels.list", side_effect=side_effect, ): assert await async_setup_component(hass, "openai_conversation", {}) await hass.async_block_till_done() assert error in caplog.text @pytest.mark.parametrize( ("service_data", "expected_args", "number_of_files"), [ ( {"prompt": "Picture of a dog", "filenames": []}, { "input": [ { "content": [ { "type": "input_text", "text": "Picture of a dog", }, ], }, ], }, 0, ), ( {"prompt": "Picture of a dog", "filenames": ["/a/b/c.pdf"]}, { "input": [ { "content": [ { "type": "input_text", "text": "Picture of a dog", }, { "type": "input_file", "file_data": "data:application/pdf;base64,BASE64IMAGE1", "filename": "/a/b/c.pdf", }, ], }, ], }, 1, ), ( {"prompt": "Picture of a dog", "filenames": ["/a/b/c.jpg"]}, { "input": [ { "content": [ { "type": "input_text", "text": "Picture of a dog", }, { "type": "input_image", "image_url": "", "detail": "auto", "file_id": "/a/b/c.jpg", }, ], }, ], }, 1, ), ( { "prompt": "Picture of a dog", "filenames": ["/a/b/c.jpg", "d/e/f.jpg"], }, { "input": [ { "content": [ { "type": "input_text", "text": "Picture of a dog", }, { "type": "input_image", "image_url": "", "detail": "auto", "file_id": "/a/b/c.jpg", }, { "type": "input_image", "image_url": "", "detail": "auto", "file_id": "d/e/f.jpg", }, ], }, ], }, 2, ), ], ) async def test_generate_content_service( hass: HomeAssistant, mock_config_entry: MockConfigEntry, mock_init_component, service_data, expected_args, number_of_files, ) -> None: """Test generate content service.""" service_data["config_entry"] = mock_config_entry.entry_id expected_args["model"] = "gpt-4o-mini" expected_args["max_output_tokens"] = 150 expected_args["top_p"] = 1.0 expected_args["temperature"] = 1.0 expected_args["user"] = None expected_args["store"] = False expected_args["input"][0]["type"] = "message" expected_args["input"][0]["role"] = "user" with ( patch( "openai.resources.responses.AsyncResponses.create", new_callable=AsyncMock, ) as mock_create, patch( "base64.b64encode", side_effect=[b"BASE64IMAGE1", b"BASE64IMAGE2"] ) as mock_b64encode, patch("builtins.open", mock_open(read_data="ABC")) as mock_file, patch("pathlib.Path.exists", return_value=True), patch.object(hass.config, "is_allowed_path", return_value=True), ): mock_create.return_value = Response( object="response", id="resp_A", created_at=1700000000, model="gpt-4o-mini", parallel_tool_calls=True, tool_choice="auto", tools=[], output=[ ResponseOutputMessage( type="message", id="msg_A", content=[ ResponseOutputText( type="output_text", text="This is the response", annotations=[], ) ], role="assistant", status="completed", ) ], ) response = await hass.services.async_call( "openai_conversation", "generate_content", service_data, blocking=True, return_response=True, ) assert response == {"text": "This is the response"} assert len(mock_create.mock_calls) == 1 assert mock_create.mock_calls[0][2] == expected_args assert mock_b64encode.call_count == number_of_files for idx, file in enumerate(service_data[CONF_FILENAMES]): assert mock_file.call_args_list[idx][0][0] == file @pytest.mark.parametrize( ( "service_data", "error", "number_of_files", "exists_side_effect", "is_allowed_side_effect", ), [ ( {"prompt": "Picture of a dog", "filenames": ["/a/b/c.jpg"]}, "`/a/b/c.jpg` does not exist", 0, [False], [True], ), ( { "prompt": "Picture of a dog", "filenames": ["/a/b/c.jpg", "d/e/f.png"], }, "Cannot read `d/e/f.png`, no access to path; `allowlist_external_dirs` may need to be adjusted in `configuration.yaml`", 1, [True, True], [True, False], ), ( {"prompt": "Not a picture of a dog", "filenames": ["/a/b/c.mov"]}, "Only images and PDF are supported by the OpenAI API,`/a/b/c.mov` is not an image file or PDF", 1, [True], [True], ), ], ) async def test_generate_content_service_invalid( hass: HomeAssistant, mock_config_entry: MockConfigEntry, mock_init_component, service_data, error, number_of_files, exists_side_effect, is_allowed_side_effect, ) -> None: """Test generate content service.""" service_data["config_entry"] = mock_config_entry.entry_id with ( patch( "openai.resources.responses.AsyncResponses.create", new_callable=AsyncMock, ) as mock_create, patch( "base64.b64encode", side_effect=[b"BASE64IMAGE1", b"BASE64IMAGE2"] ) as mock_b64encode, patch("builtins.open", mock_open(read_data="ABC")), patch("pathlib.Path.exists", side_effect=exists_side_effect), patch.object( hass.config, "is_allowed_path", side_effect=is_allowed_side_effect ), ): with pytest.raises(HomeAssistantError, match=error): await hass.services.async_call( "openai_conversation", "generate_content", service_data, blocking=True, return_response=True, ) assert len(mock_create.mock_calls) == 0 assert mock_b64encode.call_count == number_of_files @pytest.mark.usefixtures("mock_init_component") async def test_generate_content_service_error( hass: HomeAssistant, mock_config_entry: MockConfigEntry, ) -> None: """Test generate content service handles errors.""" with ( patch( "openai.resources.responses.AsyncResponses.create", side_effect=RateLimitError( response=httpx.Response( status_code=417, request=httpx.Request(method="GET", url="") ), body=None, message="Reason", ), ), pytest.raises(HomeAssistantError, match="Error generating content: Reason"), ): await hass.services.async_call( "openai_conversation", "generate_content", { "config_entry": mock_config_entry.entry_id, "prompt": "Image of an epic fail", }, blocking=True, return_response=True, )